reinstall/trans.sh

1559 lines
50 KiB
Bash
Raw Normal View History

2023-05-13 00:14:46 +08:00
#!/bin/ash
# shellcheck shell=dash
2023-09-16 20:00:05 +08:00
# shellcheck disable=SC2086,SC3047,SC3036,SC3010,SC3001
2023-05-13 00:14:46 +08:00
# alpine 默认使用 busybox ash
2023-07-08 16:02:06 +08:00
# 命令出错终止运行,将进入到登录界面,防止失联
2023-06-18 21:27:22 +08:00
set -eE
2023-09-16 20:00:05 +08:00
trap 'trap_err $LINENO $?' ERR
# 复制本脚本到 /tmp/trans.sh用于打印错误
# 也有可能从管道运行,这时删除 /tmp/trans.sh
case "$0" in
*trans.*) cp -f "$0" /tmp/trans.sh ;;
*) rm -f /tmp/trans.sh ;;
esac
# 还原改动,不然本脚本会被复制到新系统
rm -f /etc/local.d/trans.start
rm -f /etc/runlevels/default/local
trap_err() {
line_no=$1
ret_no=$2
error "Line $line_no return $ret_no"
if [ -f "/tmp/trans.sh" ]; then
sed -n "$line_no"p /tmp/trans.sh
2023-06-18 21:27:22 +08:00
fi
}
error() {
color='\e[31m'
plain='\e[0m'
2023-07-08 16:02:06 +08:00
echo -e "${color}Error: $*${plain}"
2023-06-18 21:27:22 +08:00
}
2023-05-13 00:14:46 +08:00
2023-07-25 00:21:08 +08:00
error_and_exit() {
error "$@"
exit 1
}
2023-05-25 20:15:12 +08:00
add_community_repo() {
2023-08-25 13:04:06 +08:00
# 先检查原来的repo是不是egde
if grep -x 'http.*/edge/main' /etc/apk/repositories; then
alpine_ver=edge
else
alpine_ver=v$(cut -d. -f1,2 </etc/alpine-release)
fi
if ! grep -x "http.*/$alpine_ver/community" /etc/apk/repositories; then
echo http://dl-cdn.alpinelinux.org/alpine/$alpine_ver/community >>/etc/apk/repositories
2023-07-05 21:57:27 +08:00
fi
2023-05-25 20:15:12 +08:00
}
2023-06-18 21:27:22 +08:00
download() {
2023-07-03 23:11:10 +08:00
url=$1
file=$2
echo $url
2023-06-18 21:27:22 +08:00
# 阿里云禁止 axel 下载
# axel https://mirrors.aliyun.com/alpine/latest-stable/releases/x86_64/alpine-netboot-3.17.0-x86_64.tar.gz
# Initializing download: https://mirrors.aliyun.com/alpine/latest-stable/releases/x86_64/alpine-netboot-3.17.0-x86_64.tar.gz
# HTTP/1.1 403 Forbidden
2023-07-03 23:11:10 +08:00
# axel 在 lightsail 上会占用大量cpu
# 构造 aria2 参数
# 没有指定文件名的情况
if [ -z $file ]; then
save=""
else
# 文件名是绝对路径
if [[ "$file" = "/*" ]]; then
save="-d / -o $file"
else
# 文件名是相对路径
save="-o $file"
fi
fi
2023-07-03 23:11:10 +08:00
# 先用 aria2 下载
2023-08-08 23:36:23 +08:00
# aria2 下载 fedora 官方镜像链接会将meta4文件下载下来而且占用了指定文件名造成重命名失效。而且无法指定目录
# https://download.opensuse.org/distribution/leap/15.5/appliances/openSUSE-Leap-15.5-Minimal-VM.x86_64-kvm-and-xen.qcow2
# https://aria2.github.io/manual/en/html/aria2c.html#cmdoption-o
2023-07-28 21:27:16 +08:00
if ! (command -v aria2c && aria2c -x4 --allow-overwrite=true $url $save); then
2023-08-08 23:36:23 +08:00
# 出错再用 wget
# alpine 的 busybox 没有 curl
[ -z $file ] && save="" || save="-O $file"
wget $url $save
2023-06-18 21:27:22 +08:00
fi
}
update_part() {
2023-07-18 00:01:07 +08:00
{
hdparm -z $1
partprobe $1
partx -u $1
udevadm settle
echo 1 >/sys/block/${1#/dev/}/device/rescan
} 2>/dev/null || true
}
2023-06-18 21:27:22 +08:00
2023-07-06 22:20:09 +08:00
is_efi() {
[ -d /sys/firmware/efi/ ]
}
2023-07-18 00:22:51 +08:00
is_use_cloud_image() {
[ -n "$cloud_image" ] && [ "$cloud_image" = 1 ]
}
2023-07-08 16:02:06 +08:00
setup_nginx() {
apk add nginx
cat <<EOF >/etc/nginx/http.d/default.conf
server {
listen 80 default_server;
listen [::]:80 default_server;
2023-07-08 16:02:06 +08:00
location = / {
root /;
try_files /reinstall.html /reinstall.html;
# types {
# text/plain log;
# }
2023-07-08 16:02:06 +08:00
}
}
EOF
# rc-service nginx start
2023-07-18 00:01:07 +08:00
if pgrep nginx >/dev/null; then
nginx -s reload
else
nginx
fi
2023-07-08 16:02:06 +08:00
}
2023-07-25 00:21:08 +08:00
setup_nginx_if_enough_ram() {
total_ram=$(free -m | awk '{print $2}' | sed -n '2p')
# 避免后面没内存安装程序谨慎起见512内存才安装
if [ $total_ram -gt 400 ]; then
# lighttpd 虽然运行占用内存少,但安装占用空间大
# setup_lighttpd
setup_nginx
fi
}
2023-07-08 16:02:06 +08:00
setup_lighttpd() {
apk add lighttpd
ln -sf /reinstall.html /var/www/localhost/htdocs/index.html
2023-07-08 16:02:06 +08:00
rc-service lighttpd start
}
2023-09-03 19:35:01 +08:00
# 最后一个 tty 是主tty显示的信息最多
# 有些平台例如 aws/gcp 只能截图,不能输入(没有鼠标)
# 所以如果有显示器且有鼠标tty0 放最后面,否则 tty0 放前面
2023-08-08 23:21:28 +08:00
get_ttys() {
prefix=$1
2023-09-16 20:00:05 +08:00
# shellcheck disable=SC2154
2023-09-03 19:35:01 +08:00
wget $confhome/ttys.sh -O- | sh -s $prefix
2023-08-08 23:21:28 +08:00
}
setup_tty_and_log() {
cat <<EOF >/reinstall.html
<!DOCTYPE html>
<html lang="en">
<head>
<meta http-equiv="refresh" content="2">
</head>
<body>
<script>
window.onload = function() {
// history.scrollRestoration = "manual";
window.scrollTo(0, document.body.scrollHeight);
}
</script>
<pre>
EOF
# 显示输出到前台
# script -f /dev/tty0
2023-08-08 23:21:28 +08:00
dev_ttys=$(get_ttys /dev/)
exec > >(tee -a $dev_ttys /reinstall.html) 2>&1
}
extract_env_from_cmdline() {
# 提取 finalos/extra 到变量
for prefix in finalos extra; do
while read -r line; do
if [ -n "$line" ]; then
key=$(echo $line | cut -d= -f1)
value=$(echo $line | cut -d= -f2-)
eval "$key='$value'"
fi
2023-09-21 00:12:35 +08:00
done < <(xargs -n1 </proc/cmdline | grep "^$prefix" | sed "s/^$prefix\.//")
done
}
qemu_nbd() {
command qemu-nbd "$@"
sleep 5
}
mod_motd() {
cat <<EOF >/etc/motd
Reinstalling...
Run "tail -f /reinstall.html" to view logs.
EOF
}
2023-07-18 00:01:07 +08:00
# 可能脚本不是首次运行,先清理之前的残留
clear_previous() {
{
# TODO: fuser and kill
qemu_nbd -d /dev/nbd0
2023-07-30 00:35:32 +08:00
swapoff -a
# alpine 自带的umount没有-R除非安装了util-linux
umount -R /iso /wim /installer /os/installer /os /nbd /nbd-boot /nbd-efi /mnt
umount /iso /wim /installer /os/installer /os /nbd /nbd-boot /nbd-efi /mnt
2023-07-18 00:01:07 +08:00
} 2>/dev/null || true
}
2023-08-25 23:18:23 +08:00
get_virt_to() {
if [ -z "$_virt" ]; then
apk add virt-what
_virt="$(virt-what)"
apk del virt-what
fi
eval "$1='$_virt'"
}
2023-08-25 23:18:23 +08:00
is_virt() {
get_virt_to virt
[ -n "$virt" ]
}
2023-08-25 23:18:23 +08:00
get_ra_to() {
if [ -z "$_ra" ]; then
apk add ndisc6
# 有时会重复收取,所以设置收一份后退出
_ra="$(rdisc6 -1 eth0)"
2023-08-25 23:18:23 +08:00
apk del ndisc6
fi
2023-08-25 23:18:23 +08:00
eval "$1='$_ra'"
}
2023-08-25 23:18:23 +08:00
get_netconf_to() {
case "$1" in
slaac | dhcpv6 | rdnss) get_ra_to ra ;;
esac
# shellcheck disable=SC2154
case "$1" in
slaac) echo "$ra" | grep 'Autonomous address conf' | grep Yes && res=1 || res=0 ;;
dhcpv6) echo "$ra" | grep 'Stateful address conf' | grep Yes && res=1 || res=0 ;;
rdnss) res=$(echo "$ra" | grep 'Recursive DNS server' | cut -d: -f2- | xargs) ;;
2023-09-03 19:35:02 +08:00
*) res=$(cat /dev/$1) ;;
2023-08-25 23:18:23 +08:00
esac
eval "$1='$res'"
}
is_dhcpv4() {
get_netconf_to dhcpv4
# shellcheck disable=SC2154
[ "$dhcpv4" = 1 ]
}
is_slaac() {
get_netconf_to slaac
# shellcheck disable=SC2154
[ "$slaac" = 1 ]
}
2023-08-25 23:18:23 +08:00
is_dhcpv6() {
get_netconf_to dhcpv6
# shellcheck disable=SC2154
[ "$dhcpv6" = 1 ]
}
2023-09-16 20:00:05 +08:00
insert_into_file() {
file=$1
location=$2
regex_to_find=$3
line_num=$(grep -E -n "$regex_to_find" "$file" | cut -d: -f1)
if [ "$location" = before ]; then
line_num=$((line_num - 1))
elif ! [ "$location" = after ]; then
return 1
fi
sed -i "${line_num}r /dev/stdin" "$file"
}
2023-08-25 23:18:23 +08:00
install_alpine() {
2023-05-13 00:14:46 +08:00
# 网络
# 坑1 udhcpc下ip -4 addr 无法知道是否是 dhcp
# 坑2 udhcpc不支持dhcpv6
2023-08-25 12:58:48 +08:00
# 坑3 dhcpcd的slaac默认开了隐私保护造成ip和后台面板不一致
# slaac方案1: udhcpc + rdnssd
# slaac方案2: dhcpcd + 关闭隐私保护
# dhcpv6方案: dhcpcd
# 综合使用dhcpcd方案
# 1 无需改动/etc/network/interfaces自动根据ra使用slaac和dhcpv6
# 2 自带rdnss支持
# 3 唯一要做的是关闭隐私保护
# 安装 dhcpcd
apk add dhcpcd
sed -i '/^slaac private/s/^/#/' /etc/dhcpcd.conf
sed -i '/^#slaac hwaddr/s/^#//' /etc/dhcpcd.conf
2023-08-25 23:18:23 +08:00
rc-update add networking boot
2023-08-25 13:39:48 +08:00
# 生成 lo配置 + eth0头部
cat <<EOF >/etc/network/interfaces
auto lo
iface lo inet loopback
2023-08-25 13:39:48 +08:00
auto eth0
EOF
2023-05-13 00:14:46 +08:00
# 生成 ipv4 配置
2023-08-25 23:18:23 +08:00
if is_dhcpv4; then
2023-08-25 13:39:48 +08:00
echo "iface eth0 inet dhcp" >>/etc/network/interfaces
else
2023-08-25 23:18:23 +08:00
get_netconf_to ipv4_addr
get_netconf_to ipv4_gateway
if [ -n "$ipv4_addr" ]; then
2023-08-25 23:18:23 +08:00
# shellcheck disable=SC2154
cat <<EOF >>/etc/network/interfaces
2023-08-25 13:39:48 +08:00
iface eth0 inet static
address $ipv4_addr
gateway $ipv4_gateway
EOF
fi
fi
# 生成 ipv6 配置
2023-08-25 12:58:48 +08:00
# slaac 或者 dhcpv6实测不用写配置
if false; then
2023-08-25 23:18:23 +08:00
if is_slaac; then
2023-08-25 12:58:48 +08:00
echo 'iface eth0 inet6 auto' >>/etc/network/interfaces
fi
2023-08-25 23:18:23 +08:00
if is_dhcpv6; then
2023-08-25 12:58:48 +08:00
echo 'iface eth0 inet6 dhcp' >>/etc/network/interfaces
fi
fi
2023-08-25 12:58:48 +08:00
# 静态
2023-08-25 23:18:23 +08:00
if ! is_slaac && ! is_dhcpv6; then
get_netconf_to ipv6_addr
get_netconf_to ipv6_gateway
if [ -n "$ipv6_addr" ]; then
2023-08-25 23:18:23 +08:00
# shellcheck disable=SC2154
cat <<EOF >>/etc/network/interfaces
iface eth0 inet6 static
address $ipv6_addr
gateway $ipv6_gateway
EOF
2023-08-25 12:58:48 +08:00
# 如果有rdnss则删除自己添加的dns再添加rdnss
# 也有可能 dhcpcd 会自动设置,但没环境测试
2023-08-25 23:18:23 +08:00
get_netconf_to rdnss
2023-08-25 12:58:48 +08:00
if [ -n "$rdnss" ]; then
sed -i '/^[[:blank:]]*nameserver[[:blank:]].*:/d' /etc/resolv.conf
echo "nameserver $rdnss" >>/etc/resolv.conf
fi
fi
fi
2023-08-25 23:18:23 +08:00
# 显示网络配置
echo
ip addr | cat -n
echo
echo "$ra" | cat -n
echo
cat -n /etc/network/interfaces
2023-08-25 23:18:23 +08:00
echo
2023-05-13 00:14:46 +08:00
# 设置
setup-keymap us us
setup-timezone -i Asia/Shanghai
2023-06-18 21:27:22 +08:00
setup-ntp chrony || true
2023-05-13 00:14:46 +08:00
# 在 arm netboot initramfs init 中
# 如果识别到rtc硬件就往系统添加hwclock服务否则添加swclock
# 这个设置也被复制到安装的系统中
# 但是从initramfs chroot到真正的系统后是能识别rtc硬件的
# 所以我们手动改用hwclock修复这个问题
2023-06-18 21:27:22 +08:00
rc-update del swclock boot || true
2023-08-25 23:18:23 +08:00
rc-update add hwclock boot
2023-05-13 00:14:46 +08:00
# 通过 setup-alpine 安装会多启用几个服务
# https://github.com/alpinelinux/alpine-conf/blob/c5131e9a038b09881d3d44fb35e86851e406c756/setup-alpine.in#L189
# acpid | default
# crond | default
# seedrng | boot
2023-08-25 23:18:23 +08:00
if [ -e /dev/input/event0 ]; then
rc-update add acpid
fi
rc-update add crond
rc-update add seedrng boot
# 如果是 vm 就用 virt 内核
if is_virt; then
2023-09-03 19:35:01 +08:00
kernel_flavor="virt"
2023-08-25 23:36:32 +08:00
else
2023-09-03 19:35:01 +08:00
kernel_flavor="lts"
2023-08-25 23:18:23 +08:00
fi
2023-05-13 00:14:46 +08:00
# 重置为官方仓库配置
true >/etc/apk/repositories
setup-apkrepos -1
# 安装到硬盘
# alpine默认使用 syslinux (efi 环境除外),这里强制使用 grub方便用脚本再次重装
2023-09-03 19:35:01 +08:00
KERNELOPTS="$(get_ttys console=)"
export KERNELOPTS
2023-05-13 00:14:46 +08:00
export BOOTLOADER="grub"
2023-09-03 19:35:01 +08:00
printf 'y' | setup-disk -m sys -k $kernel_flavor -s 0 /dev/$xda
2023-07-25 00:21:08 +08:00
}
2023-06-04 19:12:57 +08:00
2023-07-25 00:21:08 +08:00
# shellcheck disable=SC2154
install_dd() {
2023-06-05 19:45:07 +08:00
case "$img_type" in
gzip) prog=gzip ;;
2023-06-04 19:12:57 +08:00
xz) prog=xz ;;
esac
if [ -n "$prog" ]; then
# alpine busybox 自带 gzip xz但官方版也许性能更好
2023-06-18 21:27:22 +08:00
# wget -O- $img | $prog -dc >/dev/$xda
2023-06-04 19:12:57 +08:00
apk add curl $prog
2023-07-03 13:51:59 +08:00
# curl -L $img | $prog -dc | dd of=/dev/$xda bs=1M
2023-06-05 19:45:07 +08:00
curl -L $img | $prog -dc >/dev/$xda
2023-06-18 21:27:22 +08:00
sync
2023-06-04 19:12:57 +08:00
else
2023-07-25 00:21:08 +08:00
error_and_exit 'Not supported'
2023-06-04 19:12:57 +08:00
fi
2023-07-25 00:21:08 +08:00
}
2023-05-03 22:22:21 +08:00
2023-07-25 00:21:08 +08:00
is_xda_gt_2t() {
disk_size=$(blockdev --getsize64 /dev/$xda)
disk_2t=$((2 * 1024 * 1024 * 1024 * 1024))
[ "$disk_size" -gt "$disk_2t" ]
}
2023-05-13 00:14:46 +08:00
2023-07-25 00:21:08 +08:00
create_part() {
# 目标系统非 alpine 和 dd
# 脚本开始
apk add util-linux aria2 grub udev hdparm e2fsprogs curl parted
2023-05-03 22:22:21 +08:00
2023-07-25 00:21:08 +08:00
# 打开dev才能刷新分区名
rc-service udev start
2023-05-03 22:22:21 +08:00
2023-07-25 00:21:08 +08:00
# 反激活 lvm
# alpine live 不需要
false && vgchange -an
2023-05-03 22:22:21 +08:00
2023-07-25 00:21:08 +08:00
# 移除 lsblk 显示的分区
2023-07-28 21:27:16 +08:00
partx -d /dev/$xda || true
# 清除分区签名
wipefs -a /dev/$xda
2023-05-03 22:22:21 +08:00
2023-07-25 00:21:08 +08:00
# xda*1 星号用于 nvme0n1p1 的字母 p
2023-09-16 20:00:05 +08:00
# shellcheck disable=SC2154
2023-07-25 00:21:08 +08:00
if [ "$distro" = windows ]; then
apk add ntfs-3g-progs virt-what wimlib rsync dos2unix
# 虽然ntfs3不需要fuse但wimmount需要所以还是要保留
modprobe fuse ntfs3
if is_efi; then
# efi
apk add dosfstools
parted /dev/$xda -s -- \
mklabel gpt \
mkpart '" "' fat32 1MiB 1025MiB \
mkpart '" "' fat32 1025MiB 1041MiB \
mkpart '" "' ext4 1041MiB -6GiB \
mkpart '" "' ntfs -6GiB 100% \
set 1 boot on \
set 2 msftres on \
set 3 msftdata on
update_part /dev/$xda
2023-09-16 20:00:05 +08:00
2023-07-28 21:27:16 +08:00
mkfs.fat -n efi /dev/$xda*1 #1 efi
2023-07-25 00:21:08 +08:00
echo #2 msr
mkfs.ext4 -F -L os /dev/$xda*3 #3 os
mkfs.ntfs -f -F -L installer /dev/$xda*4 #4 installer
else
# bios
parted /dev/$xda -s -- \
mklabel msdos \
mkpart primary ntfs 1MiB -6GiB \
mkpart primary ntfs -6GiB 100% \
set 1 boot on
update_part /dev/$xda
2023-09-16 20:00:05 +08:00
2023-07-25 00:21:08 +08:00
mkfs.ext4 -F -L os /dev/$xda*1 #1 os
mkfs.ntfs -f -F -L installer /dev/$xda*2 #2 installer
fi
elif is_use_cloud_image; then
2023-09-16 20:00:05 +08:00
# 这几个系统不使用dd而是复制文件因为dd这几个系统的qcow2需要10g硬盘
if { [ "$distro" = centos ] || [ "$distro" = alma ] || [ "$distro" = rocky ]; }; then
apk add dosfstools e2fsprogs
if is_efi; then
parted /dev/$xda -s -- \
mklabel gpt \
mkpart '" "' fat32 1MiB 601MiB \
mkpart '" "' xfs 601MiB -2GiB \
mkpart '" "' ext4 -2GiB 100% \
set 1 esp on
update_part /dev/$xda
mkfs.fat -n efi /dev/$xda*1 #1 efi
echo #2 os 用目标系统的格式化工具
mkfs.ext4 -F -L installer /dev/$xda*3 #3 installer
else
parted /dev/$xda -s -- \
mklabel gpt \
mkpart '" "' ext4 1MiB 2MiB \
mkpart '" "' xfs 2MiB -2GiB \
mkpart '" "' ext4 -2GiB 100% \
set 1 bios_grub on
update_part /dev/$xda
echo #1 bios_boot
echo #2 os 用目标系统的格式化工具
mkfs.ext4 -F -L installer /dev/$xda*3 #3 installer
fi
2023-09-16 20:00:07 +08:00
else
# gentoo 的镜像解压后是 3.5g,因此设置 installer 分区 1g这样才能在5g硬盘上安装
[ "$distro" = gentoo ] && installer_part_size=1GiB || installer_part_size=2GiB
parted /dev/$xda -s -- \
mklabel gpt \
mkpart '" "' ext4 1MiB -$installer_part_size \
mkpart '" "' ext4 -$installer_part_size 100%
update_part /dev/$xda
mkfs.ext4 -F -L os /dev/$xda*1 #1 os
mkfs.ext4 -F -L installer /dev/$xda*2 #2 installer
2023-09-16 20:00:05 +08:00
fi
2023-07-25 00:21:08 +08:00
else
# 安装红帽系或ubuntu
2023-07-25 00:21:08 +08:00
# 对于红帽系是临时分区表,安装时除了 installer 分区,其他分区会重建为默认的大小
# 对于ubuntu是最终分区表因为 ubuntu 的安装器不能调整个别分区,只能重建整个分区表
# installer 2g分区用fat格式刚好塞得下ubuntu-22.04.3 iso而ext4塞不下或者需要改参数
apk add dosfstools
2023-07-25 00:21:08 +08:00
if is_efi; then
# efi
parted /dev/$xda -s -- \
mklabel gpt \
mkpart '" "' fat32 1MiB 1025MiB \
mkpart '" "' ext4 1025MiB -2GiB \
mkpart '" "' ext4 -2GiB 100% \
set 1 boot on
update_part /dev/$xda
2023-09-16 20:00:05 +08:00
mkfs.fat -n efi /dev/$xda*1 #1 efi
mkfs.ext4 -F -L os /dev/$xda*2 #2 os
mkfs.fat -n installer /dev/$xda*3 #3 installer
2023-07-25 00:21:08 +08:00
elif is_xda_gt_2t; then
# bios > 2t
parted /dev/$xda -s -- \
mklabel gpt \
mkpart '" "' ext4 1MiB 2MiB \
mkpart '" "' ext4 2MiB -2GiB \
mkpart '" "' ext4 -2GiB 100% \
set 1 bios_grub on
update_part /dev/$xda
2023-09-16 20:00:05 +08:00
echo #1 bios_boot
mkfs.ext4 -F -L os /dev/$xda*2 #2 os
mkfs.fat -n installer /dev/$xda*3 #3 installer
2023-07-25 00:21:08 +08:00
else
# bios
parted /dev/$xda -s -- \
mklabel msdos \
mkpart primary ext4 1MiB -2GiB \
mkpart primary ext4 -2GiB 100% \
set 1 boot on
update_part /dev/$xda
2023-09-16 20:00:05 +08:00
mkfs.ext4 -F -L os /dev/$xda*1 #1 os
mkfs.fat -n installer /dev/$xda*2 #2 installer
2023-07-25 00:21:08 +08:00
fi
update_part /dev/$xda
# centos 7 无法加载alpine格式化的ext4
# 要关闭这个属性
# 目前改用fat格式不用设置这个
if false && [ "$distro" = centos ]; then
apk add e2fsprogs-extra
tune2fs -O ^metadata_csum_seed /dev/disk/by-label/installer
fi
2023-05-25 20:15:12 +08:00
fi
2023-07-25 00:21:08 +08:00
update_part /dev/$xda
}
2023-05-03 22:22:21 +08:00
2023-07-28 21:27:16 +08:00
mount_pseudo_fs() {
os_dir=$1
# https://wiki.archlinux.org/title/Chroot#Using_chroot
2023-09-16 20:00:05 +08:00
mount -t proc /proc $os_dir/proc/
mount -t sysfs /sys $os_dir/sys/
mount --rbind /dev $os_dir/dev/
mount --rbind /run $os_dir/run/
2023-07-28 21:27:16 +08:00
if is_efi; then
2023-09-16 20:00:05 +08:00
mount --rbind /sys/firmware/efi/efivars $os_dir/sys/firmware/efi/efivars/
2023-07-28 21:27:16 +08:00
fi
}
create_cloud_init_network_config() {
ci_file=$1
get_netconf_to mac_addr
# TODO: 没在windows下获取网络信息,先跳过cloud-init网络不然网络不通
[ -z "$mac_addr" ] && return
get_netconf_to ipv4_addr
get_netconf_to ipv4_gateway
get_netconf_to ipv6_addr
get_netconf_to ipv6_gateway
# shellcheck disable=SC2154
is_dhcpv4 && dhcp4=true || dhcp4=false
(is_dhcpv6 || is_slaac) && dhcp6=true || dhcp6=false
cat <<EOF >>$ci_file
network:
version: 2
ethernets:
eth0:
match:
macaddress: "$mac_addr"
dhcp4: $dhcp4
dhcp6: $dhcp6
addresses:
- $ipv4_addr @ipv4_addr
- $ipv6_addr @ipv6_addr
gateway4: $ipv4_gateway @ipv4_gateway
gateway6: $ipv6_gateway @ipv6_gateway
nameservers:
addresses: @dns_addrs
EOF
if is_dhcpv4 || [ -z "$ipv4_addr" ]; then
sed -i '/@ipv4/d' $ci_file
fi
if is_slaac || is_dhcpv6 || [ -z "$ipv6_addr" ]; then
sed -i '/@ipv6/d' $ci_file
fi
if ! grep '@ipv._addr' $ci_file; then
sed -i '/addresses:/d' $ci_file
fi
# dns
dns_list=$(grep '^nameserver' /etc/resolv.conf | awk '{print $2}')
unset dns4_list dns6_list
if ! is_dhcpv4; then
dns4_list=$(echo "$dns_list" | grep '\.' || true)
fi
if ! (is_slaac || is_dhcpv6); then
dns6_list=$(echo "$dns_list" | grep ':' || true)
fi
is_have_dns=false
for cur in $dns4_list $dns6_list; do
is_have_dns=true
echo " - $cur" >>$ci_file
done
if ! $is_have_dns; then
sed -i '/nameservers:/d' $ci_file
sed -i '/@dns_addrs/d' $ci_file
fi
sed -Ei "s/@(ip|dns).*//" $ci_file
}
2023-07-28 21:27:16 +08:00
download_cloud_init_config() {
os_dir=$1
ci_file=$os_dir/etc/cloud/cloud.cfg.d/99_nocloud.cfg
2023-07-28 21:27:16 +08:00
# shellcheck disable=SC2154
download $confhome/nocloud.yaml $ci_file
# swapfile
2023-09-21 00:12:35 +08:00
# 如果分区表中已经有swapfile就跳过例如arch
if ! grep -w swap $os_dir/etc/fstab; then
2023-07-28 21:27:16 +08:00
# btrfs
if mount | grep 'on /os type btrfs'; then
2023-09-16 20:00:05 +08:00
insert_into_file $ci_file after '^runcmd:' <<EOF
2023-07-28 21:27:16 +08:00
- btrfs filesystem mkswapfile --size 1G /swapfile
- swapon /swapfile
- echo "/swapfile none swap defaults 0 0" >> /etc/fstab
EOF
else
# ext4 xfs
cat <<EOF >>$ci_file
swap:
filename: /swapfile
size: auto
EOF
fi
fi
create_cloud_init_network_config $ci_file
cat -n $ci_file
2023-07-28 21:27:16 +08:00
}
2023-08-03 22:38:38 +08:00
modify_dd_os() {
find_and_mount() {
mount_point=$1
mount_dev=$(awk "\$2==\"$mount_point\" {print \$1}" $os_dir/etc/fstab)
if [ -n "$mount_dev" ]; then
mount $mount_dev $os_dir$mount_point
fi
}
2023-08-03 22:38:38 +08:00
apk add lsblk
mkdir -p /os
# 按分区容量大到小,依次寻找系统分区
for part in $(lsblk /dev/$xda --sort SIZE -no NAME | sed '$d' | tac); do
# btrfs挂载的是默认子卷如果没有默认子卷挂载的是根目录
# fedora 云镜像没有默认子卷且系统在root子卷中
if mount /dev/$part /os; then
if etc_dir=$({ ls -d /os/etc || ls -d /os/*/etc; } 2>/dev/null); then
os_dir=$(dirname $etc_dir)
break
fi
umount /os
fi
done
if [ -z "$os_dir" ]; then
error_and_exit "can't find os partition"
fi
download_cloud_init_config $os_dir
2023-09-10 22:23:04 +08:00
2023-09-16 20:00:05 +08:00
# 为红帽系禁用 selinux kdump
2023-08-03 22:38:38 +08:00
if [ -f $os_dir/etc/redhat-release ]; then
find_and_mount /boot
find_and_mount /boot/efi
2023-08-03 22:38:38 +08:00
disable_selinux_kdump $os_dir
2023-09-16 20:00:05 +08:00
fi
# opensuse tumbleweed 需安装 wicked
if grep opensuse-tumbleweed $os_dir/etc/os-release; then
2023-09-10 22:23:04 +08:00
cp -f /etc/resolv.conf $os_dir/etc/resolv.conf
mount_pseudo_fs $os_dir
chroot $os_dir zypper install -y wicked
rm -f $os_dir/etc/resolv.conf
2023-08-03 22:38:38 +08:00
fi
2023-09-16 20:00:07 +08:00
# gentoo
if [ -f $os_dir/etc/gentoo-release ]; then
# 挂载伪文件系统
mount_pseudo_fs $os_dir
# 在这里修改密码而不是用cloud-init因为我们的默认密码太弱
sed -i 's/enforce=everyone/enforce=none/' $os_dir/etc/security/passwdqc.conf
echo 'root:123@@@' | chroot $os_dir chpasswd >/dev/null
sed -i 's/enforce=none/enforce=everyone/' $os_dir/etc/security/passwdqc.conf
# 下载仓库,选择 profile
2023-09-17 11:34:53 +08:00
if [ ! -d $os_dir/var/db/repos/gentoo/ ]; then
2023-09-16 20:00:07 +08:00
cp -f /etc/resolv.conf $os_dir/etc/resolv.conf
chroot $os_dir emerge-webrsync
profile=$(chroot $os_dir eselect profile list | grep '/[0-9\.]*/systemd (stable)' | awk '{print $2}')
chroot $os_dir eselect profile set $profile
fi
# 删除 resolv.conf不然 systemd-resolved 无法创建软链接
rm -f $os_dir/etc/resolv.conf
# 启用网络服务
chroot $os_dir systemctl enable systemd-networkd
chroot $os_dir systemctl enable systemd-resolved
# systemd-networkd 有时不会运行
# https://bugs.gentoo.org/910404 补丁好像没用
# https://github.com/systemd/systemd/issues/27718#issuecomment-1564877478
# 临时的解决办法是运行 networkctl如果启用了systemd-networkd服务会运行服务
insert_into_file $os_dir/lib/systemd/system/systemd-logind.service after '\[Service\]' <<EOF
ExecStartPost=-networkctl
EOF
fi
2023-08-03 22:38:38 +08:00
}
2023-07-28 21:27:16 +08:00
install_cloud_image_by_dd() {
apk add util-linux udev hdparm curl
rc-service udev start
install_dd
update_part /dev/$xda
2023-08-03 22:38:38 +08:00
modify_dd_os
2023-07-28 21:27:16 +08:00
}
2023-07-30 00:35:32 +08:00
create_swap() {
swapfile=$1
if ! grep $swapfile /proc/swaps; then
apk add util-linux
ram_size=$(lsmem -b 2>/dev/null | grep 'Total online memory:' | awk '{ print $NF/1024/1024 }')
if [ -z $ram_size ] || [ $ram_size -lt 1024 ]; then
fallocate -l 512M $swapfile
chmod 0600 $swapfile
mkswap $swapfile
swapon $swapfile
fi
fi
}
disable_selinux_kdump() {
os_dir=$1
releasever=$(awk -F: '{ print $5 }' <$os_dir/etc/system-release-cpe)
if ! chroot $os_dir command -v grubby; then
if [ "$releasever" = 7 ]; then
chroot $os_dir yum -y --disablerepo=* --enablerepo=base,updates grubby
else
chroot $os_dir dnf -y --disablerepo=* --enablerepo=baseos --setopt=install_weak_deps=False grubby
fi
fi
# selinux
sed -i 's/^SELINUX=enforcing/SELINUX=disabled/g' $os_dir/etc/selinux/config
# https://access.redhat.com/solutions/3176
if [ "$releasever" -ge 9 ]; then
chroot $os_dir grubby --update-kernel ALL --args selinux=0
fi
# kdump
chroot $os_dir grubby --update-kernel ALL --args crashkernel=no
if [ "$releasever" -eq 7 ]; then
# el7 上面那条 grubby 命令不能设置 /etc/default/grub
sed -iE 's/crashkernel=[^ "]*/crashkernel=no/' $os_dir/etc/default/grub
fi
rm -rf $os_dir/etc/systemd/system/multi-user.target.wants/kdump.service
}
2023-07-25 00:21:08 +08:00
install_cloud_image() {
2023-07-22 22:02:33 +08:00
apk add qemu-img lsblk
2023-07-18 00:22:51 +08:00
mkdir -p /installer
mount /dev/disk/by-label/installer /installer
qcow_file=/installer/cloud_image.qcow2
download $img $qcow_file
2023-07-28 21:27:16 +08:00
# centos/alma/rocky cloud image系统分区是8~9g xfs而我们的目标是能在5g硬盘上运行因此改成复制系统文件
if [ "$distro" = "centos" ] || [ "$distro" = "alma" ] || [ "$distro" = "rocky" ]; then
2023-07-30 00:35:32 +08:00
yum() {
if [ "$releasever" = 7 ]; then
chroot /os/ yum -y --disablerepo=* --enablerepo=base,updates "$@"
else
chroot /os/ dnf -y --disablerepo=* --enablerepo=baseos --setopt=install_weak_deps=False "$@"
fi
}
2023-07-18 00:22:51 +08:00
modprobe nbd
qemu_nbd -c /dev/nbd0 $qcow_file
2023-07-28 21:27:16 +08:00
# TODO: 改成循环mount找出os+fstab查找剩余分区
2023-07-30 00:35:32 +08:00
os_part=$(lsblk /dev/nbd0p*[0-9] --sort SIZE -no NAME,FSTYPE | grep xfs | tail -1 | cut -d' ' -f1)
efi_part=$(lsblk /dev/nbd0p*[0-9] --sort SIZE -no NAME,FSTYPE | grep fat | tail -1 | cut -d' ' -f1)
boot_part=$(lsblk /dev/nbd0p*[0-9] --sort SIZE -no NAME,FSTYPE | grep xfs | sed '$d' | tail -1 | cut -d' ' -f1)
os_part_uuid=$(lsblk /dev/$os_part -no UUID)
if [ -n "$efi_part" ]; then
efi_part_uuid=$(lsblk /dev/$efi_part -no UUID)
fi
2023-07-28 21:27:16 +08:00
mkdir -p /nbd /nbd-boot /nbd-efi /os
# 使用目标系统的格式化程序
# centos8 如果用alpine格式化xfsgrub2-mkconfig和grub2里面都无法识别xfs分区
mount -o nouuid /dev/$os_part /nbd/
mount_pseudo_fs /nbd/
2023-08-03 22:03:51 +08:00
chroot /nbd mkfs.xfs -f -m uuid=$os_part_uuid /dev/$xda*2
2023-07-28 21:27:16 +08:00
umount -R /nbd/
# 复制系统
2023-07-30 00:35:32 +08:00
echo Copying os partition
2023-07-28 21:27:16 +08:00
mount -o ro,nouuid /dev/$os_part /nbd/
2023-08-03 22:03:51 +08:00
mount -o noatime /dev/$xda*2 /os/
2023-07-28 21:27:16 +08:00
cp -a /nbd/* /os/
# 复制boot分区如果有
if [ -n "$boot_part" ]; then
2023-07-30 00:35:32 +08:00
echo Copying boot partition
2023-07-28 21:27:16 +08:00
mount -o ro,nouuid /dev/$boot_part /nbd-boot/
cp -a /nbd-boot/* /os/boot/
fi
2023-07-30 00:35:32 +08:00
# efi 分区
efi_mount_opts="defaults,uid=0,gid=0,umask=077,shortname=winnt"
if is_efi; then
2023-07-30 00:35:32 +08:00
# 挂载 efi
mkdir -p /os/boot/efi/
mount -o $efi_mount_opts /dev/$xda*1 /os/boot/efi/
2023-07-28 21:27:16 +08:00
# 复制文件
2023-07-28 21:27:16 +08:00
if [ -n "$efi_part" ]; then
2023-07-30 00:35:32 +08:00
echo Copying efi partition
2023-07-28 21:27:16 +08:00
mount -o ro /dev/$efi_part /nbd-efi/
cp -a /nbd-efi/* /os/boot/efi/
fi
fi
2023-07-30 00:35:32 +08:00
# 取消挂载 nbd
2023-07-28 21:27:16 +08:00
umount /nbd/ /nbd-boot/ /nbd-efi/ || true
qemu_nbd -d /dev/nbd0
2023-07-30 00:35:32 +08:00
# 如果镜像有efi分区复制其uuid
# 如果有相同uuid的fat分区则无法挂载
# 所以要先复制efi分区断开nbd再复制uuid
if is_efi && [ -n "$efi_part_uuid" ]; then
umount /os/boot/efi/
apk add mtools
mlabel -N "$(echo $efi_part_uuid | sed 's/-//')" -i /dev/$xda*1
update_part /dev/$xda
mount -o $efi_mount_opts /dev/$xda*1 /os/boot/efi/
fi
# 挂载伪文件系统
mount_pseudo_fs /os/
2023-07-30 00:35:32 +08:00
# 创建 swap
rm -rf /installer/*
create_swap /installer/swapfile
# resolv.conf
mv /os/etc/resolv.conf /os/etc/resolv.conf.orig
cp /etc/resolv.conf /os/etc/resolv.conf
# selinux kdump
disable_selinux_kdump /os
2023-08-03 22:38:38 +08:00
# cloud-init
download_cloud_init_config /os
2023-07-28 21:27:16 +08:00
# fstab 删除 boot 分区
# alma/rocky 镜像本身有boot分区但我们不需要
sed -i '/[[:blank:]]\/boot[[:blank:]]/d' /os/etc/fstab
2023-07-28 21:27:16 +08:00
# fstab 添加 efi 分区
if is_efi; then
2023-07-30 00:35:32 +08:00
# centos 要创建efi条目
2023-07-28 21:27:16 +08:00
if ! grep /boot/efi /os/etc/fstab; then
efi_part_uuid=$(lsblk /dev/$xda*1 -no UUID)
2023-07-30 00:35:32 +08:00
echo "UUID=$efi_part_uuid /boot/efi vfat $efi_mount_opts 0 0" >>/os/etc/fstab
2023-07-28 21:27:16 +08:00
fi
else
# 删除 efi 条目
sed -i '/[[:blank:]]\/boot\/efi[[:blank:]]/d' /os/etc/fstab
fi
2023-07-30 00:35:32 +08:00
distro_full=$(awk -F: '{ print $3 }' </os/etc/system-release-cpe)
releasever=$(awk -F: '{ print $5 }' </os/etc/system-release-cpe)
2023-07-28 21:27:16 +08:00
2023-07-30 00:35:32 +08:00
remove_grub_conflict_files() {
# bios 和 efi 转换前先删除
# bios转efi出错
# centos7是bios镜像/boot/grub2/grubenv 是真身
# 安装grub-efi时grubenv 会改成指向efi分区grubenv软连接
# 如果安装grub-efi前没有删除原来的grubenv原来的grubenv将不变新建的软连接将变成 grubenv.rpmnew
# 后续grubenv的改动无法同步到efi分区会造成grub2-setdefault失效
# efi转bios出错
# 如果是指向efi目录的软连接例如el8先删除它否则 grub2-install 会报错
rm -rf /os/boot/grub2/grubenv /os/boot/grub2/grub.cfg
}
# 安装 efi 引导
# 只有centos镜像没有efi其他系统镜像已经从efi分区复制了文件
if [ "$distro" = "centos" ] && is_efi; then
remove_grub_conflict_files
[ "$(uname -m)" = x86_64 ] && arch=x64 || arch=aa64
yum install efibootmgr grub2-efi-$arch grub2-efi-$arch-modules shim-$arch
fi
# 安装 bios 引导
if ! is_efi; then
remove_grub_conflict_files
yum install grub2-pc grub2-pc-modules
chroot /os/ grub2-install /dev/$xda
fi
2023-07-28 21:27:16 +08:00
# blscfg 启动项
# rocky/alma镜像是独立的boot分区但我们不是
# 因此要添加boot目录
if [ -d /os/boot/loader/entries/ ] && ! grep -q 'initrd /boot/' /os/boot/loader/entries/*.conf; then
sed -i -E 's,((linux|initrd) /),\1boot/,g' /os/boot/loader/entries/*.conf
fi
2023-07-30 00:35:32 +08:00
# efi 分区 grub.cfg
# https://github.com/rhinstaller/anaconda/blob/346b932a26a19b339e9073c049b08bdef7f166c3/pyanaconda/modules/storage/bootloader/efi.py#L198
if is_efi && [ "$releasever" -ge 9 ]; then
cat <<EOF >/os/boot/efi/EFI/$distro_full/grub.cfg
search --no-floppy --fs-uuid --set=dev $os_part_uuid
set prefix=(\$dev)/boot/grub2
export \$prefix
configfile \$prefix/grub.cfg
EOF
fi
# 主 grub.cfg
if is_efi && [ "$releasever" -le 8 ]; then
chroot /os/ grub2-mkconfig -o /boot/efi/EFI/$distro_full/grub.cfg
2023-07-28 21:27:16 +08:00
else
chroot /os/ grub2-mkconfig -o /boot/grub2/grub.cfg
fi
2023-05-03 22:22:21 +08:00
# 还原 resolv.conf
mv /os/etc/resolv.conf.orig /os/etc/resolv.conf
2023-07-30 00:35:32 +08:00
# 删除installer分区重启后cloud init会自动扩容
swapoff -a
umount /installer
parted /dev/$xda -s rm 3
2023-07-18 00:22:51 +08:00
else
2023-09-16 20:00:07 +08:00
# debian ubuntu arch opensuse gentoo
if true; then
modprobe nbd
qemu_nbd -c /dev/nbd0 $qcow_file
2023-08-01 21:51:36 +08:00
# 检查最后一个分区是否是 btrfs
# 即使awk结果为空返回值也是0加上 grep . 检查是否结果为空
if part_num=$(parted /dev/nbd0 -s print | awk NF | tail -1 | grep btrfs | awk '{print $1}' | grep .); then
apk add btrfs-progs
mkdir -p /mnt/btrfs
mount /dev/nbd0p$part_num /mnt/btrfs
# 回收空数据块
btrfs device usage /mnt/btrfs
btrfs balance start -dusage=0 /mnt/btrfs
btrfs device usage /mnt/btrfs
# 计算可以缩小的空间
free_bytes=$(btrfs device usage /mnt/btrfs -b | grep Unallocated: | awk '{print $2}')
reserve_bytes=$((100 * 1024 * 1024)) # 预留 100M 可用空间
skrink_bytes=$((free_bytes - reserve_bytes))
if [ $skrink_bytes -gt 0 ]; then
# 缩小文件系统
btrfs filesystem resize -$skrink_bytes /mnt/btrfs
# 缩小分区
part_start=$(parted /dev/nbd0 -s 'unit b print' | awk "\$1==$part_num {print \$2}" | sed 's/B//')
part_size=$(btrfs filesystem usage /mnt/btrfs -b | grep 'Device size:' | awk '{print $3}')
part_end=$((part_start + part_size))
umount /mnt/btrfs
printf "yes" | parted /dev/nbd0 resizepart $part_num ${part_end}B ---pretend-input-tty
# 缩小 qcow2
qemu_nbd -d /dev/nbd0
qemu-img resize --shrink $qcow_file $part_end
# 重新连接
qemu_nbd -c /dev/nbd0 $qcow_file
else
umount /mnt/btrfs
fi
fi
2023-08-03 22:38:38 +08:00
# 显示分区
2023-09-16 20:00:05 +08:00
lsblk -o NAME,SIZE,FSTYPE,LABEL /dev/nbd0
2023-08-03 22:38:38 +08:00
# 将前1M dd到内存
dd if=/dev/nbd0 of=/first-1M bs=1M count=1
# 将1M之后 dd到硬盘
# shellcheck disable=SC2194
case 3 in
1)
# BusyBox dd
dd if=/dev/nbd0 of=/dev/$xda bs=1M skip=1 seek=1
;;
2)
# 用原版 dd status=progress但没有进度和剩余时间
apk add coreutils
dd if=/dev/nbd0 of=/dev/$xda bs=1M skip=1 seek=1 status=progress
;;
3)
# 用 pv
apk add pv
pv -f /dev/nbd0 | dd of=/dev/$xda bs=1M skip=1 seek=1 iflag=fullblock
;;
esac
qemu_nbd -d /dev/nbd0
else
# 将前1M dd到内存将1M之后 dd到硬盘
qemu-img dd if=$qcow_file of=/first-1M bs=1M count=1
qemu-img dd if=$qcow_file of=/dev/disk/by-label/os bs=1M skip=1
fi
2023-07-18 00:22:51 +08:00
# 将前1M从内存 dd 到硬盘
umount /installer/
dd if=/first-1M of=/dev/$xda
update_part /dev/$xda
2023-07-22 22:02:33 +08:00
2023-09-16 20:00:07 +08:00
# 提前扩容
# 1 修复 vultr 512m debian 10/11 generic/genericcloud 首次启动 kernel panic
# 2 修复 gentoo websync 时空间不足
if [ "$distro" = debian ] || [ "$distro" = gentoo ]; then
apk add parted
if parted /dev/$xda -s print 2>&1 | grep 'Not all of the space'; then
printf "fix" | parted /dev/$xda print ---pretend-input-tty
2023-09-16 20:00:07 +08:00
# TODO: 获取 ext4 分区编号
[ "$distro" = debian ] && system_part_num=1 || system_part_num=3
printf "yes" | parted /dev/$xda resizepart $system_part_num 100% ---pretend-input-tty
update_part /dev/$xda
2023-09-16 20:00:07 +08:00
if [ "$distro" = gentoo ]; then
apk add e2fsprogs-extra
e2fsck -p -f /dev/$xda$system_part_num
resize2fs /dev/$xda$system_part_num
fi
fi
fi
2023-08-03 22:38:38 +08:00
modify_dd_os
fi
2023-07-25 00:21:08 +08:00
}
mount_part() {
2023-07-18 00:22:51 +08:00
# 挂载主分区
mkdir -p /os
mount /dev/disk/by-label/os /os
# 挂载其他分区
mkdir -p /os/boot/efi
if is_efi; then
mount /dev/disk/by-label/efi /os/boot/efi
fi
mkdir -p /os/installer
if [ "$distro" = windows ]; then
mount_args="-t ntfs3"
fi
mount $mount_args /dev/disk/by-label/installer /os/installer
2023-07-25 00:21:08 +08:00
}
2023-05-03 22:22:21 +08:00
2023-07-25 00:21:08 +08:00
install_windows() {
# shellcheck disable=SC2154
2023-05-25 20:15:12 +08:00
download $iso /os/windows.iso
mkdir -p /iso
2023-05-25 20:15:12 +08:00
mount /os/windows.iso /iso
# 从iso复制文件
# efi: 复制boot开头的文件+efi目录到efi分区复制iso全部文件(除了boot.wim)到installer分区
# bios: 复制iso全部文件到installer分区
2023-07-06 22:20:09 +08:00
if is_efi; then
mkdir -p /os/boot/efi/sources/
cp -rv /iso/boot* /os/boot/efi/
cp -rv /iso/efi/ /os/boot/efi/
cp -rv /iso/sources/boot.wim /os/boot/efi/sources/
rsync -rv --exclude=/sources/boot.wim /iso/* /os/installer/
boot_wim=/os/boot/efi/sources/boot.wim
else
rsync -rv /iso/* /os/installer/
boot_wim=/os/installer/sources/boot.wim
fi
if [ -e /os/installer/sources/install.esd ]; then
install_wim=/os/installer/sources/install.esd
else
install_wim=/os/installer/sources/install.wim
fi
# 匹配映像版本
# 需要整行匹配,因为要区分 Windows 10 Pro 和 Windows 10 Pro for Workstations
# TODO: 如果无法匹配,等待用户输入?安装第一个?
image_count=$(wiminfo $install_wim | grep "Image Count:" | cut -d: -f2 | xargs)
if [ "$image_count" = 1 ]; then
# 只有一个版本就使用第一个版本
image_name=$(wiminfo $install_wim | grep -ix "Name:[[:blank:]]*.*" | cut -d: -f2 | xargs)
else
# 否则改成正确的大小写
image_name=$(wiminfo $install_wim | grep -ix "Name:[[:blank:]]*$image_name" | cut -d: -f2 | xargs)
fi
2023-07-06 22:20:09 +08:00
is_win7_or_win2008r2() {
echo $image_name | grep -iEw '^Windows (7|Server 2008 R2)'
2023-07-06 22:20:09 +08:00
}
2023-07-03 23:11:10 +08:00
# 变量名 使用场景
# arch_uname uname -m x86_64 aarch64
# arch_wim wiminfo x86 x86_64 ARM64
# arch virtio驱动/unattend.xml x86 amd64 arm64
# arch_xen xen驱动 x86 x64
2023-06-05 19:54:00 +08:00
# 将 wim 的 arch 转为驱动和应答文件的 arch
arch_wim=$(wiminfo $install_wim 1 | grep Architecture: | awk '{print $2}' | tr '[:upper:]' '[:lower:]')
2023-06-05 19:54:00 +08:00
case "$arch_wim" in
2023-07-03 23:11:10 +08:00
x86)
arch=x86
arch_xen=x86
;;
x86_64)
arch=amd64
arch_xen=x64
;;
arm64)
arch=arm64
arch_xen= # xen 没有 arm64 驱动
;;
2023-06-05 19:54:00 +08:00
esac
# virt-what 要用最新版
# vultr 1G High Frequency LAX 实际上是 kvm
# debian 11 virt-what 1.19 显示为 hyperv qemu
# debian 11 systemd-detect-virt 显示为 microsoft
# alpine virt-what 1.25 显示为 kvm
# 所以不要在原系统上判断具体虚拟化环境
# lscpu 也可查看虚拟化环境,但 alpine on lightsail 运行结果为 Microsoft
# 猜测 lscpu 只参考了 cpuid 没参考 dmi
2023-05-25 20:15:12 +08:00
# 下载 virtio 驱动
2023-07-03 23:11:10 +08:00
# virt-what 可能会输出多行结果,因此用 grep
drv=/os/drivers
mkdir -p $drv
if virt-what | grep aws &&
virt-what | grep kvm &&
[ "$arch_wim" = x86_64 ]; then
2023-07-03 23:11:10 +08:00
# aws nitro
# 只有 x64 位驱动
2023-07-03 13:51:59 +08:00
# https://docs.aws.amazon.com/zh_cn/AWSEC2/latest/WindowsGuide/migrating-latest-types.html
apk add unzip
2023-07-06 22:20:09 +08:00
if is_win7_or_win2008r2; then
download https://s3.amazonaws.com/ec2-windows-drivers-downloads/NVMe/1.3.2/AWSNVMe.zip $drv/AWSNVMe.zip
download https://s3.amazonaws.com/ec2-windows-drivers-downloads/ENA/x64/2.2.3/AwsEnaNetworkDriver.zip $drv/AwsEnaNetworkDriver.zip
else
download https://s3.amazonaws.com/ec2-windows-drivers-downloads/NVMe/Latest/AWSNVMe.zip $drv/AWSNVMe.zip
download https://s3.amazonaws.com/ec2-windows-drivers-downloads/ENA/Latest/AwsEnaNetworkDriver.zip $drv/AwsEnaNetworkDriver.zip
fi
2023-07-03 23:11:10 +08:00
unzip -o -d $drv/aws/ $drv/AWSNVMe.zip
unzip -o -d $drv/aws/ $drv/AwsEnaNetworkDriver.zip
elif virt-what | grep aws &&
virt-what | grep xen &&
[ "$arch_wim" = x86_64 ]; then
2023-07-03 23:11:10 +08:00
# aws xen
# 只有 64 位驱动
# 未测试
# https://docs.aws.amazon.com/zh_cn/AWSEC2/latest/WindowsGuide/Upgrading_PV_drivers.html
apk add unzip msitools
2023-07-06 22:20:09 +08:00
if is_win7_or_win2008r2; then
download https://s3.amazonaws.com/ec2-windows-drivers-downloads/AWSPV/8.3.5/AWSPVDriver.zip $drv/AWSPVDriver.zip
else
download https://s3.amazonaws.com/ec2-windows-drivers-downloads/AWSPV/Latest/AWSPVDriver.zip $drv/AWSPVDriver.zip
fi
2023-07-03 23:11:10 +08:00
unzip -o -d $drv $drv/AWSPVDriver.zip
msiextract $drv/AWSPVDriverSetup.msi -C $drv
mkdir -p $drv/aws/
cp -rf $drv/.Drivers/* $drv/aws/
elif virt-what | grep xen &&
[ "$arch_wim" != arm64 ]; then
2023-07-03 23:11:10 +08:00
# xen
# 有 x86 x64没arm64驱动
2023-07-03 23:11:10 +08:00
# https://xenbits.xenproject.org/pvdrivers/win/
ver='9.0.0'
2023-07-18 00:01:07 +08:00
# 在 aws t2 上测试,安装 xenbus 会蓝屏装了其他7个驱动后能进系统但没网络
# 但 aws 应该用aws官方xen驱动所以测试仅供参考
2023-07-03 23:11:10 +08:00
parts='xenbus xencons xenhid xeniface xennet xenvbd xenvif xenvkbd'
mkdir -p $drv/xen/
for part in $parts; do
download https://xenbits.xenproject.org/pvdrivers/win/$ver/$part.tar $drv/$part.tar
tar -xf $drv/$part.tar -C $drv/xen/
done
2023-07-03 13:51:59 +08:00
elif virt-what | grep kvm; then
2023-07-03 23:11:10 +08:00
# virtio
# x86 x64 arm64 都有
2023-07-03 23:11:10 +08:00
# https://fedorapeople.org/groups/virt/virtio-win/direct-downloads/stable-virtio/
2023-05-25 20:15:12 +08:00
case $(echo "$image_name" | tr '[:upper:]' '[:lower:]') in
'windows server 2022'*) sys=2k22 ;;
'windows server 2019'*) sys=2k19 ;;
'windows server 2016'*) sys=2k16 ;;
'windows server 2012 R2'*) sys=2k12R2 ;;
'windows server 2012'*) sys=2k12 ;;
'windows server 2008 R2'*) sys=2k8R2 ;;
'windows server 2008'*) sys=2k8 ;;
'windows 11'*) sys=w11 ;;
'windows 10'*) sys=w10 ;;
'windows 8.1'*) sys=w8.1 ;;
'windows 8'*) sys=w8 ;;
'windows 7'*) sys=w7 ;;
2023-06-18 21:27:22 +08:00
'windows vista'*) sys=2k8 ;; # virtio 没有 vista 专用驱动
2023-05-25 20:15:12 +08:00
esac
2023-06-04 19:05:17 +08:00
case "$sys" in
# https://github.com/virtio-win/virtio-win-pkg-scripts/issues/40
w7) dir=archive-virtio/virtio-win-0.1.173-9 ;;
# https://github.com/virtio-win/virtio-win-pkg-scripts/issues/61
2k12*) dir=archive-virtio/virtio-win-0.1.215-1 ;;
*) dir=stable-virtio ;;
esac
2023-07-03 23:11:10 +08:00
download https://fedorapeople.org/groups/virt/virtio-win/direct-downloads/$dir/virtio-win.iso $drv/virtio-win.iso
mkdir -p $drv/virtio
mount $drv/virtio-win.iso $drv/virtio
2023-05-25 20:15:12 +08:00
fi
# 修改应答文件
2023-07-22 21:19:22 +08:00
download $confhome/windows.xml /tmp/Autounattend.xml
2023-05-28 00:52:48 +08:00
locale=$(wiminfo $install_wim | grep 'Default Language' | head -1 | awk '{print $NF}')
2023-06-05 19:54:00 +08:00
sed -i "s|%arch%|$arch|; s|%image_name%|$image_name|; s|%locale%|$locale|" /tmp/Autounattend.xml
2023-05-28 00:52:48 +08:00
# 修改应答文件,分区配置
2023-07-06 22:20:09 +08:00
if is_efi; then
2023-05-28 00:52:48 +08:00
sed -i "s|%installto_partitionid%|3|" /tmp/Autounattend.xml
2023-09-16 20:00:05 +08:00
insert_into_file /tmp/Autounattend.xml after '<ModifyPartitions>' <<EOF
2023-05-28 00:52:48 +08:00
<ModifyPartition wcm:action="add">
<Order>1</Order>
<PartitionID>1</PartitionID>
<Format>FAT32</Format>
</ModifyPartition>
<ModifyPartition wcm:action="add">
<Order>2</Order>
<PartitionID>2</PartitionID>
</ModifyPartition>
<ModifyPartition wcm:action="add">
<Order>3</Order>
<PartitionID>3</PartitionID>
<Format>NTFS</Format>
</ModifyPartition>
EOF
else
sed -i "s|%installto_partitionid%|1|" /tmp/Autounattend.xml
2023-09-16 20:00:05 +08:00
insert_into_file /tmp/Autounattend.xml after '<ModifyPartitions>' <<EOF
2023-05-28 00:52:48 +08:00
<ModifyPartition wcm:action="add">
<Order>1</Order>
<PartitionID>1</PartitionID>
<Format>NTFS</Format>
</ModifyPartition>
EOF
fi
unix2dos /tmp/Autounattend.xml
2023-05-25 20:15:12 +08:00
2023-06-04 19:09:40 +08:00
# # ei.cfg
# cat <<EOF >/os/installer/sources/ei.cfg
# [Channel]
# OEM
# EOF
# unix2dos /os/installer/sources/ei.cfg
2023-05-28 00:52:48 +08:00
# 挂载 boot.wim
mkdir -p /wim
2023-05-28 00:52:48 +08:00
wimmountrw $boot_wim 2 /wim/
2023-05-25 20:15:12 +08:00
2023-07-03 23:11:10 +08:00
cp_drivers() {
src=$1
dist=$2
path=$3
[ -n "$path" ] && filter="-ipath $path" || filter=""
find $src \
$filter \
-type f \
2023-07-03 23:11:10 +08:00
-not -iname "*.pdb" \
-not -iname "dpinst.exe" \
-exec /bin/cp -rfv {} $dist \;
2023-07-03 23:11:10 +08:00
}
# 添加驱动
mkdir -p /wim/drivers
[ -d $drv/virtio ] && cp_drivers $drv/virtio /wim/drivers "*/$sys/$arch/*"
[ -d $drv/aws ] && cp_drivers $drv/aws /wim/drivers
[ -d $drv/xen ] && cp_drivers $drv/xen /wim/drivers "*/$arch_xen/*"
2023-05-25 20:15:12 +08:00
# win7 要添加 bootx64.efi 到 efi 目录
2023-06-05 19:54:00 +08:00
[ $arch = amd64 ] && boot_efi=bootx64.efi || boot_efi=bootaa64.efi
2023-07-06 22:20:09 +08:00
if is_efi && [ ! -e /os/boot/efi/efi/boot/$boot_efi ]; then
2023-05-25 20:15:12 +08:00
mkdir -p /os/boot/efi/efi/boot/
cp /wim/Windows/Boot/EFI/bootmgfw.efi /os/boot/efi/efi/boot/$boot_efi
fi
2023-05-28 00:52:48 +08:00
# 复制应答文件
2023-05-25 20:15:12 +08:00
cp /tmp/Autounattend.xml /wim/
2023-05-28 00:52:48 +08:00
# 提交修改 boot.wim
wimunmount --commit /wim/
# windows 7 没有 invoke-webrequest
# installer分区盘符不一定是D盘
# 所以复制 resize.bat 到 install.wim
2023-07-06 21:52:48 +08:00
# TODO: 由于esd文件无法修改要将resize.bat放到boot.wim
if [[ "$install_wim" = "*.wim" ]]; then
wimmountrw $install_wim "$image_name" /wim/
download $confhome/resize.bat /wim/resize.bat
wimunmount --commit /wim/
fi
2023-05-25 20:15:12 +08:00
2023-05-28 00:52:48 +08:00
# 添加引导
2023-07-06 22:20:09 +08:00
if is_efi; then
2023-05-28 00:52:48 +08:00
apk add efibootmgr
efibootmgr -c -L "Windows Installer" -d /dev/$xda -p1 -l "\\EFI\\boot\\$boot_efi"
else
# 或者用 ms-sys
apk add grub-bios
grub-install --boot-directory=/os/boot /dev/$xda
cat <<EOF >/os/boot/grub/grub.cfg
set timeout=5
menuentry "reinstall" {
search --no-floppy --label --set=root installer
ntldr /bootmgr
}
EOF
fi
2023-07-25 00:21:08 +08:00
}
2023-05-25 20:15:12 +08:00
2023-07-25 00:21:08 +08:00
install_redhat_ubuntu() {
# 安装 grub2
if is_efi; then
# 注意低版本的grub无法启动f38 arm的内核
# https://forums.fedoraforum.org/showthread.php?330104-aarch64-pxeboot-vmlinuz-file-format-changed-broke-PXE-installs
2023-05-04 21:45:01 +08:00
2023-07-25 00:21:08 +08:00
apk add grub-efi efibootmgr
grub-install --efi-directory=/os/boot/efi --boot-directory=/os/boot
2023-05-03 22:22:21 +08:00
2023-07-25 00:21:08 +08:00
# 添加 netboot 备用
arch_uname=$(uname -m)
cd /os/boot/efi
if [ "$arch_uname" = aarch64 ]; then
download https://boot.netboot.xyz/ipxe/netboot.xyz-arm64.efi
else
download https://boot.netboot.xyz/ipxe/netboot.xyz.efi
fi
2023-05-13 00:14:46 +08:00
else
2023-07-25 00:21:08 +08:00
apk add grub-bios
grub-install --boot-directory=/os/boot /dev/$xda
2023-05-13 00:14:46 +08:00
fi
2023-05-03 22:22:21 +08:00
2023-07-25 00:21:08 +08:00
# 重新整理 extra因为grub会处理掉引号要重新添加引号
for var in $(grep -o '\bextra\.[^ ]*' /proc/cmdline | xargs); do
extra_cmdline="$extra_cmdline $(echo $var | sed -E "s/(extra\.[^=]*)=(.*)/\1='\2'/")"
done
2023-05-03 22:22:21 +08:00
2023-09-10 22:23:03 +08:00
# 安装红帽系时,只有最后一个有安装界面显示
# https://anaconda-installer.readthedocs.io/en/latest/boot-options.html#console
console_cmdline=$(get_ttys console=)
2023-07-25 00:21:08 +08:00
grub_cfg=/os/boot/grub/grub.cfg
2023-05-03 22:22:21 +08:00
2023-07-25 00:21:08 +08:00
# 新版grub不区分linux/linuxefi
# shellcheck disable=SC2154
if [ "$distro" = "ubuntu" ]; then
download $iso /os/installer/ubuntu.iso
2023-05-03 22:22:21 +08:00
2023-08-28 19:33:53 +08:00
apk add dmidecode
dmi=$(dmidecode)
# https://github.com/systemd/systemd/blob/main/src/basic/virt.c
# https://github.com/canonical/cloud-init/blob/main/tools/ds-identify
# http://git.annexia.org/?p=virt-what.git;a=blob;f=virt-what.in;hb=HEAD
if echo "$dmi" | grep -Eiw "amazon|ec2"; then
kernel=aws
elif echo "$dmi" | grep -Eiw "Google Compute Engine|GoogleCloud"; then
kernel=gcp
elif echo "$dmi" | grep -Eiw "OracleCloud"; then
kernel=oracle
elif echo "$dmi" | grep -Eiw "7783-7084-3265-9085-8269-3286-77"; then
kernel=azure
else
kernel=generic
fi
2023-07-25 00:21:08 +08:00
# 正常写法应该是 ds="nocloud-net;s=https://xxx/" 但是甲骨文云的ds更优先自己的ds根本无访问记录
# $seed 是 https://xxx/
cat <<EOF >$grub_cfg
2023-05-03 22:22:21 +08:00
set timeout=5
menuentry "reinstall" {
# https://bugs.launchpad.net/ubuntu/+source/grub2/+bug/1851311
2023-05-28 00:52:48 +08:00
# rmmod tpm
2023-05-03 22:22:21 +08:00
search --no-floppy --label --set=root installer
loopback loop /ubuntu.iso
2023-08-28 19:33:53 +08:00
linux (loop)/casper/vmlinuz iso-scan/filename=/ubuntu.iso autoinstall noprompt noeject cloud-config-url=$ks $extra_cmdline extra.kernel=$kernel --- $console_cmdline
2023-05-03 22:22:21 +08:00
initrd (loop)/casper/initrd
}
EOF
2023-07-25 00:21:08 +08:00
else
download $vmlinuz /os/vmlinuz
download $initrd /os/initrd.img
download $squashfs /os/installer/install.img
2023-05-03 22:22:21 +08:00
2023-07-25 00:21:08 +08:00
cat <<EOF >$grub_cfg
2023-05-03 22:22:21 +08:00
set timeout=5
menuentry "reinstall" {
search --no-floppy --label --set=root os
2023-09-10 22:23:03 +08:00
linux /vmlinuz inst.stage2=hd:LABEL=installer:/install.img inst.ks=$ks $extra_cmdline $console_cmdline
2023-05-13 00:14:46 +08:00
initrd /initrd.img
2023-05-03 22:22:21 +08:00
}
EOF
2023-07-25 00:21:08 +08:00
fi
}
# 脚本入口
# arm要手动从硬件同步时间避免访问https出错
hwclock -s
# 设置密码,安装并打开 ssh
echo root:123@@@ | chpasswd
printf '\nyes' | setup-sshd
extract_env_from_cmdline
# shellcheck disable=SC2154
if [ "$sleep" = 1 ]; then
exit
fi
2023-09-10 22:23:03 +08:00
if [ "$distro" != "alpine" ]; then
mod_motd
fi
2023-07-25 00:21:08 +08:00
setup_tty_and_log
clear_previous
2023-08-25 23:36:32 +08:00
add_community_repo
2023-07-25 00:21:08 +08:00
# 找到主硬盘
# shellcheck disable=SC2010
xda=$(ls /dev/ | grep -Ex 'sda|hda|xda|vda|xvda|nvme0n1')
if [ "$distro" != "alpine" ]; then
setup_nginx_if_enough_ram
fi
# shellcheck disable=SC2154
2023-07-25 00:21:08 +08:00
if [ "$distro" = "alpine" ]; then
install_alpine
elif [ "$distro" = "dd" ]; then
install_dd
elif is_use_cloud_image; then
if [ "$img_type" = "xz" ] || [ "$img_type" = "gzip" ]; then
2023-07-28 21:27:16 +08:00
install_cloud_image_by_dd
2023-07-25 00:21:08 +08:00
else
2023-07-28 21:27:16 +08:00
create_part
install_cloud_image
2023-07-25 00:21:08 +08:00
fi
2023-07-28 21:27:16 +08:00
else
create_part
mount_part
if [ "$distro" = "windows" ]; then
install_windows
else
install_redhat_ubuntu
fi
2023-07-25 00:21:08 +08:00
fi
if [ "$sleep" = 2 ]; then
exit
2023-05-03 22:22:21 +08:00
fi
reboot