自定义centos系统镜像

1,271 阅读7分钟

实际工作中,总是会面对各种各样的需求,除了开发需求之外,还可能会有操作系统方面的自定义方面的需求,如果使用的是 centos 系统,需要对 centos 系统做一些特殊的自定义的操作,那么这篇文章希望能对你有所帮助。

欢迎关注的我的个人博客
blog.wuzhenyu.com.cnblog.csdn.net/honglicu123

tutorial

自定义 centos 镜像的思路,利用的也是官方已有的镜像。从官网下载最小化安装的镜像,以此作为基础,在这个基础上可以添加或者删除该镜像中预装的 rpm 包,来达到自定义镜像的目的。

挂载镜像

将iso镜像添加到虚拟机上,挂载后的设备为 /dev/sr1,将其挂载到 /media 目录中

mount /dev/sr1 /media
lsblk

安装制作发行版的工具

yum -y install createrepo mkisofs isomd5sum rsync

同步光盘文件到制作ISO的目录

#同步/media下的文件到/ISO路径下,除了Packages和repodata文件夹
/usr/bin/rsync -a --exclude=Packages/ --exclude=repodata/ /media/ /ISO/
# 在/ISO/文件夹下新建Packages和repodata文件夹
mkdir -p /ISO/{Packages,repodata}

拷贝 rpm

生成install.log

rpm -qa >> /root/install.log

拷贝已安装rpm

awk '{print $0}'  /root/install.log |xargs -i cp /media/Packages/{}.rpm /ISO/Packages/

配置yum下载指定软件包列表(rpms_list.txt)的所有依赖包

将需要安装的包及其以来写入 rpms_list.txt

#!/bin/sh   
for line in `cat /root/rpms_list.txt`
    do
       echo "download file>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>and it's denpendies"$line
     yum install -y --downloadonly --downloaddir=/root/test/ $line
#或者先安装yum -y install yum-utils再yumdownloader $line
    done
