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
|
2023-10-27 11:50:06 +08:00
|
|
|
|
if grep -q '^http.*/edge/main$' /etc/apk/repositories; then
|
2023-08-25 13:04:06 +08:00
|
|
|
|
alpine_ver=edge
|
|
|
|
|
else
|
|
|
|
|
alpine_ver=v$(cut -d. -f1,2 </etc/alpine-release)
|
|
|
|
|
fi
|
|
|
|
|
|
2023-10-27 11:50:06 +08:00
|
|
|
|
if ! grep -q "^http.*/$alpine_ver/community$" /etc/apk/repositories; then
|
|
|
|
|
mirror=$(grep '^http.*/main$' /etc/apk/repositories | sed 's,/[^/]*/main$,,' | head -1)
|
|
|
|
|
echo $mirror/$alpine_ver/community >>/etc/apk/repositories
|
2023-07-05 21:57:27 +08:00
|
|
|
|
fi
|
2023-05-25 20:15:12 +08:00
|
|
|
|
}
|
|
|
|
|
|
2023-10-22 17:29:54 +08:00
|
|
|
|
# busybox 的 wget 没有重试功能
|
|
|
|
|
wget() {
|
|
|
|
|
for i in 1 2 3; do
|
|
|
|
|
command wget "$@" && return
|
|
|
|
|
done
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
is_have_cmd() {
|
|
|
|
|
command -v "$1" >/dev/null
|
|
|
|
|
}
|
|
|
|
|
|
2023-06-18 21:27:22 +08:00
|
|
|
|
download() {
|
2023-07-03 23:11:10 +08:00
|
|
|
|
url=$1
|
2023-10-22 17:29:54 +08:00
|
|
|
|
path=$2
|
2023-07-03 23:11:10 +08:00
|
|
|
|
echo $url
|
2023-06-18 21:27:22 +08:00
|
|
|
|
|
2023-10-22 17:29:54 +08:00
|
|
|
|
# 有ipv4地址无ipv4网关的情况下,aria2可能会用ipv4下载,而不是ipv6
|
2023-07-03 23:11:10 +08:00
|
|
|
|
# axel 在 lightsail 上会占用大量cpu
|
2023-10-22 17:29:54 +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-08 15:55:06 +08:00
|
|
|
|
# 构造 aria2 参数
|
|
|
|
|
# 没有指定文件名的情况
|
2023-10-22 17:29:54 +08:00
|
|
|
|
if [ -z "$path" ]; then
|
2023-07-08 15:55:06 +08:00
|
|
|
|
save=""
|
|
|
|
|
else
|
|
|
|
|
# 文件名是绝对路径
|
2023-11-03 23:50:02 +08:00
|
|
|
|
if [[ "$path" = '/*' ]]; then
|
2023-10-22 17:29:54 +08:00
|
|
|
|
save="-d / -o $path"
|
2023-07-08 15:55:06 +08:00
|
|
|
|
else
|
|
|
|
|
# 文件名是相对路径
|
2023-10-22 17:29:54 +08:00
|
|
|
|
save="-o $path"
|
2023-07-08 15:55:06 +08:00
|
|
|
|
fi
|
|
|
|
|
fi
|
2023-10-22 17:29:54 +08:00
|
|
|
|
|
|
|
|
|
# stdbuf 在 coreutils 包里面
|
|
|
|
|
if ! is_have_cmd aria2c; then
|
|
|
|
|
apk add aria2 coreutils
|
2023-06-18 21:27:22 +08:00
|
|
|
|
fi
|
2023-10-22 17:29:54 +08:00
|
|
|
|
|
2023-12-02 23:14:43 +08:00
|
|
|
|
# 阿里云源检测 user-agent 禁止 axel/aria2 下载
|
|
|
|
|
# aria2 默认 --max-tries 5
|
|
|
|
|
stdbuf -o0 -e0 \
|
|
|
|
|
aria2c -x4 \
|
|
|
|
|
--allow-overwrite=true \
|
|
|
|
|
--summary-interval=0 \
|
|
|
|
|
--user-agent=Wget/1.21.1 \
|
|
|
|
|
$save $url
|
2023-06-18 21:27:22 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
update_part() {
|
2023-12-13 21:56:37 +08:00
|
|
|
|
sleep 1
|
2023-12-02 23:26:09 +08:00
|
|
|
|
|
2023-12-13 21:56:37 +08:00
|
|
|
|
# 玄学
|
|
|
|
|
for i in 1 2 3; do
|
|
|
|
|
sync
|
|
|
|
|
partprobe /dev/$xda 2>/dev/null
|
|
|
|
|
|
|
|
|
|
# partx
|
|
|
|
|
# https://access.redhat.com/solutions/199573
|
|
|
|
|
if is_have_cmd partx; then
|
|
|
|
|
partx -u $1
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
if rc-service --exists udev && rc-service -q udev status; then
|
|
|
|
|
# udev
|
|
|
|
|
udevadm trigger
|
|
|
|
|
udevadm settle
|
|
|
|
|
else
|
|
|
|
|
# busybox mdev
|
|
|
|
|
# -f 好像没用,而且 3.16 没有
|
|
|
|
|
mdev -s 2>/dev/null
|
|
|
|
|
fi
|
|
|
|
|
done
|
2023-07-18 00:01:07 +08:00
|
|
|
|
}
|
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
|
2023-10-25 14:19:25 +08:00
|
|
|
|
# shellcheck disable=SC2154
|
|
|
|
|
wget $confhome/logviewer.html -O /logviewer.html
|
|
|
|
|
wget $confhome/logviewer-nginx.conf -O /etc/nginx/http.d/default.conf
|
|
|
|
|
|
2023-07-08 16:02:06 +08:00
|
|
|
|
# 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-09 00:04:18 +08:00
|
|
|
|
|
2023-12-02 23:30:39 +08:00
|
|
|
|
get_approximate_ram_size() {
|
|
|
|
|
# lsmem 需要 util-linux
|
|
|
|
|
if is_have_cmd lsmem; then
|
|
|
|
|
ram_size=$(lsmem -b 2>/dev/null | grep 'Total online memory:' | awk '{ print $NF/1024/1024 }')
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
if [ -z $ram_size ]; then
|
|
|
|
|
ram_size=$(free -m | awk '{print $2}' | sed -n '2p')
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
echo "$ram_size"
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-25 00:21:08 +08:00
|
|
|
|
setup_nginx_if_enough_ram() {
|
2023-12-02 23:30:39 +08:00
|
|
|
|
total_ram=$(get_approximate_ram_size)
|
|
|
|
|
# 512内存才安装
|
2023-07-25 00:21:08 +08:00
|
|
|
|
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
|
2023-07-09 00:04:18 +08:00
|
|
|
|
ln -sf /reinstall.html /var/www/localhost/htdocs/index.html
|
2023-07-08 16:02:06 +08:00
|
|
|
|
rc-service lighttpd start
|
|
|
|
|
}
|
2023-07-09 00:04:18 +08:00
|
|
|
|
|
2023-12-02 23:26:09 +08:00
|
|
|
|
setup_udev_util_linux() {
|
|
|
|
|
# mdev 不会删除 /sys/block/by-label 的旧分区名,所以用 udev
|
|
|
|
|
# util-linux 包含 lsblk
|
|
|
|
|
# util-linux 可自动探测 mount 格式
|
|
|
|
|
apk add udev util-linux
|
|
|
|
|
rc-service udev start
|
|
|
|
|
}
|
|
|
|
|
|
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
|
|
|
|
}
|
|
|
|
|
|
2023-10-08 21:46:45 +08:00
|
|
|
|
get_xda() {
|
|
|
|
|
# 排除只读盘,vda 放前面
|
|
|
|
|
# 有的机器有sda和vda,vda是主硬盘,另一个盘是只读
|
2023-10-22 19:07:12 +08:00
|
|
|
|
# TODO: 找出容量最大的?
|
2023-10-08 21:46:45 +08:00
|
|
|
|
for _xda in vda xda sda hda xvda nvme0n1; do
|
|
|
|
|
if [ -e "/sys/class/block/$_xda/ro" ] &&
|
|
|
|
|
[ "$(cat /sys/class/block/$_xda/ro)" = 0 ]; then
|
|
|
|
|
echo $_xda
|
|
|
|
|
return
|
|
|
|
|
fi
|
|
|
|
|
done
|
|
|
|
|
return 1
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-09 00:04:18 +08:00
|
|
|
|
setup_tty_and_log() {
|
|
|
|
|
# 显示输出到前台
|
|
|
|
|
# script -f /dev/tty0
|
2023-08-08 23:21:28 +08:00
|
|
|
|
dev_ttys=$(get_ttys /dev/)
|
2023-10-25 14:19:25 +08:00
|
|
|
|
exec > >(tee -a $dev_ttys /reinstall.log) 2>&1
|
2023-07-09 00:04:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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\.//")
|
2023-07-09 00:04:18 +08:00
|
|
|
|
done
|
|
|
|
|
}
|
|
|
|
|
|
2023-08-01 21:48:20 +08:00
|
|
|
|
qemu_nbd() {
|
|
|
|
|
command qemu-nbd "$@"
|
|
|
|
|
sleep 5
|
|
|
|
|
}
|
|
|
|
|
|
2023-08-08 23:51:12 +08:00
|
|
|
|
mod_motd() {
|
2023-10-22 17:00:32 +08:00
|
|
|
|
# 安装后 alpine 后要恢复默认
|
2023-11-28 23:59:17 +08:00
|
|
|
|
# 自动安装失败后,可能手动安装 alpine,因此无需判断 $distro
|
2023-12-02 23:30:39 +08:00
|
|
|
|
file=/etc/motd
|
|
|
|
|
if ! [ -e $file.orig ]; then
|
|
|
|
|
cp $file $file.orig
|
2023-10-22 17:00:32 +08:00
|
|
|
|
# shellcheck disable=SC2016
|
2023-12-02 23:30:39 +08:00
|
|
|
|
echo "mv "\$mnt$file.orig" "\$mnt$file"" |
|
|
|
|
|
insert_into_file /sbin/setup-disk before 'cleanup_chroot_mounts "\$mnt"'
|
2023-10-22 17:00:32 +08:00
|
|
|
|
|
2023-12-02 23:30:39 +08:00
|
|
|
|
cat <<EOF >$file
|
2023-08-08 23:51:12 +08:00
|
|
|
|
Reinstalling...
|
2023-10-22 17:00:32 +08:00
|
|
|
|
To view logs run:
|
2023-10-25 14:19:25 +08:00
|
|
|
|
tail -fn+1 /reinstall.log
|
2023-08-08 23:51:12 +08:00
|
|
|
|
EOF
|
2023-12-02 23:30:39 +08:00
|
|
|
|
fi
|
|
|
|
|
}
|
2023-12-02 23:26:09 +08:00
|
|
|
|
|
|
|
|
|
umount_all() {
|
|
|
|
|
dirs="/os /iso /wim /installer /nbd /nbd-boot /nbd-efi /root"
|
|
|
|
|
regex=$(echo "$dirs" | sed 's, ,|,g')
|
|
|
|
|
if mounts=$(mount | grep -Ew "$regex" | awk '{print $3}' | tac); then
|
|
|
|
|
for mount in $mounts; do
|
|
|
|
|
echo "umount $mount"
|
|
|
|
|
umount $mount
|
|
|
|
|
done
|
|
|
|
|
fi
|
2023-08-08 23:51:12 +08:00
|
|
|
|
}
|
|
|
|
|
|
2023-07-18 00:01:07 +08:00
|
|
|
|
# 可能脚本不是首次运行,先清理之前的残留
|
|
|
|
|
clear_previous() {
|
2023-12-02 23:26:09 +08:00
|
|
|
|
qemu_nbd -d /dev/nbd0 2>/dev/null || true
|
|
|
|
|
swapoff -a
|
|
|
|
|
umount_all
|
|
|
|
|
|
|
|
|
|
# 以下情况 umount -R /1 会提示 busy
|
|
|
|
|
# mount /file1 /1
|
|
|
|
|
# mount /1/file2 /2
|
2023-07-18 00:01:07 +08:00
|
|
|
|
}
|
|
|
|
|
|
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-22 19:53:26 +08:00
|
|
|
|
}
|
|
|
|
|
|
2023-08-25 23:18:23 +08:00
|
|
|
|
is_virt() {
|
|
|
|
|
get_virt_to virt
|
|
|
|
|
[ -n "$virt" ]
|
|
|
|
|
}
|
2023-08-21 23:50:20 +08:00
|
|
|
|
|
2023-08-25 23:18:23 +08:00
|
|
|
|
get_ra_to() {
|
|
|
|
|
if [ -z "$_ra" ]; then
|
|
|
|
|
apk add ndisc6
|
2023-09-03 19:35:03 +08:00
|
|
|
|
# 有时会重复收取,所以设置收一份后退出
|
2023-10-28 21:46:38 +08:00
|
|
|
|
echo "Gathering network info..."
|
2023-12-05 21:53:24 +08:00
|
|
|
|
get_netconf_to ethx
|
|
|
|
|
# shellcheck disable=SC2154
|
|
|
|
|
_ra="$(rdisc6 -1 "$ethx")"
|
2023-08-25 23:18:23 +08:00
|
|
|
|
apk del ndisc6
|
2023-11-03 23:55:04 +08:00
|
|
|
|
|
|
|
|
|
# 显示网络配置
|
|
|
|
|
echo
|
|
|
|
|
echo "$_ra" | cat -n
|
|
|
|
|
echo
|
|
|
|
|
ip addr | cat -n
|
|
|
|
|
echo
|
2023-08-21 23:50:20 +08:00
|
|
|
|
fi
|
2023-08-25 23:18:23 +08:00
|
|
|
|
eval "$1='$_ra'"
|
|
|
|
|
}
|
2023-08-21 23:50:20 +08:00
|
|
|
|
|
2023-08-25 23:18:23 +08:00
|
|
|
|
get_netconf_to() {
|
|
|
|
|
case "$1" in
|
2023-10-28 21:46:38 +08:00
|
|
|
|
slaac | dhcpv6 | rdnss | other) get_ra_to ra ;;
|
2023-08-25 23:18:23 +08:00
|
|
|
|
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-10-28 21:46:38 +08:00
|
|
|
|
other) echo "$ra" | grep 'Stateful other conf' | grep Yes && res=1 || res=0 ;;
|
2023-09-03 19:35:02 +08:00
|
|
|
|
*) res=$(cat /dev/$1) ;;
|
2023-08-25 23:18:23 +08:00
|
|
|
|
esac
|
|
|
|
|
|
|
|
|
|
eval "$1='$res'"
|
|
|
|
|
}
|
|
|
|
|
|
2023-10-22 19:07:12 +08:00
|
|
|
|
is_in_china() {
|
|
|
|
|
get_netconf_to is_in_china
|
|
|
|
|
# shellcheck disable=SC2154
|
|
|
|
|
[ "$is_in_china" = 1 ]
|
|
|
|
|
}
|
|
|
|
|
|
2023-10-22 17:06:35 +08:00
|
|
|
|
# 有 dhcpv4 不等于有网关,例如 vultr 纯 ipv6
|
|
|
|
|
# 没有 dhcpv4 不等于是静态ip,可能是没有 ip
|
2023-08-25 23:18:23 +08:00
|
|
|
|
is_dhcpv4() {
|
|
|
|
|
get_netconf_to dhcpv4
|
|
|
|
|
# shellcheck disable=SC2154
|
|
|
|
|
[ "$dhcpv4" = 1 ]
|
|
|
|
|
}
|
|
|
|
|
|
2023-10-22 17:06:35 +08:00
|
|
|
|
is_staticv4() {
|
|
|
|
|
if ! is_dhcpv4; then
|
|
|
|
|
get_netconf_to ipv4_addr
|
|
|
|
|
get_netconf_to ipv4_gateway
|
|
|
|
|
if [ -n "$ipv4_addr" ] && [ -n "$ipv4_gateway" ]; then
|
|
|
|
|
return 0
|
|
|
|
|
fi
|
|
|
|
|
fi
|
|
|
|
|
return 1
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
is_staticv6() {
|
|
|
|
|
if ! is_slaac && ! is_dhcpv6; then
|
|
|
|
|
get_netconf_to ipv6_addr
|
|
|
|
|
get_netconf_to ipv6_gateway
|
|
|
|
|
if [ -n "$ipv6_addr" ] && [ -n "$ipv6_gateway" ]; then
|
|
|
|
|
return 0
|
|
|
|
|
fi
|
|
|
|
|
fi
|
|
|
|
|
return 1
|
|
|
|
|
}
|
|
|
|
|
|
2023-08-25 23:18:23 +08:00
|
|
|
|
is_slaac() {
|
|
|
|
|
get_netconf_to slaac
|
|
|
|
|
# shellcheck disable=SC2154
|
|
|
|
|
[ "$slaac" = 1 ]
|
|
|
|
|
}
|
2023-08-21 23:50:20 +08:00
|
|
|
|
|
2023-08-25 23:18:23 +08:00
|
|
|
|
is_dhcpv6() {
|
|
|
|
|
get_netconf_to dhcpv6
|
|
|
|
|
# shellcheck disable=SC2154
|
|
|
|
|
[ "$dhcpv6" = 1 ]
|
|
|
|
|
}
|
|
|
|
|
|
2023-10-28 21:46:38 +08:00
|
|
|
|
is_have_ipv6() {
|
|
|
|
|
is_slaac || is_dhcpv6 || is_staticv6
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
is_enable_other_flag() {
|
|
|
|
|
get_netconf_to other
|
|
|
|
|
# shellcheck disable=SC2154
|
|
|
|
|
[ "$other" = 1 ]
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
is_have_rdnss() {
|
2023-10-31 22:32:50 +08:00
|
|
|
|
# rdnss 可能有几个
|
2023-10-28 21:46:38 +08:00
|
|
|
|
get_netconf_to rdnss
|
|
|
|
|
[ -n "$rdnss" ]
|
|
|
|
|
}
|
|
|
|
|
|
2023-10-31 22:32:50 +08:00
|
|
|
|
is_windows() {
|
|
|
|
|
for dir in /os /wim; do
|
|
|
|
|
[ -d $dir/Windows/System32 ] && return 0
|
|
|
|
|
done
|
|
|
|
|
return 1
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
is_windows_support_rdnss() {
|
|
|
|
|
apk add pev
|
|
|
|
|
for dir in /os /wim; do
|
|
|
|
|
dll=$dir/Windows/System32/kernel32.dll
|
|
|
|
|
# 15063 或之后才支持 rdnss
|
|
|
|
|
if [ -f $dll ]; then
|
|
|
|
|
build_ver="$(peres -v $dll | grep 'Product Version:' | cut -d. -f3)"
|
|
|
|
|
echo "Windows Build Version: $build_ver"
|
|
|
|
|
if [ "$build_ver" -ge 15063 ]; then
|
|
|
|
|
return 0
|
|
|
|
|
else
|
|
|
|
|
return 1
|
|
|
|
|
fi
|
|
|
|
|
fi
|
|
|
|
|
done
|
|
|
|
|
error_and_exit "Not found kernel32.dll"
|
|
|
|
|
}
|
|
|
|
|
|
2023-10-28 21:46:38 +08:00
|
|
|
|
is_need_manual_set_dnsv6() {
|
|
|
|
|
# 有没有可能是静态但是有 rdnss?
|
2023-10-31 22:32:50 +08:00
|
|
|
|
is_have_ipv6 &&
|
|
|
|
|
! is_dhcpv6 &&
|
|
|
|
|
! is_enable_other_flag &&
|
|
|
|
|
{ ! is_have_rdnss || { is_have_rdnss && is_windows && ! is_windows_support_rdnss; }; }
|
2023-10-28 21:46:38 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
get_current_dns_v4() {
|
|
|
|
|
grep '^nameserver' /etc/resolv.conf | awk '{print $2}' | grep '\.'
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
get_current_dns_v6() {
|
|
|
|
|
grep '^nameserver' /etc/resolv.conf | awk '{print $2}' | grep ':'
|
|
|
|
|
}
|
|
|
|
|
|
2023-10-22 18:43:54 +08:00
|
|
|
|
to_upper() {
|
|
|
|
|
tr '[:lower:]' '[:upper:]'
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
to_lower() {
|
|
|
|
|
tr '[:upper:]' '[:lower:]'
|
|
|
|
|
}
|
|
|
|
|
|
2023-12-22 23:56:12 +08:00
|
|
|
|
get_part_num_by_part() {
|
|
|
|
|
dev_part=$1
|
|
|
|
|
echo "$dev_part" | grep -o '[0-9]*' | tail -1
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
get_fallback_efi_file_name() {
|
|
|
|
|
case $(arch) in
|
|
|
|
|
x86_64) echo bootx64.efi ;;
|
|
|
|
|
aarch64) echo bootaa64.efi ;;
|
|
|
|
|
*) error_and_exit ;;
|
|
|
|
|
esac
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
del_invalid_efi_entry() {
|
|
|
|
|
apk add lsblk efibootmgr
|
|
|
|
|
|
|
|
|
|
efibootmgr --quiet --remove-dups
|
|
|
|
|
|
|
|
|
|
while read -r line; do
|
|
|
|
|
part_uuid=$(echo "$line" | awk -F ',' '{print $3}')
|
|
|
|
|
efi_index=$(echo "$line" | grep_efi_index)
|
|
|
|
|
if ! lsblk -o PARTUUID | grep "$part_uuid"; then
|
|
|
|
|
echo "Delete invalid EFI Entry: $line"
|
|
|
|
|
efibootmgr --quiet --bootnum "$efi_index" --delete-bootnum
|
|
|
|
|
fi
|
|
|
|
|
done < <(efibootmgr | grep 'HD(.*,GPT,')
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
grep_efi_index() {
|
|
|
|
|
awk -F '*' '{print $1}' | sed 's/Boot//'
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
# 某些机器可能不会回落到 bootx64.efi
|
|
|
|
|
# 因此手动添加一个回落项
|
|
|
|
|
add_fallback_efi_to_nvram() {
|
|
|
|
|
apk add lsblk efibootmgr
|
|
|
|
|
|
|
|
|
|
EFI_UUID=C12A7328-F81F-11D2-BA4B-00A0C93EC93B
|
|
|
|
|
efi_row=$(lsblk /dev/$xda -ro NAME,PARTTYPE,PARTUUID | grep -i "$EFI_UUID")
|
|
|
|
|
efi_part_uuid=$(echo "$efi_row" | awk '{print $3}')
|
|
|
|
|
efi_part_name=$(echo "$efi_row" | awk '{print $1}')
|
|
|
|
|
efi_part_num=$(get_part_num_by_part "$efi_part_name")
|
|
|
|
|
efi_file=$(get_fallback_efi_file_name)
|
|
|
|
|
|
|
|
|
|
# 创建条目,先判断是否已经存在
|
|
|
|
|
if ! efibootmgr | grep -i "HD($efi_part_num,GPT,$efi_part_uuid,.*)/File(\\\EFI\\\boot\\\\$efi_file)"; then
|
|
|
|
|
fallback_id=$(efibootmgr --create-only \
|
|
|
|
|
--disk "/dev/$xda" \
|
|
|
|
|
--part "$efi_part_num" \
|
|
|
|
|
--label "fallback" \
|
|
|
|
|
--loader "\\EFI\\boot\\$efi_file" |
|
|
|
|
|
tail -1 | grep_efi_index)
|
|
|
|
|
|
|
|
|
|
# 添加到最后
|
|
|
|
|
orig_order=$(efibootmgr | grep -F BootOrder: | awk '{print $2}')
|
|
|
|
|
if [ -n "$orig_order" ]; then
|
|
|
|
|
new_order="$orig_order,$fallback_id"
|
|
|
|
|
else
|
|
|
|
|
new_order="$fallback_id"
|
|
|
|
|
fi
|
|
|
|
|
efibootmgr --bootorder "$new_order"
|
|
|
|
|
fi
|
|
|
|
|
}
|
|
|
|
|
|
2023-10-22 18:43:54 +08:00
|
|
|
|
unix2dos() {
|
|
|
|
|
target=$1
|
|
|
|
|
|
2023-12-14 00:11:32 +08:00
|
|
|
|
# 先原地unix2dos,出错再用cat,可最大限度保留文件权限
|
|
|
|
|
if ! command unix2dos $target 2>/tmp/unix2dos.log; then
|
2023-10-22 18:43:54 +08:00
|
|
|
|
# 出错后删除 unix2dos 创建的临时文件
|
2023-12-14 00:11:32 +08:00
|
|
|
|
rm "$(awk -F: '{print $2}' /tmp/unix2dos.log | xargs)"
|
2023-10-22 18:43:54 +08:00
|
|
|
|
tmp=$(mktemp)
|
|
|
|
|
cp $target $tmp
|
|
|
|
|
command unix2dos $tmp
|
2023-12-14 00:11:32 +08:00
|
|
|
|
# cat 可以保留权限
|
|
|
|
|
cat $tmp >$target
|
2023-10-22 18:43:54 +08:00
|
|
|
|
rm $tmp
|
|
|
|
|
fi
|
|
|
|
|
}
|
|
|
|
|
|
2023-09-16 20:00:05 +08:00
|
|
|
|
insert_into_file() {
|
|
|
|
|
file=$1
|
|
|
|
|
location=$2
|
|
|
|
|
regex_to_find=$3
|
|
|
|
|
|
2023-12-14 00:11:32 +08:00
|
|
|
|
if [ "$location" = head ]; then
|
|
|
|
|
bak=$(mktemp)
|
|
|
|
|
cp $file $bak
|
|
|
|
|
cat - $bak >$file
|
2023-10-22 18:43:54 +08:00
|
|
|
|
else
|
|
|
|
|
line_num=$(grep -E -n "$regex_to_find" "$file" | cut -d: -f1)
|
|
|
|
|
|
|
|
|
|
found_count=$(echo "$line_num" | wc -l)
|
|
|
|
|
if [ ! "$found_count" -eq 1 ]; then
|
|
|
|
|
return 1
|
|
|
|
|
fi
|
2023-09-16 20:00:05 +08:00
|
|
|
|
|
2023-10-22 18:43:54 +08:00
|
|
|
|
case "$location" in
|
|
|
|
|
before) line_num=$((line_num - 1)) ;;
|
|
|
|
|
after) ;;
|
|
|
|
|
*) return 1 ;;
|
|
|
|
|
esac
|
|
|
|
|
|
|
|
|
|
sed -i "${line_num}r /dev/stdin" "$file"
|
|
|
|
|
fi
|
2023-09-16 20:00:05 +08:00
|
|
|
|
}
|
|
|
|
|
|
2023-08-25 23:18:23 +08:00
|
|
|
|
install_alpine() {
|
2023-12-02 23:30:39 +08:00
|
|
|
|
hack_lowram_modloop=true
|
|
|
|
|
hack_lowram_swap=true
|
|
|
|
|
mount / -o remount,size=100%
|
|
|
|
|
|
|
|
|
|
if $hack_lowram_modloop; then
|
2023-09-21 00:12:49 +08:00
|
|
|
|
# 预先加载需要的模块
|
|
|
|
|
if rc-service modloop status; then
|
|
|
|
|
modules="ext4 vfat nls_utf8 nls_cp437 crc32c"
|
|
|
|
|
for mod in $modules; do
|
|
|
|
|
modprobe $mod
|
|
|
|
|
done
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
# 删除 modloop ,释放内存
|
|
|
|
|
rc-service modloop stop
|
|
|
|
|
rm -f /lib/modloop-lts /lib/modloop-virt
|
2023-12-02 23:30:39 +08:00
|
|
|
|
fi
|
2023-09-21 00:12:49 +08:00
|
|
|
|
|
2023-12-02 23:30:39 +08:00
|
|
|
|
# bios机器用 setup-disk 自动分区会有 boot 分区
|
|
|
|
|
# 因此手动分区安装
|
|
|
|
|
create_part
|
2023-09-21 00:12:49 +08:00
|
|
|
|
|
2023-12-02 23:30:39 +08:00
|
|
|
|
# 挂载系统分区
|
|
|
|
|
if is_efi || is_xda_gt_2t; then
|
|
|
|
|
os_part_num=2
|
|
|
|
|
else
|
|
|
|
|
os_part_num=1
|
|
|
|
|
fi
|
|
|
|
|
mkdir -p /os
|
|
|
|
|
mount -t ext4 /dev/${xda}*${os_part_num} /os
|
2023-09-21 00:12:49 +08:00
|
|
|
|
|
2023-12-02 23:30:39 +08:00
|
|
|
|
# 挂载 efi
|
|
|
|
|
if is_efi; then
|
|
|
|
|
mkdir -p /os/boot/efi
|
|
|
|
|
mount -t vfat /dev/${xda}*1 /os/boot/efi
|
2023-09-21 00:12:49 +08:00
|
|
|
|
fi
|
|
|
|
|
|
2023-12-02 23:30:39 +08:00
|
|
|
|
# 创建 swap
|
|
|
|
|
if $hack_lowram_swap; then
|
|
|
|
|
create_swap 256 /os/swapfile
|
2023-10-29 22:16:29 +08:00
|
|
|
|
fi
|
|
|
|
|
|
2023-05-13 00:14:46 +08:00
|
|
|
|
# 网络
|
2023-08-21 23:50:20 +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-22 19:53:26 +08:00
|
|
|
|
|
2023-12-05 21:53:24 +08:00
|
|
|
|
get_netconf_to ethx
|
|
|
|
|
|
|
|
|
|
# 生成 lo配置 + ethx头部
|
2023-08-22 19:53:26 +08:00
|
|
|
|
cat <<EOF >/etc/network/interfaces
|
|
|
|
|
auto lo
|
|
|
|
|
iface lo inet loopback
|
2023-08-25 13:39:48 +08:00
|
|
|
|
|
2023-12-05 21:53:24 +08:00
|
|
|
|
auto $ethx
|
2023-08-22 19:53:26 +08:00
|
|
|
|
EOF
|
2023-05-13 00:14:46 +08:00
|
|
|
|
|
2023-10-28 21:46:38 +08:00
|
|
|
|
# ipv4
|
|
|
|
|
if is_dhcpv4; then
|
2023-12-05 21:53:24 +08:00
|
|
|
|
echo "iface $ethx inet dhcp" >>/etc/network/interfaces
|
2023-10-28 21:46:38 +08:00
|
|
|
|
|
|
|
|
|
elif is_staticv4; then
|
2023-08-25 23:18:23 +08:00
|
|
|
|
get_netconf_to ipv4_addr
|
|
|
|
|
get_netconf_to ipv4_gateway
|
2023-10-22 17:06:35 +08:00
|
|
|
|
cat <<EOF >>/etc/network/interfaces
|
2023-12-05 21:53:24 +08:00
|
|
|
|
iface $ethx inet static
|
2023-08-22 19:53:26 +08:00
|
|
|
|
address $ipv4_addr
|
|
|
|
|
gateway $ipv4_gateway
|
|
|
|
|
EOF
|
2023-10-28 21:46:38 +08:00
|
|
|
|
# dns
|
|
|
|
|
if list=$(get_current_dns_v4); then
|
|
|
|
|
for dns in $list; do
|
|
|
|
|
cat <<EOF >>/etc/network/interfaces
|
|
|
|
|
dns-nameserver $dns
|
|
|
|
|
EOF
|
|
|
|
|
done
|
|
|
|
|
fi
|
2023-08-21 23:50:20 +08:00
|
|
|
|
fi
|
|
|
|
|
|
2023-10-28 21:46:38 +08:00
|
|
|
|
# ipv6
|
|
|
|
|
if is_slaac; then
|
2023-12-05 21:53:24 +08:00
|
|
|
|
echo "iface $ethx inet6 auto" >>/etc/network/interfaces
|
2023-10-28 21:46:38 +08:00
|
|
|
|
|
|
|
|
|
elif is_dhcpv6; then
|
2023-12-05 21:53:24 +08:00
|
|
|
|
echo "iface $ethx inet6 dhcp" >>/etc/network/interfaces
|
2023-10-28 21:46:38 +08:00
|
|
|
|
|
|
|
|
|
elif is_staticv6; then
|
2023-08-25 23:18:23 +08:00
|
|
|
|
get_netconf_to ipv6_addr
|
|
|
|
|
get_netconf_to ipv6_gateway
|
2023-10-22 17:06:35 +08:00
|
|
|
|
cat <<EOF >>/etc/network/interfaces
|
2023-12-05 21:53:24 +08:00
|
|
|
|
iface $ethx inet6 static
|
2023-08-21 23:50:20 +08:00
|
|
|
|
address $ipv6_addr
|
|
|
|
|
gateway $ipv6_gateway
|
|
|
|
|
EOF
|
2023-10-28 21:46:38 +08:00
|
|
|
|
fi
|
2023-10-22 17:06:35 +08:00
|
|
|
|
|
2023-10-28 21:46:38 +08:00
|
|
|
|
# dns
|
2023-10-31 22:32:50 +08:00
|
|
|
|
# 有 ipv6 但需设置 dns 的情况
|
2023-10-28 21:46:38 +08:00
|
|
|
|
if is_need_manual_set_dnsv6 && list=$(get_current_dns_v6); then
|
|
|
|
|
for dns in $list; do
|
|
|
|
|
cat <<EOF >>/etc/network/interfaces
|
|
|
|
|
dns-nameserver $dns
|
|
|
|
|
EOF
|
|
|
|
|
done
|
2023-08-21 23:50:20 +08:00
|
|
|
|
fi
|
2023-08-22 19:53:26 +08:00
|
|
|
|
|
2023-08-25 23:18:23 +08:00
|
|
|
|
# 显示网络配置
|
|
|
|
|
echo
|
2023-08-21 23:50:20 +08:00
|
|
|
|
cat -n /etc/network/interfaces
|
2023-08-25 23:18:23 +08:00
|
|
|
|
echo
|
2023-08-21 23:50:20 +08:00
|
|
|
|
|
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
|
2023-11-28 23:59:17 +08:00
|
|
|
|
|
|
|
|
|
# 3.16 没有 seedrng
|
2023-08-25 23:18:23 +08:00
|
|
|
|
rc-update add crond
|
2023-11-28 23:59:17 +08:00
|
|
|
|
rc-update add seedrng boot || true
|
2023-08-25 23:18:23 +08:00
|
|
|
|
|
|
|
|
|
# 如果是 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
|
|
|
|
|
|
|
|
|
# 重置为官方仓库配置
|
2023-10-31 22:33:59 +08:00
|
|
|
|
# 国内机可能无法访问mirror列表而报错
|
|
|
|
|
if false; then
|
|
|
|
|
true >/etc/apk/repositories
|
|
|
|
|
setup-apkrepos -1
|
|
|
|
|
fi
|
2023-05-13 00:14:46 +08:00
|
|
|
|
|
2023-12-02 23:30:39 +08:00
|
|
|
|
# 修复3.16安装后无法引导
|
|
|
|
|
# https://github.com/alpinelinux/alpine-conf/commit/bc7aeab868bf4d94dde2ff5d6eb97daede5975b9
|
|
|
|
|
if grep -F '3.16' /etc/alpine-release; then
|
|
|
|
|
file=/sbin/setup-disk
|
|
|
|
|
if ! [ -e $file.orig ]; then
|
|
|
|
|
cp $file $file.orig
|
|
|
|
|
# shellcheck disable=SC2016
|
|
|
|
|
echo 'apk add --quiet $(select_bootloader_pkg)' |
|
|
|
|
|
insert_into_file $file after '# install to given mounted root'
|
|
|
|
|
fi
|
|
|
|
|
fi
|
|
|
|
|
|
2023-12-22 23:56:12 +08:00
|
|
|
|
# setup-disk 安装 grub 跳过了添加引导项到 nvram
|
|
|
|
|
# 防止部分机器不会 fallback 到 bootx64.efi
|
|
|
|
|
if is_efi; then
|
|
|
|
|
apk add efibootmgr
|
|
|
|
|
sed -i 's/--no-nvram//' /sbin/setup-disk
|
|
|
|
|
fi
|
|
|
|
|
|
2023-05-13 00:14:46 +08:00
|
|
|
|
# 安装到硬盘
|
|
|
|
|
# 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-12-02 23:30:39 +08:00
|
|
|
|
printf 'y' | setup-disk -m sys -k $kernel_flavor /os
|
|
|
|
|
|
2023-12-07 23:06:15 +08:00
|
|
|
|
# 3.19 或以上,非 efi 需要手动安装 grub
|
|
|
|
|
if ! is_efi && grep -F '3.19' /etc/alpine-release; then
|
|
|
|
|
grub-install --boot-directory=/os/boot --target=i386-pc /dev/$xda
|
|
|
|
|
fi
|
|
|
|
|
|
2023-12-22 23:56:12 +08:00
|
|
|
|
# 删除无效的 efi 条目
|
2023-12-23 08:27:41 +08:00
|
|
|
|
if is_efi; then
|
|
|
|
|
del_invalid_efi_entry
|
|
|
|
|
fi
|
2023-12-22 23:56:12 +08:00
|
|
|
|
|
2023-12-02 23:30:39 +08:00
|
|
|
|
# 是否保留 swap
|
|
|
|
|
if [ -e /os/swapfile ]; then
|
|
|
|
|
if false; then
|
|
|
|
|
echo "/swapfile swap swap defaults 0 0" >>/os/etc/fstab
|
|
|
|
|
ln -sf /etc/init.d/swap /os/etc/runlevels/boot/swap
|
|
|
|
|
else
|
|
|
|
|
swapoff -a
|
|
|
|
|
rm /os/swapfile
|
|
|
|
|
fi
|
|
|
|
|
fi
|
2023-07-25 00:21:08 +08:00
|
|
|
|
}
|
2023-06-04 19:12:57 +08:00
|
|
|
|
|
2023-10-22 19:07:12 +08:00
|
|
|
|
get_http_file_size_to() {
|
|
|
|
|
var_name=$1
|
|
|
|
|
url=$2
|
|
|
|
|
|
|
|
|
|
size=''
|
|
|
|
|
if wget --spider -S $url -o /tmp/headers.log; then
|
|
|
|
|
# 网址重定向可能得到多个 Content-Length, 选最后一个
|
|
|
|
|
set -o pipefail
|
|
|
|
|
if size=$(grep 'Content-Length:' /tmp/headers.log |
|
|
|
|
|
tail -1 | awk '{print $2}'); then
|
|
|
|
|
eval "$var_name='$size'"
|
|
|
|
|
fi
|
|
|
|
|
set +o pipefail
|
|
|
|
|
else
|
|
|
|
|
error_and_exit "Can't access $url"
|
|
|
|
|
fi
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-25 00:21:08 +08:00
|
|
|
|
# shellcheck disable=SC2154
|
2023-10-22 19:07:12 +08:00
|
|
|
|
dd_gzip_xz() {
|
2023-06-05 19:45:07 +08:00
|
|
|
|
case "$img_type" in
|
2023-06-05 19:50:21 +08:00
|
|
|
|
gzip) prog=gzip ;;
|
2023-06-04 19:12:57 +08:00
|
|
|
|
xz) prog=xz ;;
|
2023-10-22 17:29:54 +08:00
|
|
|
|
*) error_and_exit 'Not supported' ;;
|
2023-06-04 19:12:57 +08:00
|
|
|
|
esac
|
|
|
|
|
|
2023-10-22 17:29:54 +08:00
|
|
|
|
# alpine busybox 自带 gzip xz,但官方版也许性能更好
|
|
|
|
|
# 用官方 wget,一来带进度条,二来自带重试
|
|
|
|
|
apk add wget $prog
|
|
|
|
|
command wget $img -O- --tries=3 --progress=bar:force | $prog -dc >/dev/$xda
|
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() {
|
2023-12-02 23:26:09 +08:00
|
|
|
|
# 除了 dd 都会用到
|
2023-05-03 22:22:21 +08:00
|
|
|
|
|
2023-12-02 23:26:09 +08:00
|
|
|
|
# 分区工具
|
|
|
|
|
apk add parted e2fsprogs
|
|
|
|
|
if is_efi; then
|
|
|
|
|
apk add dosfstools
|
|
|
|
|
fi
|
2023-07-28 21:27:16 +08:00
|
|
|
|
|
|
|
|
|
# 清除分区签名
|
2023-10-22 19:07:12 +08:00
|
|
|
|
# TODO: 先检测iso链接/各种链接
|
|
|
|
|
# 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
|
2023-10-22 19:07:12 +08:00
|
|
|
|
get_http_file_size_to size_bytes $iso
|
2023-12-02 23:26:09 +08:00
|
|
|
|
|
|
|
|
|
# 默认值,最大的iso 23h2 需要7g
|
|
|
|
|
if [ -z "$size_bytes" ]; then
|
|
|
|
|
size_bytes=$((7 * 1024 * 1024 * 1024))
|
2023-09-28 23:51:30 +08:00
|
|
|
|
fi
|
|
|
|
|
|
2023-12-02 23:26:09 +08:00
|
|
|
|
# 按iso容量计算分区大小,200m用于驱动和文件系统自身占用
|
|
|
|
|
part_size="$((size_bytes / 1024 / 1024 + 200))MiB"
|
|
|
|
|
|
2024-01-22 19:00:17 +08:00
|
|
|
|
apk add ntfs-3g-progs
|
2023-07-25 00:21:08 +08:00
|
|
|
|
# 虽然ntfs3不需要fuse,但wimmount需要,所以还是要保留
|
|
|
|
|
modprobe fuse ntfs3
|
|
|
|
|
if is_efi; then
|
|
|
|
|
# efi
|
|
|
|
|
parted /dev/$xda -s -- \
|
|
|
|
|
mklabel gpt \
|
|
|
|
|
mkpart '" "' fat32 1MiB 1025MiB \
|
|
|
|
|
mkpart '" "' fat32 1025MiB 1041MiB \
|
2023-10-22 19:07:12 +08:00
|
|
|
|
mkpart '" "' ext4 1041MiB -${part_size} \
|
|
|
|
|
mkpart '" "' ntfs -${part_size} 100% \
|
2023-07-25 00:21:08 +08:00
|
|
|
|
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 \
|
2023-10-22 19:07:12 +08:00
|
|
|
|
mkpart primary ntfs 1MiB -${part_size} \
|
|
|
|
|
mkpart primary ntfs -${part_size} 100% \
|
2023-07-25 00:21:08 +08:00
|
|
|
|
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
|
|
|
|
|
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
|
2024-01-22 00:21:37 +08:00
|
|
|
|
# fedora debian ubuntu opensuse arch gentoo
|
|
|
|
|
installer_part_size=1GiB
|
2023-09-16 20:00:07 +08:00
|
|
|
|
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-12-02 23:30:39 +08:00
|
|
|
|
elif [ "$distro" = alpine ]; then
|
|
|
|
|
if is_efi; then
|
|
|
|
|
# efi
|
|
|
|
|
parted /dev/$xda -s -- \
|
|
|
|
|
mklabel gpt \
|
|
|
|
|
mkpart '" "' fat32 1MiB 101MiB \
|
|
|
|
|
mkpart '" "' ext4 101MiB 100% \
|
|
|
|
|
set 1 boot on
|
|
|
|
|
update_part /dev/$xda
|
|
|
|
|
|
|
|
|
|
mkfs.fat /dev/$xda*1 #1 efi
|
|
|
|
|
mkfs.ext4 -F /dev/$xda*2 #2 os
|
|
|
|
|
elif is_xda_gt_2t; then
|
|
|
|
|
# bios > 2t
|
|
|
|
|
parted /dev/$xda -s -- \
|
|
|
|
|
mklabel gpt \
|
|
|
|
|
mkpart '" "' ext4 1MiB 2MiB \
|
|
|
|
|
mkpart '" "' ext4 2MiB 100% \
|
|
|
|
|
set 1 bios_grub on
|
|
|
|
|
update_part /dev/$xda
|
|
|
|
|
|
|
|
|
|
echo #1 bios_boot
|
|
|
|
|
mkfs.ext4 -F /dev/$xda*2 #2 os
|
|
|
|
|
else
|
|
|
|
|
# bios
|
|
|
|
|
parted /dev/$xda -s -- \
|
|
|
|
|
mklabel msdos \
|
|
|
|
|
mkpart primary ext4 1MiB 100% \
|
|
|
|
|
set 1 boot on
|
|
|
|
|
update_part /dev/$xda
|
|
|
|
|
|
|
|
|
|
mkfs.ext4 -F /dev/$xda*1 #1 os
|
|
|
|
|
fi
|
2023-07-25 00:21:08 +08:00
|
|
|
|
else
|
2023-08-15 11:20:07 +08:00
|
|
|
|
# 安装红帽系或ubuntu
|
2023-07-25 00:21:08 +08:00
|
|
|
|
# 对于红帽系是临时分区表,安装时除了 installer 分区,其他分区会重建为默认的大小
|
|
|
|
|
# 对于ubuntu是最终分区表,因为 ubuntu 的安装器不能调整个别分区,只能重建整个分区表
|
2023-08-15 11:20:07 +08:00
|
|
|
|
# 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
|
|
|
|
|
2023-08-15 11:20:07 +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
|
|
|
|
|
2023-08-15 11:20:07 +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
|
|
|
|
|
2023-08-15 11:20:07 +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
|
2023-07-30 00:34:58 +08:00
|
|
|
|
update_part /dev/$xda
|
|
|
|
|
|
|
|
|
|
# centos 7 无法加载alpine格式化的ext4
|
|
|
|
|
# 要关闭这个属性
|
2023-08-15 11:20:07 +08:00
|
|
|
|
# 目前改用fat格式,不用设置这个
|
|
|
|
|
if false && [ "$distro" = centos ]; then
|
2023-07-30 00:34:58 +08:00
|
|
|
|
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-12-02 23:30:39 +08:00
|
|
|
|
|
|
|
|
|
# alpine 删除分区工具,防止 256M 小机爆内存
|
|
|
|
|
# setup-disk /dev/sda 会保留格式化工具,我们也保留
|
|
|
|
|
if [ "$distro" = alpine ]; then
|
|
|
|
|
apk del parted
|
|
|
|
|
fi
|
2023-07-25 00:21:08 +08:00
|
|
|
|
}
|
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
|
|
|
|
|
}
|
|
|
|
|
|
2023-09-03 19:35:03 +08:00
|
|
|
|
create_cloud_init_network_config() {
|
|
|
|
|
ci_file=$1
|
|
|
|
|
|
2023-12-05 21:53:24 +08:00
|
|
|
|
get_netconf_to ethx
|
2023-09-03 19:35:03 +08:00
|
|
|
|
get_netconf_to mac_addr
|
2023-10-27 12:01:45 +08:00
|
|
|
|
apk add yq
|
2023-09-03 19:35:03 +08:00
|
|
|
|
|
2023-11-13 18:42:05 +08:00
|
|
|
|
# shellcheck disable=SC2154
|
|
|
|
|
yq -i ".network.version=1 |
|
|
|
|
|
.network.config[0].type=\"physical\" |
|
2023-12-05 21:53:24 +08:00
|
|
|
|
.network.config[0].name=\"$ethx\" |
|
2023-11-15 20:46:15 +08:00
|
|
|
|
.network.config[0].mac_address=\"$mac_addr\" |
|
|
|
|
|
.network.config[1].type=\"nameserver\"
|
2023-11-13 18:42:05 +08:00
|
|
|
|
" $ci_file
|
|
|
|
|
|
|
|
|
|
ip_index=0
|
2023-10-27 12:01:45 +08:00
|
|
|
|
|
|
|
|
|
# ipv4
|
|
|
|
|
if is_dhcpv4; then
|
2023-11-13 18:42:05 +08:00
|
|
|
|
yq -i ".network.config[0].subnets[$ip_index] = {\"type\": \"dhcp4\"}" $ci_file
|
|
|
|
|
ip_index=$((ip_index + 1))
|
2023-10-28 21:46:38 +08:00
|
|
|
|
elif is_staticv4; then
|
|
|
|
|
get_netconf_to ipv4_addr
|
|
|
|
|
get_netconf_to ipv4_gateway
|
2023-11-13 18:42:05 +08:00
|
|
|
|
yq -i ".network.config[0].subnets[$ip_index] = {
|
|
|
|
|
\"type\": \"static\",
|
|
|
|
|
\"address\": \"$ipv4_addr\",
|
|
|
|
|
\"gateway\": \"$ipv4_gateway\" }
|
|
|
|
|
" $ci_file
|
2023-10-27 12:01:45 +08:00
|
|
|
|
|
2023-11-15 20:46:15 +08:00
|
|
|
|
# 旧版 cloud-init 有 bug
|
|
|
|
|
# 有的版本会只从第一种配置中读取 dns,有的从第二种读取
|
|
|
|
|
# 因此写两种配置
|
2023-10-28 21:46:38 +08:00
|
|
|
|
if dns4_list=$(get_current_dns_v4); then
|
2023-10-27 12:01:45 +08:00
|
|
|
|
for cur in $dns4_list; do
|
2023-11-13 18:42:05 +08:00
|
|
|
|
yq -i ".network.config[0].subnets[$ip_index].dns_nameservers += [\"$cur\"]" $ci_file
|
2023-11-15 20:46:15 +08:00
|
|
|
|
yq -i ".network.config[1].address += [\"$cur\"]" $ci_file
|
2023-10-27 12:01:45 +08:00
|
|
|
|
done
|
|
|
|
|
fi
|
2023-11-13 18:42:05 +08:00
|
|
|
|
ip_index=$((ip_index + 1))
|
2023-09-03 19:35:03 +08:00
|
|
|
|
fi
|
|
|
|
|
|
2023-10-27 12:01:45 +08:00
|
|
|
|
# ipv6
|
|
|
|
|
if is_slaac; then
|
2023-10-28 21:46:38 +08:00
|
|
|
|
if is_enable_other_flag; then
|
|
|
|
|
type=ipv6_dhcpv6-stateless
|
|
|
|
|
else
|
|
|
|
|
type=ipv6_slaac
|
|
|
|
|
fi
|
2023-11-13 18:42:05 +08:00
|
|
|
|
yq -i ".network.config[0].subnets[$ip_index] = {\"type\": \"$type\"}" $ci_file
|
2023-10-27 12:01:45 +08:00
|
|
|
|
|
|
|
|
|
elif is_dhcpv6; then
|
2023-11-13 18:42:05 +08:00
|
|
|
|
yq -i ".network.config[0].subnets[$ip_index] = {\"type\": \"ipv6_dhcpv6-stateful\"}" $ci_file
|
2023-10-27 12:01:45 +08:00
|
|
|
|
|
2023-10-28 21:46:38 +08:00
|
|
|
|
elif is_staticv6; then
|
|
|
|
|
get_netconf_to ipv6_addr
|
|
|
|
|
get_netconf_to ipv6_gateway
|
2023-10-27 12:01:45 +08:00
|
|
|
|
# centos7 不认识 static6,但可改成 static,作用相同
|
|
|
|
|
# https://github.com/canonical/cloud-init/commit/dacdd30080bd8183d1f1c1dc9dbcbc8448301529
|
2023-11-12 22:14:56 +08:00
|
|
|
|
if [ -f /os/etc/system-release-cpe ] &&
|
|
|
|
|
grep centos:7 /os/etc/system-release-cpe; then
|
|
|
|
|
type_ipv6_static=static
|
|
|
|
|
else
|
|
|
|
|
type_ipv6_static=static6
|
|
|
|
|
fi
|
2023-11-13 18:42:05 +08:00
|
|
|
|
yq -i ".network.config[0].subnets[$ip_index] = {
|
|
|
|
|
\"type\": \"$type_ipv6_static\",
|
|
|
|
|
\"address\": \"$ipv6_addr\",
|
|
|
|
|
\"gateway\": \"$ipv6_gateway\" }
|
|
|
|
|
" $ci_file
|
2023-10-28 21:46:38 +08:00
|
|
|
|
fi
|
2023-10-27 12:01:45 +08:00
|
|
|
|
|
2023-10-31 22:32:50 +08:00
|
|
|
|
# 有 ipv6 但需设置 dns 的情况
|
2023-10-28 21:46:38 +08:00
|
|
|
|
if is_need_manual_set_dnsv6 && dns6_list=$(get_current_dns_v6); then
|
|
|
|
|
for cur in $dns6_list; do
|
2023-11-13 18:42:05 +08:00
|
|
|
|
yq -i ".network.config[0].subnets[$ip_index].dns_nameservers += [\"$cur\"]" $ci_file
|
2023-11-15 20:46:15 +08:00
|
|
|
|
yq -i ".network.config[1].address += [\"$cur\"]" $ci_file
|
2023-10-28 21:46:38 +08:00
|
|
|
|
done
|
2023-10-27 12:01:45 +08:00
|
|
|
|
fi
|
2023-11-16 22:08:57 +08:00
|
|
|
|
|
|
|
|
|
# 如果 network.config[1] 没有 address,则删除,避免低版本 cloud-init 报错
|
|
|
|
|
yq -i 'del(.network.config[1] | select(has("address") | not))' $ci_file
|
2023-09-03 19:35:03 +08:00
|
|
|
|
}
|
|
|
|
|
|
2023-11-12 17:29:16 +08:00
|
|
|
|
truncate_machine_id() {
|
|
|
|
|
os_dir=$1
|
|
|
|
|
|
|
|
|
|
truncate -s 0 $os_dir/etc/machine-id
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-28 21:27:16 +08:00
|
|
|
|
download_cloud_init_config() {
|
2023-08-02 00:02:00 +08:00
|
|
|
|
os_dir=$1
|
2023-07-28 21:27:16 +08:00
|
|
|
|
|
2023-11-12 17:29:15 +08:00
|
|
|
|
ci_file=$os_dir/etc/cloud/cloud.cfg.d/99_fallback.cfg
|
2023-10-22 00:46:53 +08:00
|
|
|
|
download $confhome/cloud-init.yaml $ci_file
|
|
|
|
|
# 删除注释行,除了第一行
|
|
|
|
|
sed -i '1!{/^[[:space:]]*#/d}' $ci_file
|
2023-07-28 21:27:16 +08:00
|
|
|
|
|
|
|
|
|
# swapfile
|
2023-09-21 00:12:35 +08:00
|
|
|
|
# 如果分区表中已经有swapfile就跳过,例如arch
|
2023-08-02 00:02:00 +08:00
|
|
|
|
if ! grep -w swap $os_dir/etc/fstab; then
|
2023-07-28 21:27:16 +08:00
|
|
|
|
# btrfs
|
2023-11-13 18:42:36 +08:00
|
|
|
|
# 目前只有 arch 和 fedora 镜像使用 btrfs
|
|
|
|
|
# 等 fedora 38 cloud-init 升级到 v23.3 后删除
|
2023-07-28 21:27:16 +08:00
|
|
|
|
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
|
2023-11-13 18:42:36 +08:00
|
|
|
|
- systemctl daemon-reload
|
2023-07-28 21:27:16 +08:00
|
|
|
|
EOF
|
|
|
|
|
else
|
|
|
|
|
# ext4 xfs
|
|
|
|
|
cat <<EOF >>$ci_file
|
|
|
|
|
swap:
|
|
|
|
|
filename: /swapfile
|
|
|
|
|
size: auto
|
|
|
|
|
EOF
|
|
|
|
|
fi
|
|
|
|
|
fi
|
2023-09-03 19:35:03 +08:00
|
|
|
|
|
|
|
|
|
create_cloud_init_network_config $ci_file
|
|
|
|
|
cat -n $ci_file
|
2023-07-28 21:27:16 +08:00
|
|
|
|
}
|
|
|
|
|
|
2023-10-22 18:43:54 +08:00
|
|
|
|
modify_windows() {
|
2023-11-02 23:53:15 +08:00
|
|
|
|
os_dir=$1
|
|
|
|
|
|
2023-10-22 18:43:54 +08:00
|
|
|
|
# https://learn.microsoft.com/zh-cn/windows-hardware/manufacture/desktop/windows-setup-states
|
|
|
|
|
# https://learn.microsoft.com/zh-cn/troubleshoot/azure/virtual-machines/reset-local-password-without-agent
|
|
|
|
|
# https://learn.microsoft.com/zh-cn/windows-hardware/manufacture/desktop/add-a-custom-script-to-windows-setup
|
|
|
|
|
|
|
|
|
|
# 判断用 SetupComplete 还是组策略
|
2023-11-02 23:53:15 +08:00
|
|
|
|
state_ini=$os_dir/Windows/Setup/State/State.ini
|
2023-10-22 18:43:54 +08:00
|
|
|
|
cat $state_ini
|
|
|
|
|
if grep -q IMAGE_STATE_COMPLETE $state_ini; then
|
|
|
|
|
use_gpo=true
|
|
|
|
|
else
|
|
|
|
|
use_gpo=false
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
# 下载共同的子脚本
|
|
|
|
|
# 可能 unattend.xml 已经设置了ExtendOSPartition,不过运行resize没副作用
|
|
|
|
|
bats="windows-resize.bat windows-set-netconf.bat"
|
2023-11-02 23:53:15 +08:00
|
|
|
|
download $confhome/windows-resize.bat $os_dir/windows-resize.bat
|
|
|
|
|
create_win_set_netconf_script $os_dir/windows-set-netconf.bat
|
2023-10-22 18:43:54 +08:00
|
|
|
|
|
|
|
|
|
if $use_gpo; then
|
|
|
|
|
# 使用组策略
|
2023-11-02 23:53:15 +08:00
|
|
|
|
gpt_ini=$os_dir/Windows/System32/GroupPolicy/gpt.ini
|
|
|
|
|
scripts_ini=$os_dir/Windows/System32/GroupPolicy/Machine/Scripts/scripts.ini
|
2023-10-22 18:43:54 +08:00
|
|
|
|
mkdir -p "$(dirname $scripts_ini)"
|
|
|
|
|
|
|
|
|
|
# 备份 ini
|
|
|
|
|
for file in $gpt_ini $scripts_ini; do
|
|
|
|
|
if [ -f $file ]; then
|
|
|
|
|
cp $file $file.orig
|
|
|
|
|
fi
|
|
|
|
|
done
|
|
|
|
|
|
|
|
|
|
# gpt.ini
|
|
|
|
|
cat >$gpt_ini <<EOF
|
|
|
|
|
[General]
|
|
|
|
|
gPCFunctionalityVersion=2
|
|
|
|
|
gPCMachineExtensionNames=[{42B5FAAE-6536-11D2-AE5A-0000F87571E3}{40B6664F-4972-11D1-A7CA-0000F87571E3}]
|
|
|
|
|
Version=1
|
|
|
|
|
EOF
|
|
|
|
|
unix2dos $gpt_ini
|
|
|
|
|
|
|
|
|
|
# scripts.ini
|
|
|
|
|
if ! [ -e $scripts_ini ]; then
|
|
|
|
|
touch $scripts_ini
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
if ! grep -F '[Startup]' $scripts_ini; then
|
|
|
|
|
echo '[Startup]' >>$scripts_ini
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
# 注意没用 pipefail 的话,错误码取自最后一个管道
|
|
|
|
|
if num=$(grep -Eo '^[0-9]+' $scripts_ini | sort -n | tail -1 | grep .); then
|
|
|
|
|
num=$((num + 1))
|
|
|
|
|
else
|
|
|
|
|
num=0
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
bats="$bats windows-del-gpo.bat"
|
|
|
|
|
for bat in $bats; do
|
|
|
|
|
echo "${num}CmdLine=%SystemDrive%\\$bat" >>$scripts_ini
|
|
|
|
|
echo "${num}Parameters=" >>$scripts_ini
|
|
|
|
|
num=$((num + 1))
|
|
|
|
|
done
|
|
|
|
|
cat $scripts_ini
|
|
|
|
|
unix2dos $scripts_ini
|
|
|
|
|
|
|
|
|
|
# windows-del-gpo.bat
|
2023-11-02 23:53:15 +08:00
|
|
|
|
download $confhome/windows-del-gpo.bat $os_dir/windows-del-gpo.bat
|
2023-10-22 18:43:54 +08:00
|
|
|
|
else
|
|
|
|
|
# 使用 SetupComplete
|
2023-11-02 23:53:15 +08:00
|
|
|
|
setup_complete=$os_dir/Windows/Setup/Scripts/SetupComplete.cmd
|
2023-10-22 18:43:54 +08:00
|
|
|
|
mkdir -p "$(dirname $setup_complete)"
|
|
|
|
|
|
|
|
|
|
# 添加到 C:\Setup\Scripts\SetupComplete.cmd 最前面
|
|
|
|
|
# call 防止子 bat 删除自身后中断主脚本
|
2023-12-14 00:11:32 +08:00
|
|
|
|
setup_complete_mod=$(mktemp)
|
2023-10-22 18:43:54 +08:00
|
|
|
|
for bat in $bats; do
|
2023-12-14 00:11:32 +08:00
|
|
|
|
echo "if exist %SystemDrive%\\$bat (call %SystemDrive%\\$bat)" >>$setup_complete_mod
|
2023-10-22 18:43:54 +08:00
|
|
|
|
done
|
|
|
|
|
|
2023-12-14 00:11:32 +08:00
|
|
|
|
# 复制原来的内容
|
2023-10-22 18:43:54 +08:00
|
|
|
|
if [ -f $setup_complete ]; then
|
2023-12-14 00:11:32 +08:00
|
|
|
|
cat $setup_complete >>$setup_complete_mod
|
2023-10-22 18:43:54 +08:00
|
|
|
|
fi
|
|
|
|
|
|
2023-12-14 00:11:32 +08:00
|
|
|
|
unix2dos $setup_complete_mod
|
|
|
|
|
|
|
|
|
|
# cat 可以保留权限
|
|
|
|
|
cat $setup_complete_mod >$setup_complete
|
2023-10-22 18:43:54 +08:00
|
|
|
|
fi
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
modify_linux() {
|
|
|
|
|
os_dir=$1
|
|
|
|
|
|
2023-08-06 22:16:27 +08:00
|
|
|
|
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
|
|
|
|
download_cloud_init_config $os_dir
|
2023-09-10 22:23:04 +08:00
|
|
|
|
|
2023-11-12 17:29:16 +08:00
|
|
|
|
truncate_machine_id $os_dir
|
|
|
|
|
|
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
|
2023-08-06 22:16:27 +08:00
|
|
|
|
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
|
|
|
|
|
|
2023-11-13 18:42:36 +08:00
|
|
|
|
# 修复 fedora 38 或以下用静态 ipv6 会掉线
|
|
|
|
|
# el 全系也用 NetworkManager,但他们的配置文件是 sysconfig,因此不受影响
|
|
|
|
|
# https://github.com/canonical/cloud-init/commit/5d440856cb6d2b4c908015fe4eb7227615c17c8b
|
|
|
|
|
if grep -E 'fedora:(37|38)' $os_dir/etc/os-release; then
|
|
|
|
|
network_manager_py=$os_dir/usr/lib/python3.11/site-packages/cloudinit/net/network_manager.py &&
|
|
|
|
|
if ! grep '"static6": "manual",' $network_manager_py; then
|
|
|
|
|
echo '"static6": "manual",' | insert_into_file $network_manager_py after '"static": "manual",'
|
|
|
|
|
fi
|
|
|
|
|
fi
|
|
|
|
|
|
2023-11-16 21:33:11 +08:00
|
|
|
|
# debian 10/11 网络问题
|
2023-10-28 21:46:38 +08:00
|
|
|
|
if [ -f $os_dir/etc/debian_version ] && grep -E '^(10|11)' $os_dir/etc/debian_version; then
|
|
|
|
|
mv $os_dir/etc/resolv.conf $os_dir/etc/resolv.conf.orig
|
|
|
|
|
cp -f /etc/resolv.conf $os_dir/etc/resolv.conf
|
|
|
|
|
mount_pseudo_fs $os_dir
|
|
|
|
|
chroot $os_dir apt update
|
2023-11-16 21:33:11 +08:00
|
|
|
|
|
|
|
|
|
if true; then
|
|
|
|
|
# 将 debian 10/11 设置为 12 一样的网络管理器
|
|
|
|
|
# 可解决 ifupdown dhcp 不支持 24位掩码+不规则网关的问题
|
|
|
|
|
chroot $os_dir apt install -y netplan.io
|
|
|
|
|
chroot $os_dir systemctl disable networking resolvconf
|
|
|
|
|
chroot $os_dir systemctl enable systemd-networkd systemd-resolved
|
|
|
|
|
rm -f $os_dir/etc/resolv.conf $os_dir/etc/resolv.conf.orig
|
|
|
|
|
ln -sf ../run/systemd/resolve/stub-resolv.conf $os_dir/etc/resolv.conf
|
|
|
|
|
insert_into_file $os_dir/etc/cloud/cloud.cfg.d/99_fallback.cfg after '#cloud-config' <<EOF
|
|
|
|
|
system_info:
|
|
|
|
|
network:
|
|
|
|
|
renderers: [netplan]
|
|
|
|
|
activators: [netplan]
|
|
|
|
|
EOF
|
|
|
|
|
|
|
|
|
|
else
|
|
|
|
|
# debian 10/11 默认不支持 rdnss,要安装 rdnssd 或者 nm
|
|
|
|
|
chroot $os_dir apt install -y rdnssd
|
|
|
|
|
# 不会自动建立链接,因此不能删除
|
|
|
|
|
mv $os_dir/etc/resolv.conf.orig $os_dir/etc/resolv.conf
|
|
|
|
|
fi
|
2023-10-28 21:46:38 +08:00
|
|
|
|
fi
|
|
|
|
|
|
2024-01-18 23:52:54 +08:00
|
|
|
|
# opensuse tumbleweed
|
|
|
|
|
# TODO: cloud-init 更新后删除
|
2023-09-16 20:00:05 +08:00
|
|
|
|
if grep opensuse-tumbleweed $os_dir/etc/os-release; then
|
2024-01-18 23:52:54 +08:00
|
|
|
|
touch $os_dir/etc/NetworkManager/NetworkManager.conf
|
2023-08-03 22:38:38 +08:00
|
|
|
|
fi
|
2023-09-16 20:00:07 +08:00
|
|
|
|
|
2023-12-22 23:52:47 +08:00
|
|
|
|
# arch
|
|
|
|
|
if [ -f $os_dir/etc/arch-release ]; then
|
|
|
|
|
rm $os_dir/etc/resolv.conf
|
|
|
|
|
cp /etc/resolv.conf $os_dir/etc/resolv.conf
|
|
|
|
|
mount_pseudo_fs $os_dir
|
|
|
|
|
chroot $os_dir pacman-key --init
|
|
|
|
|
chroot $os_dir pacman-key --populate
|
|
|
|
|
rm $os_dir/etc/resolv.conf
|
|
|
|
|
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
|
2023-11-04 12:44:07 +08:00
|
|
|
|
EOF
|
|
|
|
|
|
|
|
|
|
# 如果创建了 cloud-init.disabled,重启后网络不受 networkd 管理
|
|
|
|
|
# 因为网卡名变回了 ens3 而不是 eth0
|
|
|
|
|
# 因此要删除 networkd 的网卡名匹配
|
|
|
|
|
insert_into_file $ci_file after '^runcmd:' <<EOF
|
|
|
|
|
- sed -i '/^Name=/d' /etc/systemd/network/10-cloud-init-eth*.network
|
2023-09-16 20:00:07 +08:00
|
|
|
|
EOF
|
|
|
|
|
fi
|
2023-08-03 22:38:38 +08:00
|
|
|
|
}
|
|
|
|
|
|
2023-10-22 18:43:54 +08:00
|
|
|
|
modify_os_on_disk() {
|
2023-10-25 14:19:24 +08:00
|
|
|
|
only_process=$1
|
|
|
|
|
|
2023-07-28 21:27:16 +08:00
|
|
|
|
update_part /dev/$xda
|
2023-10-22 18:43:54 +08:00
|
|
|
|
|
2023-10-25 14:19:24 +08:00
|
|
|
|
# dd linux 的时候不用修改硬盘内容
|
|
|
|
|
if [ "$distro" = "dd" ] && ! lsblk -f /dev/$xda | grep ntfs; then
|
|
|
|
|
return
|
|
|
|
|
fi
|
|
|
|
|
|
2023-10-22 18:43:54 +08:00
|
|
|
|
mkdir -p /os
|
|
|
|
|
# 按分区容量大到小,依次寻找系统分区
|
|
|
|
|
for part in $(lsblk /dev/$xda --sort SIZE -no NAME | sed '$d' | tac); do
|
|
|
|
|
# btrfs挂载的是默认子卷,如果没有默认子卷,挂载的是根目录
|
|
|
|
|
# fedora 云镜像没有默认子卷,且系统在root子卷中
|
|
|
|
|
if mount -o ro /dev/$part /os; then
|
2023-10-25 14:19:24 +08:00
|
|
|
|
if [ "$only_process" = linux ]; then
|
|
|
|
|
if etc_dir=$({ ls -d /os/etc/ || ls -d /os/*/etc/; } 2>/dev/null); then
|
|
|
|
|
os_dir=$(dirname $etc_dir)
|
|
|
|
|
# 重新挂载为读写
|
|
|
|
|
mount -o remount,rw /os
|
|
|
|
|
modify_linux $os_dir
|
|
|
|
|
return
|
|
|
|
|
fi
|
|
|
|
|
elif [ "$only_process" = windows ]; then
|
|
|
|
|
# find 有时会报 I/O error
|
|
|
|
|
if ls -d /os/*/ | grep -i '/windows/' 2>/dev/null; then
|
|
|
|
|
# 重新挂载为读写、忽略大小写
|
|
|
|
|
umount /os
|
|
|
|
|
apk add ntfs-3g
|
|
|
|
|
mount.lowntfs-3g /dev/$part /os -o ignore_case
|
|
|
|
|
modify_windows /os
|
|
|
|
|
return
|
|
|
|
|
fi
|
2023-10-22 18:43:54 +08:00
|
|
|
|
fi
|
2023-10-23 21:33:14 +08:00
|
|
|
|
umount /os
|
2023-10-22 18:43:54 +08:00
|
|
|
|
fi
|
|
|
|
|
done
|
|
|
|
|
error_and_exit "Can't find os partition."
|
2023-07-28 21:27:16 +08:00
|
|
|
|
}
|
|
|
|
|
|
2023-12-02 23:30:39 +08:00
|
|
|
|
create_swap_if_ram_less_than() {
|
|
|
|
|
need_ram=$1
|
|
|
|
|
swapfile=$2
|
|
|
|
|
|
|
|
|
|
phy_ram=$(get_approximate_ram_size)
|
|
|
|
|
swapsize=$((need_ram - phy_ram))
|
|
|
|
|
if [ $swapsize -gt 0 ]; then
|
|
|
|
|
create_swap $swapsize $swapfile
|
|
|
|
|
fi
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-30 00:35:32 +08:00
|
|
|
|
create_swap() {
|
2023-12-02 23:30:39 +08:00
|
|
|
|
swapsize=$1
|
|
|
|
|
swapfile=$2
|
|
|
|
|
|
2023-07-30 00:35:32 +08:00
|
|
|
|
if ! grep $swapfile /proc/swaps; then
|
2023-12-02 23:30:39 +08:00
|
|
|
|
fallocate -l ${swapsize}M $swapfile
|
|
|
|
|
chmod 0600 $swapfile
|
|
|
|
|
mkswap $swapfile
|
|
|
|
|
swapon $swapfile
|
2023-07-30 00:35:32 +08:00
|
|
|
|
fi
|
|
|
|
|
}
|
|
|
|
|
|
2023-08-03 22:41:30 +08:00
|
|
|
|
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-10-22 18:50:54 +08:00
|
|
|
|
download_qcow() {
|
2023-12-02 23:26:09 +08:00
|
|
|
|
apk add qemu-img
|
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-10-22 18:50:54 +08:00
|
|
|
|
}
|
2023-07-28 21:27:16 +08:00
|
|
|
|
|
2023-10-22 18:50:54 +08:00
|
|
|
|
install_qcow_el() {
|
|
|
|
|
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-28 21:27:16 +08:00
|
|
|
|
|
2023-11-09 22:30:21 +08:00
|
|
|
|
modprobe nbd nbds_max=1
|
2023-10-22 18:50:54 +08:00
|
|
|
|
qemu_nbd -c /dev/nbd0 $qcow_file
|
2023-08-03 22:30:35 +08:00
|
|
|
|
|
2023-10-22 18:50:54 +08:00
|
|
|
|
# TODO: 改成循环mount找出os+fstab查找剩余分区?
|
|
|
|
|
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)
|
2023-07-28 21:27:16 +08:00
|
|
|
|
|
2023-10-22 18:50:54 +08:00
|
|
|
|
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
|
|
|
|
|
2023-10-22 18:50:54 +08:00
|
|
|
|
mkdir -p /nbd /nbd-boot /nbd-efi /os
|
|
|
|
|
|
|
|
|
|
# 使用目标系统的格式化程序
|
|
|
|
|
# centos8 如果用alpine格式化xfs,grub2-mkconfig和grub2里面都无法识别xfs分区
|
|
|
|
|
mount -o nouuid /dev/$os_part /nbd/
|
|
|
|
|
mount_pseudo_fs /nbd/
|
|
|
|
|
chroot /nbd mkfs.xfs -f -m uuid=$os_part_uuid /dev/$xda*2
|
|
|
|
|
umount -R /nbd/
|
|
|
|
|
|
|
|
|
|
# 复制系统
|
|
|
|
|
echo Copying os partition
|
|
|
|
|
mount -o ro,nouuid /dev/$os_part /nbd/
|
|
|
|
|
mount -o noatime /dev/$xda*2 /os/
|
|
|
|
|
cp -a /nbd/* /os/
|
|
|
|
|
|
|
|
|
|
# 复制boot分区,如果有
|
|
|
|
|
if [ -n "$boot_part" ]; then
|
|
|
|
|
echo Copying boot partition
|
|
|
|
|
mount -o ro,nouuid /dev/$boot_part /nbd-boot/
|
|
|
|
|
cp -a /nbd-boot/* /os/boot/
|
|
|
|
|
fi
|
2023-07-26 23:46:24 +08:00
|
|
|
|
|
2023-10-22 18:50:54 +08:00
|
|
|
|
# efi 分区
|
|
|
|
|
efi_mount_opts="defaults,uid=0,gid=0,umask=077,shortname=winnt"
|
|
|
|
|
if is_efi; then
|
|
|
|
|
# 挂载 efi
|
|
|
|
|
mkdir -p /os/boot/efi/
|
|
|
|
|
mount -o $efi_mount_opts /dev/$xda*1 /os/boot/efi/
|
2023-07-30 00:35:32 +08:00
|
|
|
|
|
2023-10-22 18:50:54 +08:00
|
|
|
|
# 复制文件
|
|
|
|
|
if [ -n "$efi_part" ]; then
|
|
|
|
|
echo Copying efi partition
|
|
|
|
|
mount -o ro /dev/$efi_part /nbd-efi/
|
|
|
|
|
cp -a /nbd-efi/* /os/boot/efi/
|
2023-08-03 22:30:35 +08:00
|
|
|
|
fi
|
2023-10-22 18:50:54 +08:00
|
|
|
|
fi
|
2023-08-03 22:30:35 +08:00
|
|
|
|
|
2023-10-22 18:50:54 +08:00
|
|
|
|
# 取消挂载 nbd
|
|
|
|
|
umount /nbd/ /nbd-boot/ /nbd-efi/ || true
|
|
|
|
|
qemu_nbd -d /dev/nbd0
|
|
|
|
|
|
|
|
|
|
# 如果镜像有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
|
2023-08-03 22:30:35 +08:00
|
|
|
|
|
2023-10-22 18:50:54 +08:00
|
|
|
|
# 挂载伪文件系统
|
|
|
|
|
mount_pseudo_fs /os/
|
2023-07-26 23:46:24 +08:00
|
|
|
|
|
2023-10-22 18:50:54 +08:00
|
|
|
|
# 创建 swap
|
|
|
|
|
rm -rf /installer/*
|
2023-12-02 23:30:39 +08:00
|
|
|
|
create_swap 1024 /installer/swapfile
|
2023-07-26 23:46:24 +08:00
|
|
|
|
|
2023-10-22 18:50:54 +08:00
|
|
|
|
# resolv.conf
|
|
|
|
|
cp /etc/resolv.conf /os/etc/resolv.conf
|
2023-08-03 22:41:30 +08:00
|
|
|
|
|
2023-10-22 18:50:54 +08:00
|
|
|
|
# selinux kdump
|
|
|
|
|
disable_selinux_kdump /os
|
2023-08-03 22:38:38 +08:00
|
|
|
|
|
2023-10-22 18:50:54 +08:00
|
|
|
|
# cloud-init
|
|
|
|
|
download_cloud_init_config /os
|
2023-07-26 23:46:24 +08:00
|
|
|
|
|
2023-11-12 17:29:16 +08:00
|
|
|
|
# 部分镜像例如 centos7 要手动删除 machine-id
|
|
|
|
|
truncate_machine_id /os
|
|
|
|
|
|
2023-10-25 14:19:24 +08:00
|
|
|
|
# 为 centos 7 ci 安装 NetworkManager
|
|
|
|
|
# 1. 能够自动配置 onlink 网关
|
|
|
|
|
# 2. 解决 cloud-init 关闭了 ra,因为 nm 无视内核 ra 设置
|
|
|
|
|
if grep 'centos:7' /os/etc/system-release-cpe; then
|
|
|
|
|
yum install -y NetworkManager
|
|
|
|
|
chroot /os/ systemctl enable NetworkManager
|
|
|
|
|
fi
|
|
|
|
|
|
2023-10-22 18:50:54 +08:00
|
|
|
|
# fstab 删除 boot 分区
|
|
|
|
|
# alma/rocky 镜像本身有boot分区,但我们不需要
|
|
|
|
|
sed -i '/[[:blank:]]\/boot[[:blank:]]/d' /os/etc/fstab
|
|
|
|
|
|
|
|
|
|
# fstab 添加 efi 分区
|
|
|
|
|
if is_efi; then
|
|
|
|
|
# centos 要创建efi条目
|
|
|
|
|
if ! grep /boot/efi /os/etc/fstab; then
|
|
|
|
|
efi_part_uuid=$(lsblk /dev/$xda*1 -no UUID)
|
|
|
|
|
echo "UUID=$efi_part_uuid /boot/efi vfat $efi_mount_opts 0 0" >>/os/etc/fstab
|
2023-07-26 23:46:24 +08:00
|
|
|
|
fi
|
2023-10-22 18:50:54 +08:00
|
|
|
|
else
|
|
|
|
|
# 删除 efi 条目
|
|
|
|
|
sed -i '/[[:blank:]]\/boot\/efi[[:blank:]]/d' /os/etc/fstab
|
|
|
|
|
fi
|
2023-07-26 23:46:24 +08:00
|
|
|
|
|
2023-10-22 18:50:54 +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-10-22 18:50:54 +08:00
|
|
|
|
remove_grub_conflict_files() {
|
|
|
|
|
# bios 和 efi 转换前先删除
|
2023-07-30 00:35:32 +08:00
|
|
|
|
|
2023-10-22 18:50:54 +08:00
|
|
|
|
# bios转efi出错
|
|
|
|
|
# centos7是bios镜像,/boot/grub2/grubenv 是真身
|
|
|
|
|
# 安装grub-efi时,grubenv 会改成指向efi分区grubenv软连接
|
|
|
|
|
# 如果安装grub-efi前没有删除原来的grubenv,原来的grubenv将不变,新建的软连接将变成 grubenv.rpmnew
|
|
|
|
|
# 后续grubenv的改动无法同步到efi分区,会造成grub2-setdefault失效
|
2023-07-30 00:35:32 +08:00
|
|
|
|
|
2023-10-22 18:50:54 +08:00
|
|
|
|
# efi转bios出错
|
|
|
|
|
# 如果是指向efi目录的软连接(例如el8),先删除它,否则 grub2-install 会报错
|
|
|
|
|
rm -rf /os/boot/grub2/grubenv /os/boot/grub2/grub.cfg
|
|
|
|
|
}
|
2023-07-30 00:35:32 +08:00
|
|
|
|
|
2023-10-22 18:50:54 +08:00
|
|
|
|
# 安装 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
|
2023-07-30 00:35:32 +08:00
|
|
|
|
|
2023-10-22 18:50:54 +08:00
|
|
|
|
# 安装 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
|
|
|
|
|
2023-10-22 18:50:54 +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-28 21:27:16 +08:00
|
|
|
|
|
2023-10-22 18:50:54 +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
|
2023-07-30 00:35:32 +08:00
|
|
|
|
search --no-floppy --fs-uuid --set=dev $os_part_uuid
|
|
|
|
|
set prefix=(\$dev)/boot/grub2
|
|
|
|
|
export \$prefix
|
|
|
|
|
configfile \$prefix/grub.cfg
|
|
|
|
|
EOF
|
2023-10-22 18:50:54 +08:00
|
|
|
|
fi
|
2023-07-30 00:35:32 +08:00
|
|
|
|
|
2023-10-22 18:50:54 +08:00
|
|
|
|
# 主 grub.cfg
|
|
|
|
|
if is_efi && [ "$releasever" -le 8 ]; then
|
|
|
|
|
chroot /os/ grub2-mkconfig -o /boot/efi/EFI/$distro_full/grub.cfg
|
|
|
|
|
else
|
|
|
|
|
chroot /os/ grub2-mkconfig -o /boot/grub2/grub.cfg
|
|
|
|
|
fi
|
2023-05-03 22:22:21 +08:00
|
|
|
|
|
2023-10-27 12:03:47 +08:00
|
|
|
|
# 不删除可能网络管理器不会写入dns
|
|
|
|
|
rm -f /os/etc/resolv.conf
|
2023-07-26 23:46:24 +08:00
|
|
|
|
|
2023-10-22 18:50:54 +08:00
|
|
|
|
# 删除installer分区,重启后cloud init会自动扩容
|
|
|
|
|
swapoff -a
|
|
|
|
|
umount /installer
|
|
|
|
|
parted /dev/$xda -s rm 3
|
|
|
|
|
}
|
2023-07-30 00:35:32 +08:00
|
|
|
|
|
2023-10-22 18:50:54 +08:00
|
|
|
|
dd_qcow() {
|
|
|
|
|
if true; then
|
2023-10-22 19:07:12 +08:00
|
|
|
|
modprobe nbd nbds_max=1
|
2023-10-22 18:50:54 +08:00
|
|
|
|
qemu_nbd -c /dev/nbd0 $qcow_file
|
|
|
|
|
|
|
|
|
|
# 检查最后一个分区是否是 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
|
2023-08-01 21:51:36 +08:00
|
|
|
|
fi
|
2023-10-22 18:50:54 +08:00
|
|
|
|
fi
|
2023-08-01 21:51:36 +08:00
|
|
|
|
|
2023-10-22 18:50:54 +08:00
|
|
|
|
# 显示分区
|
|
|
|
|
lsblk -o NAME,SIZE,FSTYPE,LABEL /dev/nbd0
|
|
|
|
|
|
|
|
|
|
# 将前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
|
2023-10-22 19:07:12 +08:00
|
|
|
|
echo "Start DD Cloud Image..."
|
2023-10-22 18:50:54 +08:00
|
|
|
|
pv -f /dev/nbd0 | dd of=/dev/$xda bs=1M skip=1 seek=1 iflag=fullblock
|
|
|
|
|
;;
|
|
|
|
|
esac
|
2023-07-26 23:46:24 +08:00
|
|
|
|
|
2023-10-22 18:50:54 +08:00
|
|
|
|
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
|
|
|
|
|
2023-10-22 18:50:54 +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-10-22 18:50:54 +08:00
|
|
|
|
}
|
2023-09-16 20:00:07 +08:00
|
|
|
|
|
2023-10-22 18:50:54 +08:00
|
|
|
|
resize_after_install_cloud_image() {
|
|
|
|
|
# 提前扩容
|
|
|
|
|
# 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-10-22 18:54:30 +08:00
|
|
|
|
|
|
|
|
|
system_part_num=$(parted /dev/$xda -m print | tail -1 | cut -d: -f1)
|
2023-10-22 18:50:54 +08:00
|
|
|
|
printf "yes" | parted /dev/$xda resizepart $system_part_num 100% ---pretend-input-tty
|
|
|
|
|
update_part /dev/$xda
|
|
|
|
|
|
|
|
|
|
if [ "$distro" = gentoo ]; then
|
|
|
|
|
apk add e2fsprogs-extra
|
2023-11-03 23:46:39 +08:00
|
|
|
|
e2fsck -p -f /dev/$xda*$system_part_num
|
|
|
|
|
resize2fs /dev/$xda*$system_part_num
|
2023-08-03 23:02:12 +08:00
|
|
|
|
fi
|
2023-12-02 23:26:09 +08:00
|
|
|
|
update_part /dev/$xda
|
2023-08-03 23:02:12 +08:00
|
|
|
|
fi
|
2023-07-23 11:48:10 +08:00
|
|
|
|
fi
|
2023-07-25 00:21:08 +08:00
|
|
|
|
}
|
|
|
|
|
|
2023-10-22 18:50:54 +08:00
|
|
|
|
mount_part_for_install_mode() {
|
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-10-22 18:43:54 +08:00
|
|
|
|
get_dns_list_for_win() {
|
2023-10-28 21:46:38 +08:00
|
|
|
|
if dns_list=$(get_current_dns_v$1); then
|
2023-10-22 18:43:54 +08:00
|
|
|
|
i=0
|
|
|
|
|
for dns in $dns_list; do
|
|
|
|
|
i=$((i + 1))
|
|
|
|
|
echo "set ipv${1}_dns$i=$dns"
|
|
|
|
|
done
|
|
|
|
|
fi
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
create_win_set_netconf_script() {
|
|
|
|
|
target=$1
|
|
|
|
|
|
2023-10-28 21:46:38 +08:00
|
|
|
|
if is_staticv4 || is_staticv6 || is_need_manual_set_dnsv6; then
|
2023-10-22 18:43:54 +08:00
|
|
|
|
get_netconf_to mac_addr
|
|
|
|
|
echo "set mac_addr=$mac_addr" >$target
|
|
|
|
|
|
|
|
|
|
# 生成静态 ipv4 配置
|
|
|
|
|
if is_staticv4; then
|
|
|
|
|
get_netconf_to ipv4_addr
|
|
|
|
|
get_netconf_to ipv4_gateway
|
|
|
|
|
ipv4_dns_list="$(get_dns_list_for_win 4)"
|
|
|
|
|
cat <<EOF >>$target
|
|
|
|
|
set ipv4_addr=$ipv4_addr
|
|
|
|
|
set ipv4_gateway=$ipv4_gateway
|
|
|
|
|
$ipv4_dns_list
|
|
|
|
|
EOF
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
# 生成静态 ipv6 配置
|
|
|
|
|
if is_staticv6; then
|
|
|
|
|
get_netconf_to ipv6_addr
|
|
|
|
|
get_netconf_to ipv6_gateway
|
|
|
|
|
cat <<EOF >>$target
|
|
|
|
|
set ipv6_addr=$ipv6_addr
|
|
|
|
|
set ipv6_gateway=$ipv6_gateway
|
2023-10-28 21:46:38 +08:00
|
|
|
|
EOF
|
|
|
|
|
fi
|
|
|
|
|
|
2023-10-31 22:32:50 +08:00
|
|
|
|
# 有 ipv6 但需设置 dns 的情况
|
2023-10-28 21:46:38 +08:00
|
|
|
|
if is_need_manual_set_dnsv6 && ipv6_dns_list="$(get_dns_list_for_win 6)"; then
|
|
|
|
|
cat <<EOF >>$target
|
2023-10-22 18:43:54 +08:00
|
|
|
|
$ipv6_dns_list
|
|
|
|
|
EOF
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
cat -n $target
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
# 脚本还有关闭ipv6隐私id的功能,所以不能省略
|
|
|
|
|
# 合并脚本
|
|
|
|
|
wget $confhome/windows-set-netconf.bat -O- >>$target
|
|
|
|
|
unix2dos $target
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-25 00:21:08 +08:00
|
|
|
|
install_windows() {
|
2024-01-22 19:00:17 +08:00
|
|
|
|
apk add wimlib virt-what dmidecode rsync pev
|
|
|
|
|
|
2023-07-25 00:21:08 +08:00
|
|
|
|
# shellcheck disable=SC2154
|
2023-05-25 20:15:12 +08:00
|
|
|
|
download $iso /os/windows.iso
|
2023-06-21 23:34:25 +08:00
|
|
|
|
mkdir -p /iso
|
2023-11-09 22:30:21 +08:00
|
|
|
|
mount -o ro /os/windows.iso /iso
|
2023-05-25 20:15:12 +08:00
|
|
|
|
|
2023-07-06 21:45:27 +08:00
|
|
|
|
# 从iso复制文件
|
|
|
|
|
# efi: 复制boot开头的文件+efi目录到efi分区,复制iso全部文件(除了boot.wim)到installer分区
|
|
|
|
|
# bios: 复制iso全部文件到installer分区
|
2023-07-06 22:20:09 +08:00
|
|
|
|
if is_efi; then
|
2023-07-06 21:45:27 +08:00
|
|
|
|
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
|
2024-01-22 19:00:17 +08:00
|
|
|
|
echo "Image Name: $image_name"
|
|
|
|
|
|
|
|
|
|
# 用内核版本号筛选驱动
|
|
|
|
|
# 使得可以安装 Hyper-V Server / Azure Stack HCI 等 Windows Server 变种
|
|
|
|
|
nt_ver="$(peres -v /iso/setup.exe | grep 'Product Version:' | cut -d: -f2 | xargs | cut -d. -f 1,2)"
|
|
|
|
|
echo "NT Version: $nt_ver"
|
2023-07-06 21:45:27 +08:00
|
|
|
|
|
2024-01-22 19:00:17 +08:00
|
|
|
|
is_nt_6_1() {
|
|
|
|
|
[ "$nt_ver" = 6.1 ]
|
2023-07-06 22:20:09 +08:00
|
|
|
|
}
|
|
|
|
|
|
2023-09-28 23:51:30 +08:00
|
|
|
|
is_win11() {
|
|
|
|
|
echo $image_name | grep -iEw '^Windows 11'
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
# 跳过 win11 硬件限制
|
|
|
|
|
if is_win11; then
|
|
|
|
|
wiminfo "$install_wim" "$image_name" --image-property WINDOWS/INSTALLATIONTYPE=Server
|
|
|
|
|
fi
|
|
|
|
|
|
2023-07-03 23:11:10 +08:00
|
|
|
|
# 变量名 使用场景
|
2023-12-02 23:26:09 +08:00
|
|
|
|
# arch_uname arch命令/uname -m x86_64 aarch64
|
2023-07-03 23:11:10 +08:00
|
|
|
|
# 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
|
2023-10-22 19:07:12 +08:00
|
|
|
|
arch_wim=$(wiminfo $install_wim 1 | grep Architecture: | awk '{print $2}' | to_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
|
2023-07-06 21:45:27 +08:00
|
|
|
|
# 所以不要在原系统上判断具体虚拟化环境
|
|
|
|
|
|
|
|
|
|
# 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 &&
|
2023-07-05 21:40:00 +08:00
|
|
|
|
[ "$arch_wim" = x86_64 ]; then
|
2023-07-03 23:11:10 +08:00
|
|
|
|
# aws nitro
|
2023-07-05 21:40:00 +08:00
|
|
|
|
# 只有 x64 位驱动
|
2024-01-22 19:00:17 +08:00
|
|
|
|
# https://docs.aws.amazon.com/zh_cn/AWSEC2/latest/WindowsGuide/aws-nvme-drivers.html
|
|
|
|
|
# https://docs.aws.amazon.com/zh_cn/AWSEC2/latest/WindowsGuide/enhanced-networking-ena.html
|
|
|
|
|
if is_nt_6_1; then
|
2023-07-06 22:20:09 +08:00
|
|
|
|
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
|
|
|
|
|
|
2023-10-22 17:11:35 +08:00
|
|
|
|
elif virt-what | grep xen &&
|
2023-07-05 21:40:00 +08:00
|
|
|
|
[ "$arch_wim" = x86_64 ]; then
|
2023-07-03 23:11:10 +08:00
|
|
|
|
# aws xen
|
|
|
|
|
# 只有 64 位驱动
|
|
|
|
|
# 未测试
|
2024-01-22 19:00:17 +08:00
|
|
|
|
# https://docs.aws.amazon.com/zh_cn/AWSEC2/latest/WindowsGuide/xen-drivers-overview.html
|
2023-10-22 19:07:12 +08:00
|
|
|
|
apk add msitools
|
2023-07-06 22:20:09 +08:00
|
|
|
|
|
2024-01-22 19:00:17 +08:00
|
|
|
|
if is_nt_6_1; then
|
2023-07-06 22:20:09 +08:00
|
|
|
|
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/
|
|
|
|
|
|
2023-10-22 17:11:35 +08:00
|
|
|
|
elif false && virt-what | grep xen &&
|
2023-07-05 21:40:00 +08:00
|
|
|
|
[ "$arch_wim" != arm64 ]; then
|
2023-07-03 23:11:10 +08:00
|
|
|
|
# xen
|
2023-07-05 21:40:00 +08:00
|
|
|
|
# 有 x86 x64,没arm64驱动
|
2023-10-22 17:11:35 +08:00
|
|
|
|
# 没签名,暂时用aws的驱动代替
|
|
|
|
|
# https://lore.kernel.org/xen-devel/E1qKMmq-00035B-SS@xenbits.xenproject.org/
|
2023-07-03 23:11:10 +08:00
|
|
|
|
# https://xenbits.xenproject.org/pvdrivers/win/
|
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
|
2023-09-29 13:53:23 +08:00
|
|
|
|
download https://xenbits.xenproject.org/pvdrivers/win/$part.tar $drv/$part.tar
|
2023-07-03 23:11:10 +08:00
|
|
|
|
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
|
2023-07-05 21:40:00 +08:00
|
|
|
|
# x86 x64 arm64 都有
|
2023-07-03 23:11:10 +08:00
|
|
|
|
# https://fedorapeople.org/groups/virt/virtio-win/direct-downloads/stable-virtio/
|
2024-01-22 19:00:17 +08:00
|
|
|
|
|
|
|
|
|
# 没有 vista 文件夹
|
|
|
|
|
case "$nt_ver" in
|
|
|
|
|
6.0) sys=2k8 ;;
|
|
|
|
|
6.1) sys=w7 ;;
|
|
|
|
|
6.2) sys=w8 ;;
|
|
|
|
|
6.3) sys=w8.1 ;;
|
|
|
|
|
10.0) sys=w10 ;;
|
2023-05-25 20:15:12 +08:00
|
|
|
|
esac
|
|
|
|
|
|
2023-06-04 19:05:17 +08:00
|
|
|
|
# https://github.com/virtio-win/virtio-win-pkg-scripts/issues/40
|
|
|
|
|
# https://github.com/virtio-win/virtio-win-pkg-scripts/issues/61
|
2024-01-22 19:00:17 +08:00
|
|
|
|
case "$sys" in
|
|
|
|
|
vista | w7 | 2k8 | 2k8R2) dir=archive-virtio/virtio-win-0.1.173-9 ;;
|
|
|
|
|
w8 | w8.1 | 2k12 | 2k12R2) dir=archive-virtio/virtio-win-0.1.215-1 ;;
|
2023-06-04 19:05:17 +08:00
|
|
|
|
*) 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
|
2023-11-09 22:30:21 +08:00
|
|
|
|
mount -o ro $drv/virtio-win.iso $drv/virtio
|
2023-09-28 23:51:29 +08:00
|
|
|
|
|
|
|
|
|
apk add dmidecode
|
|
|
|
|
dmi=$(dmidecode)
|
|
|
|
|
|
|
|
|
|
if echo "$dmi" | grep -Eiw "Google Compute Engine|GoogleCloud"; then
|
|
|
|
|
gce_repo=https://packages.cloud.google.com/yuck
|
|
|
|
|
download $gce_repo/repos/google-compute-engine-stable/index /tmp/gce.json
|
|
|
|
|
# gga 好像只是用于调节后台vnc分辨率
|
|
|
|
|
for name in gvnic gga; do
|
|
|
|
|
mkdir -p $drv/gce/$name
|
|
|
|
|
link=$(grep -o "/pool/.*-google-compute-engine-driver-$name\.goo" /tmp/gce.json)
|
|
|
|
|
wget $gce_repo$link -O- | tar -xzf- -C $drv/gce/$name
|
|
|
|
|
done
|
|
|
|
|
|
2024-01-22 19:00:17 +08:00
|
|
|
|
# 没有 win6.0 文件夹
|
|
|
|
|
sys_gce=win$nt_ver
|
2023-09-28 23:51:29 +08:00
|
|
|
|
fi
|
2023-05-25 20:15:12 +08:00
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
# 修改应答文件
|
2023-12-16 16:41:34 +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-12-16 16:41:34 +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-12-16 16:41:34 +08:00
|
|
|
|
sed -i "s|%installto_partitionid%|3|" /tmp/autounattend.xml
|
2023-05-28 00:52:48 +08:00
|
|
|
|
else
|
2023-12-16 16:41:34 +08:00
|
|
|
|
sed -i "s|%installto_partitionid%|1|" /tmp/autounattend.xml
|
|
|
|
|
fi
|
2023-06-04 19:09:40 +08:00
|
|
|
|
|
2024-01-20 16:04:06 +08:00
|
|
|
|
# 评估版 iso 需要删除 autounattend.xml 里面的 <Key />
|
|
|
|
|
# 否则会出现 Windows Cannot find Microsoft software license terms
|
|
|
|
|
# shellcheck disable=SC2010
|
|
|
|
|
if ei_cfg="$(ls -d /os/installer/sources/* | grep -i ei.cfg)"; then
|
|
|
|
|
if grep -i EVAL "$ei_cfg"; then
|
|
|
|
|
sed -i '/<Key \/>/d' /tmp/autounattend.xml
|
|
|
|
|
fi
|
|
|
|
|
fi
|
|
|
|
|
|
2023-05-28 00:52:48 +08:00
|
|
|
|
# 挂载 boot.wim
|
2023-06-21 23:34:25 +08:00
|
|
|
|
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
|
2023-09-28 23:51:29 +08:00
|
|
|
|
path=$2
|
|
|
|
|
|
2023-07-03 23:11:10 +08:00
|
|
|
|
[ -n "$path" ] && filter="-ipath $path" || filter=""
|
|
|
|
|
find $src \
|
|
|
|
|
$filter \
|
2023-07-06 21:49:31 +08:00
|
|
|
|
-type f \
|
2023-07-03 23:11:10 +08:00
|
|
|
|
-not -iname "*.pdb" \
|
|
|
|
|
-not -iname "dpinst.exe" \
|
2023-09-28 23:51:29 +08:00
|
|
|
|
-exec cp -rfv {} /wim/drivers \;
|
2023-07-03 23:11:10 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
# 添加驱动
|
|
|
|
|
mkdir -p /wim/drivers
|
|
|
|
|
|
2023-09-28 23:51:29 +08:00
|
|
|
|
[ -d $drv/virtio ] && cp_drivers $drv/virtio "*/$sys/$arch/*"
|
|
|
|
|
[ -d $drv/aws ] && cp_drivers $drv/aws
|
|
|
|
|
[ -d $drv/xen ] && cp_drivers $drv/xen "*/$arch_xen/*"
|
|
|
|
|
[ -d $drv/gce ] && {
|
|
|
|
|
[ "$arch_wim" = x86 ] && gvnic_suffix=-32 || gvnic_suffix=
|
|
|
|
|
cp_drivers $drv/gce/gvnic "*/$sys_gce$gvnic_suffix/*"
|
|
|
|
|
# gga 驱动不分32/64位
|
|
|
|
|
cp_drivers $drv/gce/gga "*/$sys_gce/*"
|
|
|
|
|
}
|
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-12-16 16:41:34 +08:00
|
|
|
|
# 移除注释,否则 windows-setup.bat 重新生成的 autounattend.xml 有问题
|
2023-11-15 20:51:44 +08:00
|
|
|
|
apk add xmlstarlet
|
2023-12-16 16:41:34 +08:00
|
|
|
|
xmlstarlet ed -d '//comment()' /tmp/autounattend.xml >/wim/autounattend.xml
|
|
|
|
|
unix2dos /wim/autounattend.xml
|
2023-11-15 20:51:44 +08:00
|
|
|
|
|
|
|
|
|
# 复制安装脚本
|
|
|
|
|
# https://slightlyovercomplicated.com/2016/11/07/windows-pe-startup-sequence-explained/
|
|
|
|
|
mv /wim/setup.exe /wim/setup.exe.disabled
|
2023-12-14 19:03:59 +08:00
|
|
|
|
|
|
|
|
|
# 如果有重复的 Windows/System32 文件夹,会提示找不到 winload.exe 无法引导
|
|
|
|
|
# win7 win10 是 Windows/System32
|
|
|
|
|
# win2016 是 windows/system32
|
|
|
|
|
system32_dir=$(ls -d /wim/*/*32)
|
|
|
|
|
download $confhome/windows-setup.bat $system32_dir/startnet.cmd
|
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
|
2023-11-03 23:50:02 +08:00
|
|
|
|
if [[ "$install_wim" = '*.wim' ]]; then
|
2023-07-06 21:52:48 +08:00
|
|
|
|
wimmountrw $install_wim "$image_name" /wim/
|
2023-11-02 23:53:15 +08:00
|
|
|
|
if false; then
|
2023-12-16 16:41:34 +08:00
|
|
|
|
# 使用 autounattend.xml
|
2023-11-02 23:53:15 +08:00
|
|
|
|
# win7 在此阶段找不到网卡
|
|
|
|
|
download $confhome/windows-resize.bat /wim/windows-resize.bat
|
|
|
|
|
create_win_set_netconf_script /wim/windows-set-netconf.bat
|
|
|
|
|
else
|
|
|
|
|
modify_windows /wim
|
|
|
|
|
fi
|
2023-07-06 21:52:48 +08:00
|
|
|
|
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-12-02 23:26:09 +08:00
|
|
|
|
# 添加 netboot.efi 备用
|
|
|
|
|
download_netboot_xyz_efi() {
|
|
|
|
|
dir=$1
|
|
|
|
|
|
|
|
|
|
file=$dir/netboot.xyz.efi
|
|
|
|
|
if [ "$(uname -m)" = aarch64 ]; then
|
|
|
|
|
download https://boot.netboot.xyz/ipxe/netboot.xyz-arm64.efi $file
|
|
|
|
|
else
|
|
|
|
|
download https://boot.netboot.xyz/ipxe/netboot.xyz.efi $file
|
|
|
|
|
fi
|
|
|
|
|
}
|
|
|
|
|
|
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
|
|
|
|
|
apk add grub-efi efibootmgr
|
|
|
|
|
grub-install --efi-directory=/os/boot/efi --boot-directory=/os/boot
|
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" {
|
2023-05-03 23:22:59 +08:00
|
|
|
|
# https://bugs.launchpad.net/ubuntu/+source/grub2/+bug/1851311
|
2023-05-28 00:52:48 +08:00
|
|
|
|
# rmmod tpm
|
2023-11-09 22:30:21 +08:00
|
|
|
|
insmod all_video
|
2023-05-03 22:22:21 +08:00
|
|
|
|
search --no-floppy --label --set=root installer
|
2023-07-08 15:55:06 +08:00
|
|
|
|
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" {
|
2023-11-09 22:30:21 +08:00
|
|
|
|
insmod all_video
|
2023-05-03 22:22:21 +08:00
|
|
|
|
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出错
|
2023-10-22 19:07:12 +08:00
|
|
|
|
# do 机器第二次运行会报错
|
|
|
|
|
hwclock -s || true
|
2023-07-25 00:21:08 +08:00
|
|
|
|
|
|
|
|
|
# 设置密码,安装并打开 ssh
|
|
|
|
|
echo root:123@@@ | chpasswd
|
|
|
|
|
printf '\nyes' | setup-sshd
|
|
|
|
|
|
|
|
|
|
extract_env_from_cmdline
|
|
|
|
|
# shellcheck disable=SC2154
|
2023-10-31 22:49:49 +08:00
|
|
|
|
if [ "$hold" = 1 ]; then
|
2023-07-25 00:21:08 +08:00
|
|
|
|
exit
|
|
|
|
|
fi
|
|
|
|
|
|
2023-10-22 17:00:32 +08:00
|
|
|
|
mod_motd
|
2023-07-25 00:21:08 +08:00
|
|
|
|
setup_tty_and_log
|
2023-11-28 23:59:17 +08:00
|
|
|
|
cat /proc/cmdline
|
2023-07-25 00:21:08 +08:00
|
|
|
|
clear_previous
|
2023-08-25 23:36:32 +08:00
|
|
|
|
add_community_repo
|
2023-07-25 00:21:08 +08:00
|
|
|
|
|
|
|
|
|
# 找到主硬盘
|
2023-10-08 21:46:45 +08:00
|
|
|
|
xda=$(get_xda)
|
|
|
|
|
|
2023-07-25 00:21:08 +08:00
|
|
|
|
if [ "$distro" != "alpine" ]; then
|
|
|
|
|
setup_nginx_if_enough_ram
|
2023-12-02 23:26:09 +08:00
|
|
|
|
setup_udev_util_linux
|
2023-07-25 00:21:08 +08:00
|
|
|
|
fi
|
|
|
|
|
|
2023-10-22 18:50:54 +08:00
|
|
|
|
# dd qemu 切换成云镜像模式,暂时没用到
|
|
|
|
|
if [ "$distro" = "dd" ] && [ "$img_type" = "qemu" ]; then
|
|
|
|
|
cloud_image=1
|
|
|
|
|
fi
|
|
|
|
|
|
2023-07-25 00:21:08 +08:00
|
|
|
|
if [ "$distro" = "alpine" ]; then
|
|
|
|
|
install_alpine
|
2023-10-22 18:50:54 +08:00
|
|
|
|
elif [ "$distro" = "dd" ] && [ "$img_type" != "qemu" ]; then
|
|
|
|
|
dd_gzip_xz
|
2023-10-25 14:19:24 +08:00
|
|
|
|
modify_os_on_disk windows
|
2023-07-25 00:21:08 +08:00
|
|
|
|
elif is_use_cloud_image; then
|
2023-10-22 18:50:54 +08:00
|
|
|
|
if [ "$img_type" = "qemu" ]; then
|
2023-07-28 21:27:16 +08:00
|
|
|
|
create_part
|
2023-10-22 18:50:54 +08:00
|
|
|
|
download_qcow
|
|
|
|
|
# 这几个系统云镜像系统盘是8~9g xfs,而我们的目标是能在5g硬盘上运行,因此改成复制系统文件
|
|
|
|
|
if [ "$distro" = centos ] || [ "$distro" = alma ] || [ "$distro" = rocky ]; then
|
|
|
|
|
install_qcow_el
|
|
|
|
|
else
|
|
|
|
|
# debian ubuntu fedora opensuse arch gentoo
|
|
|
|
|
dd_qcow
|
|
|
|
|
resize_after_install_cloud_image
|
2023-10-25 14:19:24 +08:00
|
|
|
|
modify_os_on_disk linux
|
2023-10-22 18:50:54 +08:00
|
|
|
|
fi
|
|
|
|
|
else
|
|
|
|
|
# gzip xz 格式的云镜像,暂时没用到
|
|
|
|
|
dd_gzip_xz
|
|
|
|
|
resize_after_install_cloud_image
|
2023-10-25 14:19:24 +08:00
|
|
|
|
modify_os_on_disk linux
|
2023-07-25 00:21:08 +08:00
|
|
|
|
fi
|
2023-07-28 21:27:16 +08:00
|
|
|
|
else
|
2023-10-22 18:50:54 +08:00
|
|
|
|
# 安装模式: windows windows ubuntu 红帽
|
2023-07-29 00:48:37 +08:00
|
|
|
|
create_part
|
2023-10-22 18:50:54 +08:00
|
|
|
|
mount_part_for_install_mode
|
2023-07-29 00:48:37 +08:00
|
|
|
|
if [ "$distro" = "windows" ]; then
|
|
|
|
|
install_windows
|
|
|
|
|
else
|
|
|
|
|
install_redhat_ubuntu
|
|
|
|
|
fi
|
2023-07-25 00:21:08 +08:00
|
|
|
|
fi
|
2023-12-02 23:26:09 +08:00
|
|
|
|
|
2023-12-22 23:56:12 +08:00
|
|
|
|
# alpine 因内存容量问题,单独处理
|
|
|
|
|
if is_efi && [ "$distro" != "alpine" ]; then
|
|
|
|
|
del_invalid_efi_entry
|
|
|
|
|
add_fallback_efi_to_nvram
|
|
|
|
|
fi
|
|
|
|
|
|
2023-12-02 23:26:09 +08:00
|
|
|
|
echo 'done'
|
2023-10-31 22:49:49 +08:00
|
|
|
|
if [ "$hold" = 2 ]; then
|
2023-07-25 00:21:08 +08:00
|
|
|
|
exit
|
2023-05-03 22:22:21 +08:00
|
|
|
|
fi
|
2023-10-28 21:46:38 +08:00
|
|
|
|
|
2023-10-31 22:49:49 +08:00
|
|
|
|
# 让 web ssh 输出全部内容
|
|
|
|
|
for i in $(seq 10); do
|
|
|
|
|
echo
|
|
|
|
|
done
|
2023-10-28 21:46:38 +08:00
|
|
|
|
sleep 5
|
2023-05-03 22:22:21 +08:00
|
|
|
|
reboot
|