改用 alpine 作为中间系统

This commit is contained in:
bin456789
2023-05-13 00:14:46 +08:00
parent f03a178260
commit 72f486d252
7 changed files with 406 additions and 222 deletions

View File

@ -3,21 +3,47 @@ confhome=https://raw.githubusercontent.com/bin456789/reinstall/main
localtest_confhome=http://192.168.253.1
usage_and_exit() {
echo "Usage: reinstall.sh centos-7/8/9 alma-8/9 rocky-8/9 fedora-36/37/38 ubuntu-20.04/22.04"
echo "Usage: reinstall.sh centos-7/8/9 alma-8/9 rocky-8/9 fedora-36/37/38 ubuntu-20.04/22.04 alpine-3.16/3.17/3.18"
exit 1
}
is_in_china() {
# https://geoip.ubuntu.com/lookup
curl -L https://geoip.fedoraproject.org/city | grep -w CN
if [ -z $_is_in_china ]; then
# https://geoip.ubuntu.com/lookup
curl -L https://geoip.fedoraproject.org/city | grep -w CHN
_is_in_china=$?
fi
return $_is_in_china
}
setos() {
step=$1
distro=$2
releasever=$3
ks=$4
vault=$5
local step=$1
local distro=$2
local releasever=$3
local ks=$4
setos_alpine() {
eval ${step}_distro=$distro
if [ "$localtest" = 1 ]; then
mirror=$confhome/alpine-netboot-3.17.3-x86_64/boot
eval ${step}_vmlinuz=$mirror/vmlinuz-lts
eval ${step}_initrd=$mirror/initramfs-lts
eval ${step}_repo=https://mirrors.aliyun.com/alpine/v3.17/main
eval ${step}_modloop=$mirror/modloop-lts
else
# 不要用https 因为甲骨文云arm initramfs阶段不会从硬件同步时钟导致访问https出错
if is_in_china; then
mirror=http://mirrors.aliyun.com/alpine/v$releasever
else
mirror=http://dl-cdn.alpinelinux.org/alpine/v$releasever
fi
eval ${step}_vmlinuz=$mirror/releases/$basearch/netboot/vmlinuz-lts
eval ${step}_initrd=$mirror/releases/$basearch/netboot/initramfs-lts
eval ${step}_repo=$mirror/main
eval ${step}_modloop=$mirror/releases/$basearch/netboot/modloop-lts
fi
}
setos_ubuntu() {
if [ "$localtest" = 1 ]; then
@ -48,40 +74,28 @@ setos() {
eval ${step}_distro=ubuntu
}
setos_rh() {
setos_redhat() {
if [ "$localtest" = 1 ]; then
mirror=$confhome/$releasever/
eval ${step}_ks=$confhome/$ks
else
# 甲骨文 arm 1g内存用 centos 7.9 镜像会 oom 进不了安装界面所以用7.6
if [ "$vault" = 1 ]; then
if is_in_china; then
[ "$basearch" = "x86_64" ] && dir= || dir=/altarch
mirror="https://mirrors.aliyun.com/centos-vault$dir/7.6.1810/os/$basearch/"
else
[ "$basearch" = "x86_64" ] && dir=centos || dir=altarch
mirror=http://vault.centos.org/$dir/7.6.1810/os/$basearch/
fi
else
case $distro in
"centos")
case $releasever in
"7") mirrorlist="http://mirrorlist.centos.org/?release=7&arch=$basearch&repo=os" ;;
"8") mirrorlist="http://mirrorlist.centos.org/?release=8-stream&arch=$basearch&repo=BaseOS" ;;
"9") mirrorlist="https://mirrors.centos.org/mirrorlist?repo=centos-baseos-9-stream&arch=$basearch" ;;
esac
;;
"alma") mirrorlist="https://mirrors.almalinux.org/mirrorlist/$releasever/baseos" ;;
"rocky") mirrorlist="https://mirrors.rockylinux.org/mirrorlist?arch=$basearch&repo=BaseOS-$releasever" ;;
"fedora") mirrorlist="https://mirrors.fedoraproject.org/mirrorlist?arch=$basearch&repo=fedora-$releasever" ;;
case $distro in
"centos")
case $releasever in
"7") mirrorlist="http://mirrorlist.centos.org/?release=7&arch=$basearch&repo=os" ;;
"8") mirrorlist="http://mirrorlist.centos.org/?release=8-stream&arch=$basearch&repo=BaseOS" ;;
"9") mirrorlist="https://mirrors.centos.org/mirrorlist?repo=centos-baseos-9-stream&arch=$basearch" ;;
esac
# rocky/centos9 需要删除第一行注释, alma 需要替换$basearchanigil 这个源不稳定
mirror=$(curl -L $mirrorlist | sed "/^#/d" | sed "/anigil/d" | head -1 | sed "s,\$basearch,$basearch,")
eval "${step}_mirrorlist='${mirrorlist}'"
fi
eval ${step}_ks=$confhome/$ks
;;
"alma") mirrorlist="https://mirrors.almalinux.org/mirrorlist/$releasever/baseos" ;;
"rocky") mirrorlist="https://mirrors.rockylinux.org/mirrorlist?arch=$basearch&repo=BaseOS-$releasever" ;;
"fedora") mirrorlist="https://mirrors.fedoraproject.org/mirrorlist?arch=$basearch&repo=fedora-$releasever" ;;
esac
# rocky/centos9 需要删除第一行注释, alma 需要替换$basearchanigil 这个源不稳定
mirror=$(curl -L $mirrorlist | sed "/^#/d" | sed "/anigil/d" | head -1 | sed "s,\$basearch,$basearch,")
eval "${step}_mirrorlist='${mirrorlist}'"
fi
eval ${step}_ks=$confhome/$ks
eval ${step}_distro=${distro}
eval ${step}_vmlinuz=${mirror}images/pxeboot/vmlinuz
eval ${step}_initrd=${mirror}images/pxeboot/initrd.img
@ -92,20 +106,59 @@ setos() {
fi
}
if [ "$distro" = "ubuntu" ]; then
setos_ubuntu
else
setos_rh
fi
case "$distro" in
ubuntu) setos_ubuntu ;;
alpine) setos_alpine ;;
*) setos_redhat ;;
esac
}
# 检查是否为正确的系统名
verify_os_string() {
for os in 'centos-7|8|9' 'alma|rocky-8|9' 'fedora-36|37|38' 'ubuntu-20.04|22.04' 'alpine-3.16|3.17|3.18'; do
ds=$(echo $os | cut -d- -f1)
vers=$(echo $os | cut -d- -f2 | sed 's \. \\\. g')
finalos=$(echo "$@" | tr '[:upper:]' '[:lower:]' | sed -n -E "s,^($ds)[ :-]?($vers)$,\1:\2,p")
if [ -n "$finalos" ]; then
distro=$(echo $finalos | cut -d: -f1)
releasever=$(echo $finalos | cut -d: -f2)
return
fi
done
echo "Please specify a proper os."
usage_and_exit
}
apt_install() {
[ -z "$apk_updated" ] && apt update && apk_updated=1
apt install -y $pkgs
}
install_pkg() {
pkgs=$*
for pkg in $pkgs; do
if ! command -v $pkg; then
{
apt_install $pkgs ||
dnf install -y $pkgs ||
yum install -y $pkgs ||
zypper install -y $pkgs ||
pacman -Syu $pkgs ||
apk add $pkgs
} 2>/dev/null
break
fi
done
}
# 脚本入口
if [ "$EUID" -ne 0 ]; then
echo "Please run as root."
exit 1
fi
opts=$(getopt -a -n $0 --options l --long localtest -- "$@")
if [ "$?" != 0 ]; then
if ! opts=$(getopt -a -n $0 --options l --long localtest -- "$@"); then
usage_and_exit
fi
@ -128,68 +181,45 @@ while true; do
esac
done
# 检查是否为正确的系统名
for re in \
"s,.*\b(centos|alma|rocky)(linux)?[ :-]?([789])\b.*,\1:\3,p" \
"s,.*\b(fedora)(linux)?[ :-]?(3[678])\b.*,\1:\3,p" \
"s,.*\b(ubuntu)(linux)?[ :-]?(2[02]\.04)\b.*,\1:\3,p"; do
finalos=$(echo "$@" | tr '[:upper:]' '[:lower:]' | sed -n -E "$re")
if [ -n "$finalos" ]; then
break
fi
done
if [ -z "$finalos" ]; then
echo "Please specify a proper os."
usage_and_exit
fi
# 最小安装的 debian 不包含 curl dmidecode
install_pkg() {
pkgs=$1
for pkg in $pkgs; do
if ! command -v $pkg; then
{
{ apt update && apt install -y $pkgs; } ||
dnf install -y $pkgs ||
yum install -y $pkgs ||
zypper install -y $pkgs ||
pacman -Syu $pkgs
} 2>/dev/null
break
fi
done
}
# 获取内存大小lsmem最准确但centos7 arm不能用
# arm 24g dmidecode 显示少了128m
ram_size=$(lsmem -b | grep 'Total online memory:' | awk '{ print $NF/1024/1024 }')
if [ -z $ram_size ]; then
install_pkg dmidecode
ram_size=$(dmidecode -t 17 | grep "Size.*[GM]B" | awk '{if ($3=="GB") s+=$2*1024; else s+=$2} END {print s}')
fi
if [ $ram_size -lt 1024 ]; then
echo 'RAM < 1G. Unsupported.'
exit 1
fi
verify_os_string "$@"
# 必备组件
install_pkg curl
distro=$(echo $finalos | cut -d: -f1)
releasever=$(echo $finalos | cut -d: -f2)
basearch=$(uname -m)
# alpine 自带的 grep 是 busybox 里面的, 要下载完整版grep
if [ -f /etc/alpine-release ]; then
apk add grep
fi
# 以下目标系统需要两步安装
# ubuntu
# 获取内存大小lsmem最准确但centos7 arm 和alpine不能用
# arm 24g dmidecode 显示少了128m
if [ $distro != "alpine" ]; then
install_pkg util-linux
ram_size=$(lsmem -b 2>/dev/null | grep 'Total online memory:' | awk '{ print $NF/1024/1024 }')
if [ -z $ram_size ]; then
install_pkg dmidecode
ram_size=$(dmidecode -t 17 | grep "Size.*[GM]B" | awk '{if ($3=="GB") s+=$2*1024; else s+=$2} END {print s}')
fi
if [ $ram_size -lt 1024 ]; then
echo 'RAM < 1G. Unsupported.'
exit 1
fi
fi
basearch=$(uname -m)
# 以下目标系统需要进入alpine环境安装
# ubuntu/alpine
# el8/9/fedora 任何架构 <2g
# el7 aarch64 <1.5g
# shellcheck disable=SC2154
if [ $distro = "ubuntu" ] ||
[ $distro = "alpine" ] ||
{ [ $releasever -ge 8 ] && [ $ram_size -lt 2048 ]; } ||
{ [ $releasever -eq 7 ] && [ $ram_size -lt 1536 ] && [ $basearch = "aarch64" ]; }; then
[ $distro = "ubuntu" ] && ks=user-data || ks=ks.cfg
# 安装alpine时使用指定的版本。 alpine作为中间系统时使用 3.18
[ $distro = "alpine" ] && alpine_releasever=$releasever || alpine_releasever=3.18
setos finalos $distro $releasever $ks
setos nextos centos 7 ks-trans.cfg 1
setos nextos alpine $alpine_releasever
else
setos nextos $distro $releasever ks.cfg
fi
@ -198,53 +228,110 @@ fi
# shellcheck disable=SC2154
{
cd /
curl -LO $nextos_vmlinuz
curl -LO $nextos_initrd
echo $nextos_vmlinuz
curl -Lo vmlinuz $nextos_vmlinuz
echo $nextos_initrd
curl -Lo initrd $nextos_initrd
touch reinstall.mark
}
# 转换 finalos_a=1 为 finalos.a=1 ,排除 finalos_mirrorlist
build_finalos_cmdline() {
for var in $(compgen -v finalos_); do
key=${var//_/.}
[ $key != "finalos.mirrorlist" ] && finalos_cmdline+=" $key=${!var}"
for key in $(compgen -v finalos_); do
value=${!key}
key=${key#finalos_}
if [ -n "$value" ] && [ $key != "mirrorlist" ]; then
finalos_cmdline+=" finalos.$key=$value"
fi
done
}
build_extra_cmdline() {
extra_cmdline+=" extra.confhome=$confhome"
if [ "$localtest" = 1 ]; then
extra_cmdline+=" extra.localtest=$localtest"
else
# 指定最终安装系统的 mirrorlist链接有&在grub中是特殊字符所以要加引号
if [ -n "$finalos_mirrorlist" ]; then
extra_cmdline+=" extra.mirrorlist='$finalos_mirrorlist'"
elif [ -n "$nextos_mirrorlist" ]; then
extra_cmdline+=" extra.mirrorlist='$nextos_mirrorlist'"
for key in localtest confhome; do
value=${!key}
if [ -n "$value" ]; then
extra_cmdline+=" extra.$key=$value"
fi
done
# 指定最终安装系统的 mirrorlist链接有&在grub中是特殊字符所以要加引号
if [ -n "$finalos_mirrorlist" ]; then
extra_cmdline+=" extra.mirrorlist='$finalos_mirrorlist'"
elif [ -n "$nextos_mirrorlist" ]; then
extra_cmdline+=" extra.mirrorlist='$nextos_mirrorlist'"
fi
}
# arm64 不需要添加 efi 字样
if [ -d /sys/firmware/efi ] && [ "$basearch" = x86_64 ]; then
action='efi'
fi
build_finalos_cmdline
build_extra_cmdline
grub_cfg=$(find /boot -type f -name grub.cfg -exec grep -E -l 'menuentry|blscfg' {} \;)
grub_cfg_dir=$(dirname $grub_cfg)
# 在x86 efi机器上可能用 linux 或 linuxefi 加载内核
# 通过检测原有的条目有没有 linuxefi 字样就知道当前 grub 用哪一种
search_files=$(find /boot -type f -name grub.cfg)
if [ -d /boot/loader/entries/ ]; then
search_files="$search_files /boot/loader/entries/"
fi
if grep -q -r -E '^[[:blank:]]*linuxefi[[:blank:]]' $search_files; then
efi=efi
fi
# 修改 alpine 启动时运行我们的脚本
# shellcheck disable=SC2154,SC2164
if [ -n "$finalos_cmdline" ]; then
install_pkg gzip cpio
# 解压
# 先删除临时文件,避免之前运行中断有残留文件
tmp_dir=/tmp/reinstall/
rm -rf $tmp_dir
mkdir -p $tmp_dir
cd $tmp_dir
zcat /initrd | cpio -idm
# hack
# exec /bin/busybox switch_root $switch_root_opts $sysroot $chart_init "$KOPT_init" $KOPT_init_args # 3.17
# exec switch_root $switch_root_opts $sysroot $chart_init "$KOPT_init" $KOPT_init_args # 3.18
line_num=$(grep -E -n '^exec (/bin/busybox )?switch_root' init | cut -d: -f1)
line_num=$((line_num - 1))
cat <<EOF | sed -i "${line_num}r /dev/stdin" init
# alpine arm initramfs 时间问题 要添加 --no-check-certificate
wget --no-check-certificate -O \$sysroot/etc/local.d/trans.start $confhome/trans.sh
chmod a+x \$sysroot/etc/local.d/trans.start
ln -s /etc/init.d/local \$sysroot/etc/runlevels/default/
EOF
# 重建
# 注意要用 cpio -H newc 不要用 cpio -c ,不同版本的 -c 作用不一样,很坑
# -c Use the old portable (ASCII) archive format
# -c Identical to "-H newc", use the new (SVR4)
# portable format.If you wish the old portable
# (ASCII) archive format, use "-H odc" instead.
find . | cpio -o -H newc | gzip -1 >/initrd
# 删除临时文件
cd /
rm -rf $tmp_dir
# 可添加 pkgs=xxx,yyy 启动时自动安装
# apkovl=http://xxx.com/apkovl.tar.gz 可用arm https未测但应该不行
# apkovl=sda2:ext4:/apkovl.tar.gz 官方有写但不生效
cmdline="alpine_repo=$nextos_repo modloop=$nextos_modloop $extra_cmdline $finalos_cmdline "
else
cmdline="root=live:$nextos_squashfs inst.ks=$nextos_ks $extra_cmdline"
fi
custom_cfg=$grub_cfg_dir/custom.cfg
echo $custom_cfg
# shellcheck disable=SC2154
cat <<EOF | tee $custom_cfg
menuentry "reinstall" {
insmod lvm
insmod xfs
search --no-floppy --file --set=root /reinstall.mark
linux$action /vmlinuz root=live:$nextos_squashfs inst.ks=$nextos_ks $finalos_cmdline $extra_cmdline
initrd$action /initrd.img
linux$efi /vmlinuz $cmdline
initrd$efi /initrd
}
EOF