cp /root/test/* /ISO/Packages/

rpm 相关操作

yum list | grep XXX # 搜索
yum deplist XXX # 查看依赖
yum install --downloadonly --downloaddir=/packages XXX # 仅仅下载,不安装
yumdownloader --resolve --destdir /packages XXX # 下载

安装包的所有依赖必须齐全,否则安装系统时,会出错

修改 isolinux.cfg 文件

label linux
menu label ^Install CentOS 7
kernel vmlinuz
#这样是不自动安装
append initrd=initrd.img inst.stage2=hd:LABEL=CentOS7 quiet
#这样配置会自动安装,通过 kickstart 自动化脚本安装
append initrd=initrd.img inst.stage2=hd:LABEL=CentOS7 inst.ks=hd:LABEL=CentOS7:/isolinux/ks.cfg quiet

inst.ks为ks.cfg文件位置; inst.stage2为安装介质位置,hd:LABEL为介质标签,例如CentOS7。这个和后续生成ISO镜像文件的命令genisoimage的参数-V有关。 最好是把所有hd:LABEL后面的值都替换为一个后面会用到的字符串,如CentOS7

修改自动化安装配置文件

第一次手动安装系统完成后,安装程序中的所有选择都会保存到名为 anaconda-ks.cfg 的文件中,该文件位于所安装系统的 /root/ 目录下。然后可以复制这个文件,根据需要进行修改,并在以后的安装中使用得到的配置文件。

在%packages与%end中间加入需要自定义安装的包组,如下所示

#安装所需要的软件包,把自定义的安装包添加进去即可在安装系统时自定义的软件
%packages
@^web-server-environment
@base
@core
@java-platform
@python-web
@web-server
kexec-tools
@{自定义包组的groupid}
%end

自定义包组的 groupid 就是定义在 /repodata/comps.xml 文件中的 group,详细内容看下一节的描述

制作修改comps文件

进入/ISO目录,将*-x86_64-comps.xml文件拷贝到/ISO/repodata路径下,并重命名成comps.xml。

cp /media/repodata/*-x86_64-comps.xml  /ISO/repodata/comps.xml

编辑comps文件

添加定制rpm包

comps.xml 文件中添加 rpm 安装包,如下所示

<group>
    <id>{自定义包组的groupid}</id>
    <name>My Dependencies</name>
    <default>true</default>
    <uservisible>true</uservisible> # 这个值为true,在安装界面才能看到,否则不可见
    <packagelist>
        <packagereq type="default">python-webob</packagereq>
        ......
    </packagelist>
</group>

在 packagereq 这个节点中添加所需要的 rpm,添加的rpm依赖的所有相关的rpm包都必须添加进来

根据 rpms_list.txt 拼接packagereq

#!/bin/sh
for line in `cat rpms_list.txt`
do
    echo " <packagereq type=\"default\">"$line"</packagereq>" 
done

在指定的环境中添加定制的groupid

<environment>
        <id>web-server-environment</id>
        <name>Basic Web Server</name>
        <description>Server for serving static and dynamic internet content.</description>
        <display_order>30</display_order>
        <grouplist>
          <groupid>base</groupid>
          <groupid>core</groupid>
          <groupid>web-server</groupid>
          <groupid>{自定义包组的groupid}</groupid>
        </grouplist>
</environment>

如果不在指定环境中添加,也就是 environment 的节点中添加新定义的 groupid,在系统安装界面就会提示 group 未定义的错误。

生成新的 repodata 数据

切换到 ISO/ 路径下,由 comps.xml 生成 repodata 包。注意当有新包加入,或者更新comps.xml文件,均需要重新生成repodata文件夹

createrepo -g repodata/comps.xml ./

制作ISO镜像

注意参数中的-V,和上面的isolinux.cfg文件有关

cd /ISO
genisoimage -joliet-long -V CentOS7 -o CentOS-7-2.iso -b isolinux/isolinux.bin -c isolinux/boot.cat -no-emul-boot -boot-load-size 4 -boot-info-table -R -J -v -cache-inodes -T -eltorito-alt-boot -e images/efiboot.img -no-emul-boot /ISO

选项 -V CentOS7 就是上面提及的 inst.stage2=hd:LABEL=CentOS7,如果是带有空格的,那就加上双引号,比如 -V "CentOS 7 x86_64"

制作镜像MD5值

implantisomd5 /ISO/CentOS-7-2.iso

生成的 iso 镜像文件就可以直接用于安装系统了

注意: 在安装系统时,如果一键自动化安装出现了错误,退出,采用图形界面安装的方式,可以点击详细内容,查看具体的错误提示信息,进行针对的修复。

制作镜像过程中可能出现的错误

1

postfix 安装时,出现错误
2:postfix-2.10.1-6.el7.x86_64 has missing requires of libmysqlclient.so.18()(64bit)
2:postfix-2.10.1-6.el7.x86_64 has missing requires of libmysqlclient.so.18(libmysqlclient_18)(64bit)

缺少Percona-XtraDB-Cluster-shared-55-5.5.37-25.10.756.el6.x86_64.rpm这个包

2

net-snmp-agent-libs 依赖 libmysqlclient.so.18,但是实际安装的时候,并不需要这个库

实际安装依赖
net-snmp-libs
net-snmp-agent-libs
lm_sensor-libs

问题1与问题2,都是因为 mysql 库与 mariadb 库的冲突导致。mysql 库 与 mariadb 库,名称都是 mysqlclient.so,这个库导致的冲突造成的。

3 系统安装完成后,如何挂在光盘并copy数据

kickstart 安装完系统后,将 iso 挂载,同时拷贝出里面的文件,这需要利用到 kickstart 的 postscript

系统安装完成之后,光盘挂载在/tmp/cdrom,它是处于non-chroot环境下的。

%post
# copy files from the rhel.iso image to system                                                                                      
if [ ! -b /tmp/cdrom ];then
  echo "make cdrom node using mknod" >> /root/ins.log
  # get the major number
  major=$(lsblk |grep rom | awk '{print $2}' |awk -F : '{print $1}')
  # get the minor number
  minor=$(lsblk |grep rom | awk '{print $2}' |awk -F : '{print $2}')
  echo "major=${major}  minor=${minor}" >> /root/ins.log
  # make sure we have what we need; create device node if so
  [ -n "$major" -a -n "$minor" ] && mknod /tmp/cdrom b ${major} ${minor}
fi

echo "mount the iso to /mnt/source" >> /root/ins.log
[ ! -d /mnt/source ] && mkdir -p /mnt/source
mount -t iso9660 -o ro /tmp/cdrom /mnt/source

# copy ...
%end

4 anaconda log path

不管怎样,%post %end 脚本,查看脚本有没有起作用系统,在

/var/log/anaconda/

中查看 anaconda 的安装日志

5 partition

kickstart 分区

# Partition clearing information
zerombr
clearpart --all --initlabel
#auto part --type=lvm
part /boot --fstype ext4 --size 256 # 创建 /boot 分区
part swap --recommended # 创建 swap 分区,推荐大小一般是内存的2倍大小
part pv.01 --size 19456 --grow # 创建名为 pv.01 的分区
volgroup centos pv.01 # 创建 volumn group ,取名为 centos
logvol / --vgname=centos --size 10240 --name=lv_root # 创建逻辑卷 /, 名为 lv_root
logvol /home --vgname=centos --size 9126 --name=lv_home

比如下面这个分区情况,

DEVICE        MOUNTPOINT       SIZE
 /dev/sda       (total)         500,000 MB
 /dev/sda1       /boot/           128 MB
 /dev/sda2       /            20,000 MB
 /dev/sda3       /var/log/        20,000 MB
 /dev/sda5       /home/         400,000 MB
 /dev/sda6       /opt/          51,680 MB
 /dev/sda7       swap           8,192 MB

可以在 kickstart 的 ks.cfg 配置中按照如下方式编写

 part /boot  --asprimary --size=128
 part /    --asprimary --size=20000
 part /var/log --asprimary --size=20000
 part /home         --size=400000
 part /opt         --size=51680
 part swap         --size=8192

kickstart 中的分区方式,也是支持软raid分区的,就是所谓的磁盘阵列,比如下面这种方式,就是在系统安装时,进行了 raid1 的配置,同时进行分区

clearpart --drives=hda,hdc
zerombr
# Raid 1 IDE config
part raid.11 --size 1000 --asprimary --ondrive=hda
part raid.12 --size 1000 --asprimary --ondrive=hda
part raid.13 --size 2000 --asprimary --ondrive=hda
part raid.14 --size 8000 --ondrive=hda
part raid.15 --size 16384 --grow --ondrive=hda
part raid.21 --size 1000 --asprimary --ondrive=hdc
part raid.22 --size 1000 --asprimary --ondrive=hdc
part raid.23 --size 2000 --asprimary --ondrive=hdc
part raid.24 --size 8000 --ondrive=hdc
part raid.25 --size 16384 --grow --ondrive=hdc

# You can add --spares=x
raid / --fstype xfs --device root --level=RAID1 raid.11 raid.21
raid /safe --fstype xfs --device safe --level=RAID1 raid.12 raid.22
raid swap --fstype swap --device swap --level=RAID1 raid.13 raid.23
raid /usr --fstype xfs --device usr --level=RAID1 raid.14 raid.24
raid pv.01 --fstype xfs --device pv.01 --level=RAID1 raid.15 raid.25

# LVM configuration so that we can resize /var and /usr/local later
volgroup sysvg pv.01
logvol /var --vgname=sysvg --size=8000 --name=var
logvol /usr/local --vgname=sysvg --size=1 --grow --name=usrlocal

这是在 raid 基础上进行了逻辑卷的配置管理。首先, clearpart 对 hda 和 hdc 进行数据清楚,zerombr 初始化未使用的分区表。然后,分别对这两块硬盘分成了5个分区,将两两对应的分区再次划分,创建成配置有 raid1 磁盘阵列的分区,此为格式化的实际使用的分区。最终,还能继续对这些分区进行逻辑卷划分,比如对 pv.01 ,创建逻辑管组 sysvg,在此基础上创建 /var/usr/local 分区,--grow 选项,说明该分区的大小不固定,该分区可以使用 pv.01 剩余的空间

6 pre and post script

preinstall 脚本用 %pre 指示开头,脚本使用的语法和 RPM spec 文件中的类似。在安装之前这个阶段,preinstall 脚本只能做一些基本的操作,比如查找一些文件或者挂载NFS共享等等。

postintall 脚本则更有潜力,postinstall 脚本用 %post 指示开头。当OS安装已经完成以后,postinstall 脚本默认是在新安装好的系统上执行,新安装好的系统这个时候处于 chroot 环境,根目录其实是在/mnt/sysimage下。

可以使用 %post --nochroot 来执行一些非 chroot 环境下的命令。比如在 chroot 环境下,你是无法访问光驱中的安装光盘的,因为光盘挂载在/tmp/cdrom,它是处于 non-chroot 环境下的。

一般来说 postinstall 脚本以 #!/bin/sh 这样的自定义shell作为第一行,然后下边就是普通的shell脚本了。这样基本上你想做什么都可以了。

7 系统安装时找不到系统盘

dracut listqueue timeout

找不到系统安装盘,在安装启动界面,按键 e 修改,将

inst.stage2

修改成

inst.stage2=hd:sdc4 inst.ks=hd:sdc4:/isolinux/ks.cfg

笔者采用的是u盘安装,安装时 u 盘是 /dev/sdc4

同时,在安装启动界面,在 quiet 前面加上 dd 命令,进入系统后会检测硬盘信息,检测出 u 盘的盘符,然后重启,进行上述修改即可

安装系统时,引导程序寻找安装镜像,比如上面

inst.stage2=hd:LABEL=CentOS7

将镜像写入 U 盘时,U盘的标签名将被命名为 CentOs7,引导程序通过 LABEL 的名称寻找到 U 盘,同时加载相应的程序进行安装。如果找不到,就采用上面那种指定 U 盘的方式进行安装。所以上面应该改成如下的方式即可自动化安装

inst.stage2=hd:LABEL=CentOS7 inst.ks=hd:LABEL=CentOS7:/isolinux/ks.cfg