#!/bin/bash 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-37/38 ubuntu-20.04/22.04 alpine-3.16/3.17/3.18 debian-10/11 windows dd" exit 1 } error_and_exit() { echo "Error: $1" exit 1 } is_in_china() { 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 } test_url() { url=$1 expect_type=$2 tmp_file=/tmp/reinstall-img-test install_pkg file http_code=$(curl -Ls -r 0-1048576 -w "%{http_code}" -o $tmp_file $url) if [ "$http_code" != 200 ] && [ "$http_code" != 206 ]; then error_and_exit "$url not accessible" fi mime=$(file -ib $tmp_file | cut -d';' -f1 | cut -d'/' -f2) case $mime in gzip) img_type=gz ;; x-xz) img_type=xz ;; x-iso9660-image) img_type=iso ;; esac if ! echo $expect_type | grep -wo "$img_type"; then error_and_exit "$url $mime not supported" fi } setos() { local step=$1 local distro=$2 local releasever=$3 setos_alpine() { if [ "$localtest" = 1 ]; then mirror=$confhome/alpine-netboot-3.18.0-x86_64/boot eval ${step}_vmlinuz=$mirror/vmlinuz-lts eval ${step}_initrd=$mirror/initramfs-lts eval ${step}_repo=https://mirrors.aliyun.com/alpine/v$releasever/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_debian() { if [ "$localtest" = 1 ]; then mirror=$confhome/debian/install.amd eval ${step}_vmlinuz=$mirror/vmlinuz eval ${step}_initrd=$mirror/initrd.gz else case "$releasever" in 12) codename=bookworm ;; 11) codename=bullseye ;; 10) codename=buster ;; esac if is_in_china; then hostname=ftp.cn.debian.org else hostname=deb.debian.org fi mirror=http://$hostname/debian/dists/$codename/main/installer-$basearch_alt/current/images/netboot/debian-installer/$basearch_alt eval ${step}_vmlinuz=$mirror/linux eval ${step}_initrd=$mirror/initrd.gz fi eval ${step}_ks=$confhome/preseed.cfg } setos_ubuntu() { if [ "$localtest" = 1 ]; then mirror=$confhome/ else if is_in_china; then case "$basearch" in "x86_64") mirror=https://mirrors.aliyun.com/ubuntu-releases/$releasever/ ;; "aarch64") mirror=https://mirrors.aliyun.com/ubuntu-cdimage/releases/$releasever/release/ ;; esac else case "$basearch" in "x86_64") mirror=https://releases.ubuntu.com/$releasever/ ;; "aarch64") mirror=https://cdimage.ubuntu.com/releases/$releasever/release/ ;; esac fi fi filename=$(curl $mirror | grep -oP "ubuntu-$releasever.*?-live-server-$basearch_alt.iso" | head -1) iso=$mirror$filename test_url $iso iso eval ${step}_iso=$iso eval ${step}_ks=$confhome/user-data } setos_windows() { if [ -z "$iso" ] || [ -z "$image_name" ]; then echo "Install Windows need --iso --image-name" exit 1 fi test_url $iso iso eval "${step}_iso='$iso'" eval "${step}_image_name='$image_name'" } setos_dd() { if [ -z "$img" ]; then echo "dd need --img" exit 1 fi test_url $img 'xz|gz' eval "${step}_img='$img'" eval "${step}_img_type='$img_type'" } setos_redhat() { if [ "$localtest" = 1 ]; then mirror=$confhome/$releasever/ 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" ;; esac # rocky/centos9 需要删除第一行注释, alma 需要替换$basearch,anigil 这个源不稳定 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.cfg eval ${step}_vmlinuz=${mirror}images/pxeboot/vmlinuz eval ${step}_initrd=${mirror}images/pxeboot/initrd.img eval ${step}_squashfs=${mirror}images/install.img if [ "$releasever" = 7 ]; then eval ${step}_squashfs=${mirror}LiveOS/squashfs.img fi } eval ${step}_distro=$distro case "$distro" in ubuntu) setos_ubuntu ;; alpine) setos_alpine ;; debian) setos_debian ;; windows) setos_windows ;; dd) setos_dd ;; *) setos_redhat ;; esac } # 检查是否为正确的系统名 verify_os_string() { for os in 'centos-7|8|9' 'alma|rocky-8|9' 'fedora-37|38' 'ubuntu-20.04|22.04' 'alpine-3.16|3.17|3.18' 'debian-10|11|12' 'windows-' 'dd-'; 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) if [ "$distro" = centos ] || [ "$distro" = alma ] || [ "$distro" = rocky ]; then distro_like=redhat fi 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 # util-linux 用 lsmem 命令测试 [ "$pkg" = util-linux ] && pkg=lsmem if ! command -v $pkg; then { apt_install $pkgs || dnf install -y $pkgs || yum install -y $pkgs || zypper install -y $pkgs || pacman -Syu --noconfirm $pkgs || apk add $pkgs } 2>/dev/null break fi done } check_ram() { # lsmem最准确但centos7 arm 和alpine不能用 # arm 24g dmidecode 显示少了128m 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 case "$distro" in alpine) ram_requirement=0 ;; # 未测试 debian) ram_requirement=384 ;; *) ram_requirement=1024 ;; esac if [ $ram_size -lt $ram_requirement ]; then echo "Could not install $distro: RAM < $ram_requirement MB." exit 1 fi } # 脚本入口 if [ "$EUID" -ne 0 ]; then echo "Please run as root." exit 1 fi if ! opts=$(getopt -a -n $0 --options l --long localtest,iso:,image-name:,img: -- "$@"); then usage_and_exit fi eval set -- "$opts" while true; do case "$1" in -l | --localtest) localtest=1 confhome=$localtest_confhome shift ;; --img) img=$2 shift 2 ;; --iso) iso=$2 shift 2 ;; --image-name) image_name=$2 shift 2 ;; --) shift break ;; *) echo "Unexpected option: $1." usage_and_exit ;; esac done verify_os_string "$@" # 必备组件 install_pkg curl # alpine 自带的 grep 是 busybox 里面的, 要下载完整版grep if [ -f /etc/alpine-release ]; then apk add grep fi check_ram basearch=$(uname -m) case "$basearch" in "x86_64") basearch_alt=amd64 ;; "aarch64") basearch_alt=arm64 ;; esac # 以下目标系统需要进入alpine环境安装 # ubuntu/alpine # el8/9/fedora 任何架构 <2g # el7 aarch64 <1.5g if [ "$distro" = "ubuntu" ] || [ "$distro" = "alpine" ] || [ "$distro" = "windows" ] || [ "$distro" = "dd" ] || { [ "$distro_like" = "redhat" ] && [ $releasever -ge 8 ] && [ $ram_size -lt 2048 ]; } || { [ "$distro_like" = "redhat" ] && [ $releasever -eq 7 ] && [ $ram_size -lt 1536 ] && [ $basearch = "aarch64" ]; }; then # 安装alpine时,使用指定的版本。 alpine作为中间系统时,使用 3.18 [ "$distro" = "alpine" ] && alpine_releasever=$releasever || alpine_releasever=3.18 setos finalos $distro $releasever setos nextos alpine $alpine_releasever else setos nextos $distro $releasever fi # 下载启动内核 # shellcheck disable=SC2154 { cd / echo $nextos_vmlinuz curl -Lo reinstall-vmlinuz $nextos_vmlinuz echo $nextos_initrd curl -Lo reinstall-initrd $nextos_initrd } # 转换 finalos_a=1 为 finalos.a=1 ,排除 finalos_mirrorlist build_finalos_cmdline() { 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() { 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 } 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 /reinstall-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 </reinstall-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 if [ $distro = debian ]; then cmdline="lowmem=+1 lowmem/low=1 auto=true priority=critical url=$nextos_ks" else cmdline="root=live:$nextos_squashfs inst.ks=$nextos_ks $extra_cmdline" fi fi custom_cfg=$grub_cfg_dir/custom.cfg echo $custom_cfg cat <