reinstall/reinstall.sh

3144 lines
102 KiB
Bash
Raw Normal View History

2022-09-25 21:57:47 +08:00
#!/bin/bash
2023-09-16 20:00:05 +08:00
# shellcheck disable=SC2086
2023-06-18 21:27:22 +08:00
set -eE
2023-05-03 22:22:21 +08:00
confhome=https://raw.githubusercontent.com/bin456789/reinstall/main
2024-04-05 19:23:55 +08:00
github_proxy=https://mirror.ghproxy.com/https://raw.githubusercontent.com
2022-09-25 21:57:47 +08:00
# https://www.gnu.org/software/gettext/manual/html_node/The-LANGUAGE-variable.html
export LC_ALL=C
# 处理部分用户用 su 切换成 root 导致环境变量没 sbin 目录
# 不要漏了最后的 $PATH否则会找不到 windows 系统程序例如 diskpart
export PATH=/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin:$PATH
2024-06-14 23:41:15 +08:00
this_script=$(readlink -f "$0")
2023-09-16 20:00:05 +08:00
trap 'trap_err $LINENO $?' ERR
trap_err() {
line_no=$1
ret_no=$2
error "Line $line_no return $ret_no"
sed -n "$line_no"p "$this_script"
}
2023-06-18 21:27:22 +08:00
2023-05-03 22:22:21 +08:00
usage_and_exit() {
2024-03-18 23:56:10 +08:00
if is_in_windows; then
reinstall____=' reinstall.bat'
else
reinstall____='./reinstall.sh'
fi
2023-07-28 22:53:55 +08:00
cat <<EOF
Usage: $reinstall____ centos 9
anolis 7|8
alma 8|9
rocky 8|9
oracle 8|9
redhat 8|9 --img='http://xxx.qcow2'
opencloudos 8|9
fedora 39|40
debian 11|12
openeuler 20.03|22.03|24.03
ubuntu 20.04|22.04|24.04
alpine 3.17|3.18|3.19|3.20
opensuse 15.5|15.6|tumbleweed
2024-05-03 21:37:07 +08:00
kali
2024-03-18 23:56:10 +08:00
arch
gentoo
dd --img='http://xxx.gzip' or .xz
windows --image-name='windows xxx yyy' --lang=xx-yy
windows --image-name='windows xxx yyy' --iso='http://xxx.iso'
2024-03-18 23:56:10 +08:00
netboot.xyz
2023-12-16 16:41:34 +08:00
2024-03-05 23:01:24 +08:00
Manual: https://github.com/bin456789/reinstall
2024-04-03 21:53:52 +08:00
2023-07-28 22:53:55 +08:00
EOF
2023-05-03 22:22:21 +08:00
exit 1
}
2023-06-18 21:27:22 +08:00
info() {
2023-10-22 19:07:12 +08:00
upper=$(to_upper <<<"$@")
2023-09-16 20:00:05 +08:00
echo_color_text '\e[32m' "***** $upper *****"
}
warn() {
echo_color_text '\e[33m' "Warn: $*"
2023-06-18 21:27:22 +08:00
}
error() {
2023-09-16 20:00:05 +08:00
echo_color_text '\e[31m' "Error: $*"
}
echo_color_text() {
color="$1"
shift
plain="\e[0m"
echo -e "$color$*$plain"
2023-06-18 21:27:22 +08:00
}
2023-06-04 22:10:16 +08:00
error_and_exit() {
2023-06-18 21:27:22 +08:00
error "$@"
2023-06-04 22:10:16 +08:00
exit 1
}
2023-06-18 21:27:22 +08:00
curl() {
2023-09-10 22:23:02 +08:00
# 添加 -f, --fail不然 404 退出码也为0
2023-10-22 19:07:12 +08:00
# 32位 cygwin 已停止更新,证书可能有问题,先添加 --insecure
2024-03-05 23:01:24 +08:00
# centos 7 curl 不支持 --retry-connrefused --retry-all-errors
# 因此手动 retry
2023-11-28 20:03:03 +08:00
grep -o 'http[^ ]*' <<<"$@" >&2
2024-03-05 23:01:24 +08:00
for i in $(seq 5); do
if command curl --insecure --connect-timeout 10 -f "$@"; then
return
else
ret=$?
2024-07-09 23:37:51 +08:00
# 403 404 错误,或者达到重试次数
if [ $ret -eq 22 ] || [ $i -eq 5 ]; then
2024-03-05 23:01:24 +08:00
return $ret
fi
2024-07-09 23:37:51 +08:00
sleep 1
2024-03-05 23:01:24 +08:00
fi
done
2023-06-18 21:27:22 +08:00
}
2023-05-03 22:22:21 +08:00
is_in_china() {
2023-05-13 00:14:46 +08:00
if [ -z $_is_in_china ]; then
2024-03-01 22:01:22 +08:00
# 部分地区 www.cloudflare.com 被墙
curl -L http://dash.cloudflare.com/cdn-cgi/trace |
2023-10-22 19:07:12 +08:00
grep -qx 'loc=CN' && _is_in_china=true ||
_is_in_china=false
2023-05-13 00:14:46 +08:00
fi
2023-10-22 19:07:12 +08:00
$_is_in_china
2023-05-03 22:22:21 +08:00
}
2023-06-18 21:27:22 +08:00
is_in_windows() {
[ "$(uname -o)" = Cygwin ] || [ "$(uname -o)" = Msys ]
}
2023-07-15 23:25:42 +08:00
is_in_alpine() {
[ -f /etc/alpine-release ]
}
2023-07-18 00:22:51 +08:00
is_use_cloud_image() {
[ -n "$cloud_image" ] && [ "$cloud_image" = 1 ]
}
2023-07-28 21:27:16 +08:00
is_use_dd() {
[ "$distro" = dd ]
}
2023-09-10 22:23:03 +08:00
2023-07-23 11:46:31 +08:00
is_os_in_btrfs() {
mount | grep -qw 'on / type btrfs'
2023-07-23 11:46:31 +08:00
}
is_os_in_subvol() {
subvol=$(awk '($2=="/") { print $i }' /proc/mounts | grep -o 'subvol=[^ ]*' | cut -d= -f2)
[ "$subvol" != / ]
}
get_os_part() {
awk '($2=="/") { print $1 }' /proc/mounts
}
cp_to_btrfs_root() {
mount_dir=$tmp/reinstall-btrfs-root
if ! grep -q $mount_dir /proc/mounts; then
mkdir -p $mount_dir
mount "$(get_os_part)" $mount_dir -t btrfs -o subvol=/
2023-07-23 11:46:31 +08:00
fi
cp -rf "$@" $tmp/reinstall-btrfs-root
2023-07-23 11:46:31 +08:00
}
is_host_has_ipv4_and_ipv6() {
2023-09-10 22:23:03 +08:00
host=$1
install_pkg dig
# dig会显示cname结果cname结果以.结尾grep -v '\.$' 用于去除 cname 结果
res=$(dig +short $host A $host AAAA | grep -v '\.$')
# 有.表示有ipv4地址有:表示有ipv6地址
grep -q \. <<<$res && grep -q : <<<$res
}
2023-11-09 22:27:07 +08:00
is_netboot_xyz() {
[ "$distro" = netboot.xyz ]
}
2023-12-04 22:19:59 +08:00
is_alpine_live() {
[ "$distro" = alpine ] && [ "$hold" = 1 ]
}
2023-11-09 22:27:07 +08:00
is_have_initrd() {
! is_netboot_xyz
}
2024-05-03 21:37:07 +08:00
is_use_firmware() {
2024-05-08 22:17:33 +08:00
# shellcheck disable=SC2154
[ "$nextos_distro" = debian ] && ! is_virt
2024-05-03 21:37:07 +08:00
}
get_host_by_url() {
cut -d/ -f3 <<<$1
}
get_function() {
declare -f "$1"
}
2023-10-22 19:07:12 +08:00
get_function_content() {
declare -f "$1" | sed '1d;2d;$d'
}
2023-09-16 20:00:05 +08:00
insert_into_file() {
file=$1
location=$2
regex_to_find=$3
line_num=$(grep -E -n "$regex_to_find" "$file" | cut -d: -f1)
2023-10-22 19:07:12 +08:00
found_count=$(echo "$line_num" | wc -l)
if [ ! "$found_count" -eq 1 ]; then
2023-09-16 20:00:05 +08:00
return 1
fi
2023-10-22 19:07:12 +08:00
case "$location" in
before) line_num=$((line_num - 1)) ;;
after) ;;
*) return 1 ;;
esac
2023-09-16 20:00:05 +08:00
sed -i "${line_num}r /dev/stdin" "$file"
}
2023-06-04 22:10:16 +08:00
test_url() {
2023-09-16 20:00:06 +08:00
test_url_real false "$@"
}
test_url_grace() {
test_url_real true "$@"
}
test_url_real() {
grace=$1
url=$2
expect_type=$3
var_to_eval=$4
2023-06-18 21:27:22 +08:00
info test url
2023-09-16 20:00:06 +08:00
failed() {
$grace && return 1
error_and_exit "$@"
}
2023-09-10 22:23:02 +08:00
tmp_file=$tmp/reinstall-img-test
# 有的服务器不支持 rangecurl会下载整个文件
# 用 dd 限制下载 1M
# 并过滤 curl 23 错误dd限制了空间
# 也可用 ulimit -f 但好像 cygwin 不支持
2024-03-17 00:08:48 +08:00
echo $url
for i in $(seq 5 -1 0); do
if command curl --insecure --connect-timeout 10 -Lfr 0-1048575 "$url" \
1> >(dd bs=1M count=1 of=$tmp_file iflag=fullblock 2>/dev/null) \
2> >(grep -v 'curl: (23)' >&2); then
break
else
ret=$?
msg="$url not accessible"
case $ret in
22) failed "$msg" ;; # 403 404
23) break ;; # 限制了空间
*) [ $i -eq 0 ] && failed "$msg" ;; # 其他错误
esac
sleep 1
fi
2024-03-17 00:08:48 +08:00
done
2023-06-04 22:10:16 +08:00
2023-07-15 23:28:20 +08:00
if [ -n "$expect_type" ]; then
# gzip的mime有很多种写法
# centos7中显示为 x-gzip在其他系统中显示为 gzip可能还有其他
# 所以不用mime判断
# https://www.digipres.org/formats/sources/tika/formats/#application/gzip
2023-07-15 23:28:20 +08:00
# 有些 file 版本输出的是 # ISO 9660 CD-ROM filesystem data ,要去掉开头的井号
install_pkg file
2023-10-22 19:07:12 +08:00
real_type=$(file -b $tmp_file | sed 's/^# //' | cut -d' ' -f1 | to_lower)
2023-07-15 23:28:20 +08:00
[ -n "$var_to_eval" ] && eval $var_to_eval=$real_type
2023-06-04 22:10:16 +08:00
2023-09-10 22:23:02 +08:00
if ! grep -wo "$real_type" <<<"$expect_type"; then
2023-09-16 20:00:06 +08:00
failed "$url expected: $expect_type. actual: $real_type."
2023-07-15 23:28:20 +08:00
fi
2023-06-04 22:10:16 +08:00
fi
}
2023-06-18 21:27:22 +08:00
add_community_repo_for_alpine() {
2023-08-25 13:04:06 +08:00
# 先检查原来的repo是不是egde
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
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-06-18 21:27:22 +08:00
}
2023-10-22 16:39:35 +08:00
assert_not_in_container() {
_error_and_exit() {
error_and_exit "Not Supported OS in Container.\nPlease use https://github.com/LloydAsp/OsMutation"
}
is_in_windows && return
if is_have_cmd systemd-detect-virt; then
2024-03-05 23:01:24 +08:00
if systemd-detect-virt -qc; then
2023-10-22 16:39:35 +08:00
_error_and_exit
fi
else
2024-03-05 23:01:24 +08:00
if [ -d /proc/vz ] || grep -q container=lxc /proc/1/environ; then
2023-10-22 16:39:35 +08:00
_error_and_exit
fi
fi
}
2023-06-18 21:27:22 +08:00
is_virt() {
2023-11-09 22:27:07 +08:00
if [ -z "$_is_virt" ]; then
if is_in_windows; then
# https://github.com/systemd/systemd/blob/main/src/basic/virt.c
# https://sources.debian.org/src/hw-detect/1.159/hw-detect.finish-install.d/08hw-detect/
vmstr='VMware|Virtual|Virtualization|VirtualBox|VMW|Hyper-V|Bochs|QEMU|KVM|OpenStack|KubeVirt|innotek|Xen|Parallels|BHYVE'
for name in ComputerSystem BIOS BaseBoard; do
if wmic $name get /format:list | grep -Eiw $vmstr; then
2023-11-09 22:27:07 +08:00
_is_virt=true
break
fi
done
# 没有风扇和温度信息,大概是虚拟机
if [ -z "$_is_virt" ] &&
! wmic /namespace:'\\root\cimv2' PATH Win32_Fan 2>/dev/null | grep -q Name &&
! wmic /namespace:'\\root\wmi' PATH MSAcpi_ThermalZoneTemperature 2>/dev/null | grep -q Name; then
_is_virt=true
2023-11-09 22:27:07 +08:00
fi
else
# aws t4g debian 11
# systemd-detect-virt: 为 none即使装了dmidecode
# virt-what: 未装 deidecode时结果为空装了deidecode后结果为aws
# 所以综合两个命令的结果来判断
if is_have_cmd systemd-detect-virt && systemd-detect-virt -v; then
_is_virt=true
fi
2024-01-27 23:17:50 +08:00
2023-11-09 22:27:07 +08:00
if [ -z "$_is_virt" ]; then
# debian 安装 virt-what 不会自动安装 dmidecode因此结果有误
install_pkg dmidecode virt-what
# virt-what 返回值始终是0所以用是否有输出作为判断
if [ -n "$(virt-what)" ]; then
_is_virt=true
fi
fi
fi
if [ -z "$_is_virt" ]; then
_is_virt=false
2023-06-18 21:27:22 +08:00
fi
2023-12-02 22:46:16 +08:00
echo "vm: $_is_virt"
2023-06-18 21:27:22 +08:00
fi
2023-11-09 22:27:07 +08:00
$_is_virt
2023-06-18 21:27:22 +08:00
}
2024-04-05 22:45:59 +08:00
# sr-latn-rs 到 sr-latn
en_us() {
echo "$lang" | awk -F- '{print $1"-"$2}'
# zh-hk 可回落到 zh-tw
if [ "$lang" = zh-hk ]; then
echo zh-tw
fi
}
2024-04-06 21:35:48 +08:00
# fr-ca 到 ca
2024-04-05 22:45:59 +08:00
us() {
# 葡萄牙准确对应 pp
if [ "$lang" = pt-pt ]; then
echo pp
return
fi
# 巴西准确对应 pt
if [ "$lang" = pt-br ]; then
echo pt
return
fi
echo "$lang" | awk -F- '{print $2}'
# hk 额外回落到 tw
if [ "$lang" = zh-hk ]; then
echo tw
fi
}
# fr-ca 到 fr-fr
en_en() {
echo "$lang" | awk -F- '{print $1"-"$1}'
# en-gb 额外回落到 en-us
if [ "$lang" = en-gb ]; then
echo en-us
fi
}
2024-04-06 21:35:48 +08:00
# fr-ca 到 fr
2024-04-05 22:45:59 +08:00
en() {
# 巴西/葡萄牙回落到葡萄牙语
2024-04-16 23:34:27 +08:00
if [ "$lang" = pt-br ] || [ "$lang" = pt-pt ]; then
2024-04-05 22:45:59 +08:00
echo "pp"
return
fi
echo "$lang" | awk -F- '{print $1}'
}
english() {
case "$lang" in
ar-sa) echo Arabic ;;
bg-bg) echo Bulgarian ;;
cs-cz) echo Czech ;;
da-dk) echo Danish ;;
de-de) echo German ;;
el-gr) echo Greek ;;
en-gb) echo Eng_Intl ;;
en-us) echo English ;;
es-es) echo Spanish ;;
es-mx) echo Spanish_Latam ;;
et-ee) echo Estonian ;;
fi-fi) echo Finnish ;;
fr-ca) echo FrenchCanadian ;;
fr-fr) echo French ;;
he-il) echo Hebrew ;;
hr-hr) echo Croatian ;;
hu-hu) echo Hungarian ;;
it-it) echo Italian ;;
ja-jp) echo Japanese ;;
ko-kr) echo Korean ;;
lt-lt) echo Lithuanian ;;
lv-lv) echo Latvian ;;
nb-no) echo Norwegian ;;
nl-nl) echo Dutch ;;
pl-pl) echo Polish ;;
pt-pt) echo Portuguese ;;
pt-br) echo Brazilian ;;
ro-ro) echo Romanian ;;
ru-ru) echo Russian ;;
sk-sk) echo Slovak ;;
sl-si) echo Slovenian ;;
sr-latn | sr-latn-rs) echo Serbian_Latin ;;
sv-se) echo Swedish ;;
th-th) echo Thai ;;
tr-tr) echo Turkish ;;
uk-ua) echo Ukrainian ;;
zh-cn) echo ChnSimp ;;
zh-hk | zh-tw) echo ChnTrad ;;
esac
}
parse_windows_image_name() {
set -- $image_name
if ! [ "$1" = windows ]; then
return 1
fi
shift
if [ "$1" = server ]; then
server=server
shift
fi
2024-04-05 22:45:59 +08:00
version=$1
shift
if [ "$1" = r2 ]; then
version+=" r2"
shift
fi
edition=
while [ $# -gt 0 ]; do
case "$1" in
2024-04-05 22:45:59 +08:00
# windows 10 enterprise n ltsc 2021
k | n | kn) ;;
*)
if [ -n "$edition" ]; then
edition+=" "
fi
edition+="$1"
;;
esac
shift
done
}
is_have_arm_version() {
case "$version" in
10)
case "$edition" in
2024-05-22 22:32:43 +08:00
pro | education | enterprise | 'pro education' | 'pro for workstations') return ;;
2024-04-05 22:45:59 +08:00
'iot enterprise') return ;;
2024-05-22 22:32:43 +08:00
'enterprise ltsc 2021' | 'iot enterprise ltsc 2021') return ;;
2024-04-05 22:45:59 +08:00
esac
;;
11)
case "$edition" in
2024-05-22 22:32:43 +08:00
pro | education | enterprise | 'pro education' | 'pro for workstations') return ;;
'iot enterprise' | 'iot enterprise subscription') return ;;
2024-05-22 22:32:43 +08:00
'enterprise ltsc 2024' | 'iot enterprise ltsc 2024' | 'iot enterprise ltsc 2024 subscription') return ;;
2024-04-05 22:45:59 +08:00
esac
;;
esac
return 1
}
find_windows_iso() {
parse_windows_image_name || error_and_exit "--image-name wrong: $image_name"
if ! [ "$version" = 8.1 ] && [ -z "$edition" ]; then
error_and_exit "Edition is not set."
fi
if [ "$basearch" = 'aarch64' ] && ! is_have_arm_version; then
error_and_exit "No ARM iso for this Windows Version."
fi
if [ -z "$lang" ]; then
lang=en-us
fi
langs="$lang $(en_us) $(us) $(en_en) $(en)"
langs=$(echo "$langs" | xargs -n 1 | awk '!seen[$0]++')
full_lang=$(english)
case "$basearch" in
x86_64) arch_win=x64 ;;
aarch64) arch_win=arm64 ;;
esac
get_windows_iso_links
get_windows_iso_link
}
get_windows_iso_links() {
2024-04-05 23:21:07 +08:00
get_label_msdn() {
2024-04-05 22:45:59 +08:00
if [ -n "$server" ]; then
case "$version" in
2008 | '2008 r2')
case "$edition" in
serverweb | serverwebcore) echo _ ;;
serverstandard | serverstandardcore) echo _ ;;
serverenterprise | serverenterprisecore) echo _ ;;
serverdatacenter | serverdatacentercore) echo _ ;;
esac
;;
2024-04-06 00:16:10 +08:00
'2012 r2' | \
2016 | 2019 | 2022 | 2025)
2024-04-05 22:45:59 +08:00
case "$edition" in
serverstandard | serverstandardcore) echo _ ;;
serverdatacenter | serverdatacentercore) echo _ ;;
esac
;;
esac
else
case "$version" in
vista)
case "$edition" in
2024-04-06 21:35:48 +08:00
starter)
case "$arch_win" in
x86) echo _ ;;
esac
;;
2024-04-05 22:45:59 +08:00
homebasic | homepremium | business | ultimate) echo _ ;;
2024-04-06 21:35:48 +08:00
enterprise) echo enterprise ;;
2024-04-05 22:45:59 +08:00
esac
;;
7)
case "$edition" in
2024-04-06 21:35:48 +08:00
starter)
case "$arch_win" in
x86) echo ultimate ;;
esac
;;
2024-04-05 22:45:59 +08:00
professional) echo professional ;;
homebasic | homepremium | ultimate) echo ultimate ;;
2024-04-06 21:35:48 +08:00
enterprise) echo enterprise ;;
2024-04-05 22:45:59 +08:00
esac
;;
8.1)
case "$edition" in
'') echo _ ;;
pro) echo pro ;;
enterprise) echo enterprise ;;
esac
;;
10)
case "$edition" in
home | 'home single language') echo consumer ;;
2024-05-22 22:32:43 +08:00
pro | education | enterprise | 'pro education' | 'pro for workstations') echo business ;;
2024-04-05 22:45:59 +08:00
'iot enterprise') echo 'iot enterprise' ;;
2024-04-06 21:35:48 +08:00
'enterprise 2015 ltsb' | 'enterprise 2016 ltsb' | 'enterprise ltsc 2019') echo "$edition" ;;
2024-04-05 22:45:59 +08:00
'enterprise ltsc 2021')
# arm64 的 enterprise ltsc 2021 要下载 iot enterprise ltsc 2021 iso
case "$arch_win" in
arm64) echo 'iot enterprise ltsc 2021' ;;
x86 | x64) echo 'enterprise ltsc 2021' ;;
esac
;;
2024-04-06 21:35:48 +08:00
'iot enterprise ltsc 2019' | 'iot enterprise ltsc 2021') echo "$edition" ;;
2024-04-05 22:45:59 +08:00
esac
;;
11)
case "$edition" in
home | 'home single language') echo consumer ;;
2024-05-22 22:32:43 +08:00
pro | education | enterprise | 'pro education' | 'pro for workstations') echo business ;;
'iot enterprise' | 'iot enterprise subscription') echo 'iot enterprise' ;;
2024-05-22 22:32:43 +08:00
'enterprise ltsc 2024') echo 'enterprise ltsc 2024' ;;
'iot enterprise ltsc 2024' | 'iot enterprise ltsc 2024 subscription') echo 'iot enterprise ltsc 2024' ;;
2024-04-05 22:45:59 +08:00
esac
;;
esac
fi
2024-04-05 23:21:07 +08:00
}
2024-04-05 22:45:59 +08:00
2024-04-05 23:21:07 +08:00
get_label_vlsc() {
2024-04-05 22:45:59 +08:00
case "$version" in
10 | 11)
case "$edition" in
2024-05-22 22:32:43 +08:00
pro | education | enterprise | 'pro education' | 'pro for workstations') echo pro ;;
2024-04-05 22:45:59 +08:00
esac
;;
esac
2024-04-05 23:21:07 +08:00
}
get_page() {
2024-04-05 22:45:59 +08:00
if [ "$arch_win" = arm64 ]; then
echo arm
elif is_ltsc; then
2024-04-05 22:45:59 +08:00
echo ltsc
elif [ "$server" = 'server' ]; then
echo server
else
case "$version" in
vista | 7 | 8.1 | 10 | 11)
echo "$version"
;;
esac
fi
}
is_ltsc() {
grep -Ewq 'ltsb|ltsc' <<<"$edition"
}
# 部分 bash 不支持 $() 里面嵌套case所以定义成函数
label_msdn=$(get_label_msdn)
label_vlsc=$(get_label_vlsc)
page=$(get_page)
2024-04-05 22:45:59 +08:00
page_url=https://massgrave.dev/windows_${page}_links.html
info "Find windows iso"
echo "Version: $version"
echo "Edition: $edition"
echo "Label msdn: $label_msdn"
echo "Label vlsc: $label_vlsc"
echo "List: $page_url"
echo
2024-04-06 21:35:48 +08:00
if [ -z "$page" ] || { [ -z "$label_msdn" ] && [ -z "$label_vlsc" ]; }; then
2024-04-05 22:45:59 +08:00
error_and_exit "Not support find this iso. Check --image-name or set --iso manually."
fi
2024-05-22 22:32:43 +08:00
curl -L "$page_url" | grep -ioP 'https://.*?.iso' >$tmp/win.list
2024-04-06 21:35:48 +08:00
# 如果不是 ltsc ,应该先去除 ltsc 链接,否则最终链接有 ltsc 的
# 例如查找 windows 10 iot enterprise会得到
# en-us_windows_10_iot_enterprise_ltsc_2021_arm64_dvd_e8d4fc46.iso
# en-us_windows_10_iot_enterprise_version_22h2_arm64_dvd_39566b6b.iso
# sed -Ei 和 sed -iE 是不同的
if is_ltsc; then
2024-04-06 21:35:48 +08:00
sed -Ei '/ltsc|ltsb/!d' $tmp/win.list
else
sed -Ei '/ltsc|ltsb/d' $tmp/win.list
fi
2024-04-05 22:45:59 +08:00
}
get_shortest_line() {
# awk '{print length($0), $0}' | sort -n | head -1 | awk '{print $2}'
awk '(NR == 1 || length($0) < length(shortest)) { shortest = $0 } END { print shortest }'
}
get_windows_iso_link() {
regexs=()
# msdn
2024-04-06 21:35:48 +08:00
if [ -n "$label_msdn" ]; then
if [ "$label_msdn" = _ ]; then
label_msdn=
fi
for lang in $langs; do
regex=
for i in ${lang} windows ${server} ${version} ${label_msdn}; do
if [ -n "$i" ]; then
regex+="${i}_"
fi
done
regex+=".*${arch_win}.*.iso"
regexs+=("$regex")
2024-04-05 22:45:59 +08:00
done
2024-04-06 21:35:48 +08:00
fi
2024-04-05 22:45:59 +08:00
# vlsc
2024-04-06 21:35:48 +08:00
if [ -n "$label_vlsc" ]; then
regex="sw_dvd9_win_${label_vlsc}_${version}.*${arch_win}_${full_lang}.*.iso"
regexs+=("$regex")
fi
2024-04-05 22:45:59 +08:00
# 查找
for regex in "${regexs[@]}"; do
regex=${regex// /_}
2024-05-22 22:32:43 +08:00
echo "looking for: $regex" >&2
if iso=$(grep -Ei "/$regex" "$tmp/win.list" | get_shortest_line | grep .); then
2024-04-05 22:45:59 +08:00
return
fi
done
error_and_exit "Could not find windows iso."
}
2023-05-03 22:22:21 +08:00
setos() {
2023-05-13 00:14:46 +08:00
local step=$1
local distro=$2
local releasever=$3
2023-06-18 21:27:22 +08:00
info set $step $distro $releasever
2023-05-13 00:14:46 +08:00
2023-11-09 22:27:07 +08:00
setos_netboot.xyz() {
if is_efi; then
2023-12-02 22:46:16 +08:00
if [ "$basearch" = aarch64 ]; then
2023-11-09 22:27:07 +08:00
eval ${step}_efi=https://boot.netboot.xyz/ipxe/netboot.xyz-arm64.efi
else
eval ${step}_efi=https://boot.netboot.xyz/ipxe/netboot.xyz.efi
fi
else
eval ${step}_vmlinuz=https://boot.netboot.xyz/ipxe/netboot.xyz.lkrn
fi
}
2023-05-13 00:14:46 +08:00
setos_alpine() {
2023-12-16 16:41:34 +08:00
is_virt && flavour=virt || flavour=lts
2024-01-27 23:17:50 +08:00
# alpine aarch64 3.16/3.17 virt 没有直连链接
2023-12-16 16:41:34 +08:00
if [ "$basearch" = aarch64 ] &&
{ [ "$releasever" = 3.16 ] || [ "$releasever" = 3.17 ]; }; then
flavour=lts
2023-06-18 21:27:22 +08:00
fi
2023-10-08 22:10:05 +08:00
# 不要用https 因为甲骨文云arm initramfs阶段不会从硬件同步时钟导致访问https出错
if is_in_china; then
2024-06-06 20:46:07 +08:00
mirror=http://mirrors.ustc.edu.cn/alpine/v$releasever
2023-05-13 00:14:46 +08:00
else
2023-10-08 22:10:05 +08:00
mirror=http://dl-cdn.alpinelinux.org/alpine/v$releasever
2023-05-13 00:14:46 +08:00
fi
2023-10-08 22:10:05 +08:00
eval ${step}_vmlinuz=$mirror/releases/$basearch/netboot/vmlinuz-$flavour
eval ${step}_initrd=$mirror/releases/$basearch/netboot/initramfs-$flavour
eval ${step}_modloop=$mirror/releases/$basearch/netboot/modloop-$flavour
2024-01-27 23:17:50 +08:00
eval ${step}_repo=$mirror/main
2023-05-13 00:14:46 +08:00
}
2023-05-03 22:22:21 +08:00
2023-05-17 22:54:26 +08:00
setos_debian() {
2023-07-18 00:22:51 +08:00
case "$releasever" in
2023-09-10 22:23:03 +08:00
11) codename=bullseye ;;
12) codename=bookworm ;;
2023-07-18 00:22:51 +08:00
esac
2024-05-03 21:37:07 +08:00
if is_in_china; then
cdimage_mirror=https://mirrors.ustc.edu.cn/debian-cdimage
else
2024-05-22 22:32:43 +08:00
cdimage_mirror=https://cdimage.debian.org/images # 在瑞典,不是 cdn
# cloud.debian.org 同样在瑞典,不是 cdn
2024-05-03 21:37:07 +08:00
fi
2023-07-18 00:22:51 +08:00
if is_use_cloud_image; then
# cloud image
is_virt && ci_type=genericcloud || ci_type=generic
2024-05-03 21:37:07 +08:00
eval ${step}_img=$cdimage_mirror/cloud/$codename/latest/debian-$releasever-$ci_type-$basearch_alt.qcow2
2023-05-17 22:54:26 +08:00
else
2023-07-18 00:22:51 +08:00
# 传统安装
2023-10-08 22:10:05 +08:00
if is_in_china; then
2024-05-22 22:32:43 +08:00
# ftp.cn.debian.org 不在国内还严重丢包
# https://www.itdog.cn/ping/ftp.cn.debian.org
2024-06-06 20:46:07 +08:00
deb_hostname=mirrors.ustc.edu.cn
2023-05-17 22:54:26 +08:00
else
2024-05-22 22:32:43 +08:00
deb_hostname=deb.debian.org # fastly
2023-05-17 22:54:26 +08:00
fi
2023-11-01 22:22:30 +08:00
mirror=http://$deb_hostname/debian/dists/$codename/main/installer-$basearch_alt/current/images/netboot/debian-installer/$basearch_alt
2024-05-03 21:37:07 +08:00
is_virt && flavour=-cloud || flavour=
2024-05-29 21:30:23 +08:00
# 甲骨文 arm64 cloud 内核 vnc 没有显示
[ "$basearch_alt" = arm64 ] && flavour=
2024-05-03 21:37:07 +08:00
2023-10-08 22:10:05 +08:00
eval ${step}_vmlinuz=$mirror/linux
eval ${step}_initrd=$mirror/initrd.gz
2023-07-22 21:22:34 +08:00
eval ${step}_ks=$confhome/debian.cfg
2024-05-03 21:37:07 +08:00
eval ${step}_firmware=$cdimage_mirror/firmware/$codename/current/firmware.cpio.gz
eval ${step}_deb_hostname=$deb_hostname
eval ${step}_codename=$codename
eval ${step}_kernel=linux-image$flavour-$basearch_alt
fi
}
setos_kali() {
if is_use_cloud_image; then
:
else
# 传统安装
if is_in_china; then
2024-06-06 20:46:07 +08:00
deb_hostname=mirrors.ustc.edu.cn
2024-05-03 21:37:07 +08:00
else
2024-05-29 21:30:24 +08:00
# http.kali.org 没有 ipv6 地址
2024-05-22 22:32:43 +08:00
# http.kali.org (geoip 重定向) 到 kali.download (cf)
2024-05-29 21:30:24 +08:00
deb_hostname=kali.download
2024-05-03 21:37:07 +08:00
fi
codename=kali-rolling
mirror=http://$deb_hostname/kali/dists/$codename/main/installer-$basearch_alt/current/images/netboot/debian-installer/$basearch_alt
2023-08-27 23:52:06 +08:00
2023-10-22 19:07:12 +08:00
is_virt && flavour=-cloud || flavour=
2024-05-03 21:37:07 +08:00
eval ${step}_vmlinuz=$mirror/linux
eval ${step}_initrd=$mirror/initrd.gz
eval ${step}_ks=$confhome/debian.cfg
eval ${step}_deb_hostname=$deb_hostname
eval ${step}_codename=$codename
eval ${step}_kernel=linux-image$flavour-$basearch_alt
# 缺少 firmware 下载
2023-05-17 22:54:26 +08:00
fi
}
2023-05-03 22:22:21 +08:00
setos_ubuntu() {
2023-11-28 21:47:07 +08:00
case "$releasever" in
20.04) codename=focal ;;
22.04) codename=jammy ;;
2024-04-25 22:24:46 +08:00
24.04) codename=noble ;;
2023-11-28 21:47:07 +08:00
esac
2023-07-18 00:22:51 +08:00
if is_use_cloud_image; then
# cloud image
2023-05-03 22:22:21 +08:00
if is_in_china; then
2024-05-22 22:32:43 +08:00
# 有的源没有 releases 镜像
# https://mirrors.tuna.tsinghua.edu.cn/ubuntu-cloud-images/releases/
# https://unicom.mirrors.ustc.edu.cn/ubuntu-cloud-images/releases/
# https://mirror.nju.edu.cn/ubuntu-cloud-images/releases/
# mirrors.cloud.tencent.com
2023-09-10 22:23:03 +08:00
ci_mirror=https://mirror.nju.edu.cn/ubuntu-cloud-images
2023-05-03 22:22:21 +08:00
else
2023-09-10 22:23:03 +08:00
ci_mirror=https://cloud-images.ubuntu.com
2023-07-18 00:22:51 +08:00
fi
2024-04-27 09:45:10 +08:00
# 20.04/22.04 minimal 镜像没有 aarch64
if { { [ "$releasever" = 20.04 ] || [ "$releasever" = 22.04 ]; } && [ "$basearch_alt" = amd64 ]; } ||
[ "$releasever" = 24.04 ]; then
2023-11-28 21:47:07 +08:00
eval ${step}_img=$ci_mirror/minimal/releases/$codename/release/ubuntu-$releasever-minimal-cloudimg-$basearch_alt.img
2024-04-27 09:45:10 +08:00
else
eval ${step}_img=$ci_mirror/releases/$releasever/release/ubuntu-$releasever-server-cloudimg-$basearch_alt.img
2023-11-28 21:47:07 +08:00
fi
2023-07-18 00:22:51 +08:00
else
# 传统安装
2023-10-08 22:10:05 +08:00
if is_in_china; then
case "$basearch" in
2024-06-06 20:46:07 +08:00
"x86_64") mirror=https://mirrors.ustc.edu.cn/ubuntu-releases/$releasever ;;
"aarch64") mirror=https://mirrors.ustc.edu.cn/ubuntu-cdimage/releases/$releasever/release ;;
2023-10-08 22:10:05 +08:00
esac
2023-07-18 00:22:51 +08:00
else
2023-10-08 22:10:05 +08:00
case "$basearch" in
"x86_64") mirror=https://releases.ubuntu.com/$releasever ;;
"aarch64") mirror=https://cdimage.ubuntu.com/releases/$releasever/release ;;
esac
2023-05-03 22:22:21 +08:00
fi
2023-07-18 00:22:51 +08:00
# iso
filename=$(curl -L $mirror | grep -oP "ubuntu-$releasever.*?-live-server-$basearch_alt.iso" | head -1)
iso=$mirror/$filename
2023-12-02 22:46:16 +08:00
# 在 ubuntu 20.04 上file 命令检测 ubuntu 22.04 iso 结果是 DOS/MBR boot sector
test_url $iso 'iso|dos/mbr'
2023-07-18 00:22:51 +08:00
eval ${step}_iso=$iso
# ks
2023-07-22 21:32:37 +08:00
eval ${step}_ks=$confhome/ubuntu.yaml
2023-07-18 00:22:51 +08:00
fi
2023-05-03 22:22:21 +08:00
}
2023-07-22 22:09:24 +08:00
setos_arch() {
2024-03-31 00:32:05 +08:00
if [ "$basearch" = "x86_64" ]; then
if is_in_china; then
2024-06-06 20:46:07 +08:00
mirror=https://mirrors.ustc.edu.cn/archlinux
2024-03-31 00:32:05 +08:00
else
2024-05-22 22:32:43 +08:00
mirror=https://geo.mirror.pkgbuild.com # geoip
2024-03-31 00:32:05 +08:00
fi
2023-07-22 22:09:24 +08:00
else
2024-03-31 00:32:05 +08:00
if is_in_china; then
2024-06-06 20:46:07 +08:00
mirror=https://mirrors.ustc.edu.cn/archlinuxarm
2024-03-31 00:32:05 +08:00
else
# https 证书有问题
2024-05-22 22:32:43 +08:00
mirror=http://mirror.archlinuxarm.org # geoip
2024-03-31 00:32:05 +08:00
fi
fi
if is_use_cloud_image; then
# cloud image
eval ${step}_img=$mirror/images/latest/Arch-Linux-x86_64-cloudimg.qcow2
else
# 传统安装
case "$basearch" in
x86_64) dir="core/os/$basearch" ;;
aarch64) dir="$basearch/core" ;;
esac
test_url $mirror/$dir/core.db gzip
eval ${step}_mirror=$mirror
2023-07-22 22:09:24 +08:00
fi
}
2023-09-16 20:00:07 +08:00
setos_gentoo() {
if is_in_china; then
2024-06-06 20:46:07 +08:00
mirror=https://mirrors.ustc.edu.cn/gentoo
2023-09-16 20:00:07 +08:00
else
2024-05-22 22:32:43 +08:00
mirror=https://distfiles.gentoo.org # cdn77
2023-09-16 20:00:07 +08:00
fi
2024-03-28 00:16:05 +08:00
if is_use_cloud_image; then
if [ "$basearch_alt" = arm64 ]; then
error_and_exit 'Not support arm64 for gentoo cloud image.'
fi
2023-09-16 20:00:07 +08:00
2024-03-28 00:16:05 +08:00
# openrc 镜像没有附带兼容 cloud-init 的网络管理器
eval ${step}_img=$mirror/experimental/$basearch_alt/openstack/gentoo-openstack-$basearch_alt-systemd-latest.qcow2
else
prefix=stage3-$basearch_alt-systemd
2024-03-28 00:16:05 +08:00
dir=releases/$basearch_alt/autobuilds/current-$prefix
file=$(curl -L $mirror/$dir/latest-$prefix.txt | grep '.tar.xz' | awk '{print $1}')
stage3=$mirror/$dir/$file
test_url $stage3 'xz'
eval ${step}_img=$stage3
fi
2023-09-16 20:00:07 +08:00
}
setos_opensuse() {
2023-09-10 22:23:04 +08:00
# aria2 有 mata4 问题
# https://download.opensuse.org/
2023-12-02 23:10:05 +08:00
# 很多国内源缺少 aarch64 tumbleweed appliances
# https://download.opensuse.org/ports/aarch64/tumbleweed/appliances/
2023-09-10 22:23:04 +08:00
# https://mirrors.nju.edu.cn/opensuse/ports/aarch64/tumbleweed/appliances/
2023-12-02 23:10:05 +08:00
# https://mirrors.ustc.edu.cn/opensuse/ports/aarch64/tumbleweed/appliances/
# https://mirrors.tuna.tsinghua.edu.cn/opensuse/ports/aarch64/tumbleweed/appliances/
2023-09-10 22:23:04 +08:00
if is_in_china; then
2023-12-02 23:10:05 +08:00
mirror=https://mirror.sjtu.edu.cn/opensuse
2023-09-10 22:23:04 +08:00
else
mirror=https://mirror.fcix.net/opensuse
fi
2023-12-02 22:46:16 +08:00
if [ "$releasever" = tumbleweed ]; then
# tumbleweed
2023-09-10 22:23:04 +08:00
if [ "$basearch" = aarch64 ]; then
dir=ports/aarch64/tumbleweed/appliances
else
dir=tumbleweed/appliances
fi
2023-12-02 22:46:16 +08:00
file=openSUSE-Tumbleweed-Minimal-VM.$basearch-Cloud.qcow2
else
2023-09-10 22:23:04 +08:00
# 常规版本
dir=distribution/leap/$releasever/appliances
2023-12-02 22:46:16 +08:00
file=openSUSE-Leap-$releasever-Minimal-VM.$basearch-Cloud.qcow2
fi
# 有专门的kvm镜像openSUSE-Leap-15.5-Minimal-VM.x86_64-kvm-and-xen.qcow2但里面没有cloud-init
2023-12-02 22:46:16 +08:00
eval ${step}_img=$mirror/$dir/$file
}
2023-05-25 20:15:12 +08:00
setos_windows() {
2024-04-05 22:45:59 +08:00
if [ -z "$iso" ]; then
# 查找时将 windows longhorn serverdatacenter 改成 windows server 2008 serverdatacenter
image_name=${image_name/windows longhorn server/windows server 2008 server}
2024-04-05 22:45:59 +08:00
echo "iso url is not set. Try to find it."
find_windows_iso
fi
# 将上面的 windows server 2008 serverdatacenter 改回 windows longhorn serverdatacenter
# 也能纠正用户输入了 windows server 2008 serverdatacenter
# 注意 windows server 2008 r2 serverdatacenter 不用改
image_name=${image_name/windows server 2008 server/windows longhorn server}
2023-12-02 22:46:16 +08:00
test_url $iso 'iso|dos/mbr'
2023-05-25 20:15:12 +08:00
eval "${step}_iso='$iso'"
eval "${step}_image_name='$image_name'"
}
# shellcheck disable=SC2154
2023-06-04 19:12:57 +08:00
setos_dd() {
2023-12-02 22:46:16 +08:00
test_url $img 'xz|gzip' ${step}_img_type
2023-06-05 19:45:07 +08:00
eval "${step}_img='$img'"
2023-06-04 19:12:57 +08:00
}
2024-07-07 17:07:21 +08:00
setos_centos_alma_rocky_fedora() {
2023-07-18 00:22:51 +08:00
if is_use_cloud_image; then
# ci
if is_in_china; then
case $distro in
"centos") ci_mirror="https://mirror.nju.edu.cn/centos-cloud/centos" ;;
"alma") ci_mirror="https://mirror.nju.edu.cn/almalinux/$releasever/cloud/$basearch/images" ;;
"rocky") ci_mirror="https://mirror.nju.edu.cn/rocky/$releasever/images/$basearch" ;;
"fedora") ci_mirror="https://mirror.nju.edu.cn/fedora/releases/$releasever/Cloud/$basearch/images" ;;
esac
else
case $distro in
"centos") ci_mirror="https://cloud.centos.org/centos" ;;
"alma") ci_mirror="https://repo.almalinux.org/almalinux/$releasever/cloud/$basearch/images" ;;
"rocky") ci_mirror="https://download.rockylinux.org/pub/rocky/$releasever/images/$basearch" ;;
"fedora") ci_mirror="https://download.fedoraproject.org/pub/fedora/linux/releases/$releasever/Cloud/$basearch/images" ;;
esac
fi
2023-05-13 00:14:46 +08:00
case $distro in
"centos") ci_image=$ci_mirror/$releasever-stream/$basearch/images/CentOS-Stream-GenericCloud-$releasever-latest.$basearch.qcow2 ;;
2024-05-22 22:32:43 +08:00
"alma") ci_image=$ci_mirror/AlmaLinux-$releasever-GenericCloud-latest.$basearch.qcow2 ;;
2023-07-18 00:22:51 +08:00
"rocky") ci_image=$ci_mirror/Rocky-$releasever-GenericCloud-Base.latest.$basearch.qcow2 ;;
2023-07-28 21:27:16 +08:00
"fedora")
2024-04-23 23:34:53 +08:00
# Fedora-Cloud-Base-39-1.5.x86_64.qcow2
# Fedora-Cloud-Base-Generic.x86_64-40-1.14.qcow2
page=$(curl -L $ci_mirror)
# 40
filename=$(grep -oP "Fedora-Cloud-Base-Generic.*?.qcow2" <<<"$page" | head -1)
# 38/39
if [ -z "$filename" ]; then
filename=$(grep -oP "Fedora-Cloud-Base-$releasever.*?.qcow2" <<<"$page" | head -1)
fi
ci_image=$ci_mirror/$filename
2023-07-28 21:27:16 +08:00
;;
2023-05-13 00:14:46 +08:00
esac
2023-07-18 00:22:51 +08:00
eval ${step}_img=${ci_image}
else
# 传统安装
2023-10-08 22:10:05 +08:00
case $distro in
"centos") mirrorlist="https://mirrors.centos.org/mirrorlist?repo=centos-baseos-$releasever-stream&arch=$basearch" ;;
2023-10-08 22:10:05 +08:00
"alma") mirrorlist="https://mirrors.almalinux.org/mirrorlist/$releasever/baseos" ;;
"rocky") mirrorlist="https://mirrors.rockylinux.org/mirrorlist?arch=$basearch&repo=BaseOS-$releasever" ;;
"fedora") mirrorlist="https://mirrors.fedoraproject.org/mirrorlist?arch=$basearch&repo=fedora-$releasever" ;;
esac
2023-07-18 00:22:51 +08:00
2023-10-08 22:10:05 +08:00
# rocky/centos9 需要删除第一行注释, alma 需要替换$basearch
for cur_mirror in $(curl -L $mirrorlist | sed "/^#/d" | sed "s,\$basearch,$basearch,"); do
host=$(get_host_by_url $cur_mirror)
if is_host_has_ipv4_and_ipv6 $host &&
test_url_grace ${cur_mirror}images/pxeboot/vmlinuz; then
mirror=$cur_mirror
break
fi
2023-10-08 22:10:05 +08:00
done
2023-07-18 00:22:51 +08:00
2023-10-08 22:10:05 +08:00
if [ -z "$mirror" ]; then
error_and_exit "All mirror failed."
fi
2023-05-03 22:22:21 +08:00
2023-10-08 22:10:05 +08:00
eval "${step}_mirrorlist='${mirrorlist}'"
2023-07-22 21:24:43 +08:00
eval ${step}_ks=$confhome/redhat.cfg
2023-07-18 00:22:51 +08:00
eval ${step}_vmlinuz=${mirror}images/pxeboot/vmlinuz
eval ${step}_initrd=${mirror}images/pxeboot/initrd.img
eval ${step}_squashfs=${mirror}images/install.img
test_url ${mirror}images/install.img 'squashfs'
2023-05-03 22:22:21 +08:00
fi
}
2024-05-03 23:34:42 +08:00
setos_oracle() {
if is_use_cloud_image; then
# ci
install_pkg jq
mirror=https://yum.oracle.com
[ "$basearch" = aarch64 ] &&
template_prefix=ol${releasever}_${basearch}-cloud ||
template_prefix=ol${releasever}
curl -Lo $tmp/oracle.json $mirror/templates/OracleLinux/$template_prefix-template.json
dir=$(jq -r .base_url $tmp/oracle.json)
file=$(jq -r .kvm.image $tmp/oracle.json)
ci_image=$mirror$dir/$file
eval ${step}_img=${ci_image}
else
:
fi
}
2024-07-07 17:07:21 +08:00
setos_redhat() {
if is_use_cloud_image; then
# ci
eval "${step}_img='$img'"
else
:
fi
}
setos_opencloudos() {
# https://mirrors.opencloudos.tech 不支持 ipv6
mirror=https://mirrors.cloud.tencent.com/opencloudos
if is_use_cloud_image; then
# ci
dir=$releasever/images/$basearch
file=$(curl -L $mirror/$dir/ | grep -oP 'OpenCloudOS.*?\.qcow2' | head -1)
eval ${step}_img=$mirror/$dir/$file
else
:
fi
}
# anolis 23 不是 lts而且 cloud-init 好像有问题
setos_anolis() {
mirror=https://mirrors.openanolis.cn/anolis
if is_use_cloud_image; then
# ci
dir=$releasever/isos/GA/$basearch
file=$(curl -L $mirror/$dir/ | grep -oP 'AnolisOS.*?\.qcow2' | head -1)
eval ${step}_img=$mirror/$dir/$file
else
:
fi
}
setos_openeuler() {
if is_in_china; then
mirror=https://repo.openeuler.openatom.cn
else
mirror=https://repo.openeuler.org
fi
if is_use_cloud_image; then
# ci
name=$(curl -L "$mirror/" | grep -oE "openEuler-$releasever-LTS(-SP[0-9])?" | sort -u | tail -1)
eval ${step}_img=$mirror/$name/virtual_machine_img/$basearch/$name-$basearch.qcow2.xz
else
:
fi
}
2023-05-17 22:54:26 +08:00
eval ${step}_distro=$distro
2024-04-27 09:45:10 +08:00
eval ${step}_releasever=$releasever
2024-07-07 17:07:21 +08:00
case "$distro" in
centos | alma | rocky | fedora) setos_centos_alma_rocky_fedora ;;
*) setos_$distro ;;
esac
2023-12-02 22:46:16 +08:00
# debian/kali <=256M 必须使用云内核,否则不够内存
if is_distro_like_debian && ! is_in_windows && [ "$ram_size" -le 256 ]; then
exit_if_cant_use_cloud_kernel
fi
2023-12-02 22:46:16 +08:00
# 集中测试云镜像格式
if is_use_cloud_image && [ "$step" = finalos ]; then
# shellcheck disable=SC2154
test_url $finalos_img 'xz|gzip|qemu' finalos_img_type
# openeuler 云镜像格式是 .qcow2.xz
if [ "$distro" = openeuler ]; then
# shellcheck disable=SC2034
finalos_img_type=qemu
fi
2023-12-02 22:46:16 +08:00
fi
2023-05-03 22:22:21 +08:00
}
is_distro_like_redhat() {
2024-05-08 22:17:33 +08:00
if [ -n "$1" ]; then
_distro=$1
else
_distro=$distro
fi
2024-07-07 17:07:21 +08:00
[ "$_distro" = redhat ] || [ "$_distro" = centos ] || [ "$_distro" = alma ] || [ "$_distro" = rocky ] || [ "$_distro" = fedora ] || [ "$_distro" = oracle ]
2024-05-03 21:37:07 +08:00
}
is_distro_like_debian() {
2024-05-08 22:17:33 +08:00
if [ -n "$1" ]; then
_distro=$1
else
_distro=$distro
fi
[ "$_distro" = debian ] || [ "$_distro" = kali ]
}
2023-05-13 00:14:46 +08:00
# 检查是否为正确的系统名
2023-10-22 16:43:34 +08:00
verify_os_name() {
if [ -z "$*" ]; then
usage_and_exit
fi
2023-08-01 22:33:07 +08:00
for os in \
'centos |9' \
'anolis 7|8' \
'oracle 8|9' \
'alma 8|9' \
'rocky 8|9' \
'redhat 8|9' \
'opencloudos 8|9' \
'fedora 39|40' \
'debian 11|12' \
'openeuler 20.03|22.03|24.03' \
'ubuntu 20.04|22.04|24.04' \
'alpine 3.17|3.18|3.19|3.20' \
'opensuse 15.5|15.6|tumbleweed' \
2024-05-03 21:37:07 +08:00
'kali' \
2023-09-10 22:23:04 +08:00
'arch' \
2023-09-16 20:00:07 +08:00
'gentoo' \
2023-09-10 22:23:04 +08:00
'windows' \
2023-11-09 22:27:07 +08:00
'dd' \
'netboot.xyz'; do
2023-09-10 22:23:04 +08:00
ds=$(awk '{print $1}' <<<"$os")
vers=$(awk '{print $2}' <<<"$os" | sed 's \. \\\. g')
2023-10-22 19:07:12 +08:00
finalos=$(echo "$@" | to_lower | sed -n -E "s,^($ds)[ :-]?(|$vers)$,\1:\2,p")
2023-05-13 00:14:46 +08:00
if [ -n "$finalos" ]; then
distro=$(echo $finalos | cut -d: -f1)
releasever=$(echo $finalos | cut -d: -f2)
# 默认版本号
if [ -z "$releasever" ] && grep -q '|' <<<$os; then
releasever=$(awk '{print $2}' <<<$os | awk -F'|' '{print $NF}')
2023-09-10 22:23:04 +08:00
fi
2023-05-13 00:14:46 +08:00
return
fi
done
2023-06-18 21:27:22 +08:00
error "Please specify a proper os"
2023-05-13 00:14:46 +08:00
usage_and_exit
}
2023-10-22 16:43:34 +08:00
verify_os_args() {
case "$distro" in
dd) [ -n "$img" ] || error_and_exit "dd need --img" ;;
2024-07-07 17:07:21 +08:00
redhat) [ -n "$img" ] || error_and_exit "redhat need --img" ;;
windows) [ -n "$image_name" ] || error_and_exit "Install Windows need --image-name." ;;
2023-10-22 16:43:34 +08:00
esac
}
get_cmd_path() {
# arch 云镜像不带 which
# command -v 包括脚本里面的方法
2023-10-22 19:07:12 +08:00
# ash 无效
type -f -p $1
}
is_have_cmd() {
get_cmd_path $1 >/dev/null 2>&1
}
2023-09-16 20:00:06 +08:00
install_pkg() {
is_in_windows && return
2023-07-15 23:25:42 +08:00
2023-09-16 20:00:06 +08:00
find_pkg_mgr() {
if [ -z "$pkg_mgr" ]; then
2023-12-04 22:19:59 +08:00
for mgr in dnf yum apt pacman zypper emerge apk; do
is_have_cmd $mgr && pkg_mgr=$mgr && return
done
return 1
2023-07-15 23:25:42 +08:00
fi
2023-09-16 20:00:06 +08:00
}
cmd_to_pkg() {
unset USE
case $cmd in
2024-04-03 21:53:52 +08:00
ar)
case "$pkg_mgr" in
*) pkg="binutils" ;;
esac
;;
xz)
case "$pkg_mgr" in
apt) pkg="xz-utils" ;;
*) pkg="xz" ;;
esac
;;
2024-02-01 00:59:34 +08:00
lsblk | findmnt)
case "$pkg_mgr" in
apk) pkg="$cmd" ;;
*) pkg="util-linux" ;;
esac
;;
lsmem)
case "$pkg_mgr" in
apk) pkg="util-linux-misc" ;;
*) pkg="util-linux" ;;
esac
;;
fdisk)
case "$pkg_mgr" in
apt) pkg="fdisk" ;;
apk) pkg="util-linux-misc" ;;
*) pkg="util-linux" ;;
esac
;;
2023-09-16 20:00:06 +08:00
unsquashfs)
case "$pkg_mgr" in
zypper) pkg="squashfs" ;;
emerge) pkg="squashfs-tools" && export USE="lzma" ;;
*) pkg="squashfs-tools" ;;
2023-07-15 23:25:42 +08:00
esac
2023-09-16 20:00:06 +08:00
;;
nslookup | dig)
case "$pkg_mgr" in
apt) pkg="dnsutils" ;;
pacman) pkg="bind" ;;
apk | emerge) pkg="bind-tools" ;;
yum | dnf | zypper) pkg="bind-utils" ;;
esac
;;
*) pkg=$cmd ;;
esac
}
2023-08-01 22:10:34 +08:00
2024-07-09 23:37:51 +08:00
# 系统 package名称 repo名称
# centos/alma/rocky/fedora/anolis epel-release epel
# oracle linux oracle-epel-release ol9_developer_EPEL
# opencloudos epol-release EPOL
check_is_need_epel() {
is_need_epel() {
case "$pkg" in
dpkg) true ;;
jq) is_have_cmd yum && ! is_have_cmd dnf ;; # el7/ol7 的 jq 在 epel 仓库
*) false ;;
esac
}
2024-04-03 21:53:52 +08:00
2024-07-09 23:37:51 +08:00
try_find_epel_name() {
epel=$($pkg_mgr repolist --all | awk '{print $1}' | grep -Ei '(epel|epol)$')
}
if is_need_epel; then
if ! try_find_epel_name; then
epel_release="$($pkg_mgr list | grep -E '(epel|epol)-release' | awk '{print $1}' | cut -d. -f1 | head -1)"
2024-04-03 21:53:52 +08:00
$pkg_mgr install -y $epel_release
2024-07-09 23:37:51 +08:00
try_find_epel_name
2024-04-03 21:53:52 +08:00
fi
2024-07-09 23:37:51 +08:00
enable_epel="--enablerepo=$epel"
2024-04-03 21:53:52 +08:00
fi
}
2023-09-16 20:00:06 +08:00
install_pkg_real() {
2024-03-05 23:01:24 +08:00
text="$pkg"
if [ "$pkg" != "$cmd" ]; then
text+=" ($cmd)"
fi
echo "Installing package '$text'..."
2024-04-03 21:53:52 +08:00
2023-08-01 22:10:34 +08:00
case $pkg_mgr in
2024-07-09 23:37:51 +08:00
dnf)
check_is_need_epel
dnf install $enable_epel -y --setopt=install_weak_deps=False $pkg
;;
yum)
check_is_need_epel
yum install $enable_epel -y $pkg
;;
2023-09-16 20:00:06 +08:00
emerge) emerge --oneshot $pkg ;;
pacman) pacman -Syu --noconfirm --needed $pkg ;;
zypper) zypper install -y $pkg ;;
apk)
add_community_repo_for_alpine
apk add $pkg
;;
apt)
2024-06-14 23:41:15 +08:00
[ -z "$apt_updated" ] && apt update && apt_updated=1
2023-10-27 11:48:18 +08:00
DEBIAN_FRONTEND=noninteractive apt install -y $pkg
2023-09-16 20:00:06 +08:00
;;
2023-08-01 22:10:34 +08:00
esac
2023-09-16 20:00:06 +08:00
}
2024-02-01 00:59:34 +08:00
is_need_reinstall() {
2024-01-27 23:17:50 +08:00
cmd=$1
2024-02-01 00:59:34 +08:00
2024-01-27 23:17:50 +08:00
# gentoo 默认编译的 unsquashfs 不支持 xz
2024-02-01 00:59:34 +08:00
if [ "$cmd" = unsquashfs ] && is_have_cmd emerge && ! $cmd |& grep -wq xz; then
2024-01-27 23:17:50 +08:00
echo "unsquashfs not supported xz. rebuilding."
return 0
fi
2024-02-01 00:59:34 +08:00
# busybox fdisk 无法显示 mbr 分区表的 id
if [ "$cmd" = fdisk ] && is_have_cmd apk && $cmd |& grep -wq BusyBox; then
return 0
fi
# busybox grep 无法 grep -oP
if [ "$cmd" = grep ] && is_have_cmd apk && $cmd |& grep -wq BusyBox; then
return 0
fi
2024-01-27 23:17:50 +08:00
return 1
}
for cmd in "$@"; do
2024-02-01 00:59:34 +08:00
if ! is_have_cmd $cmd || is_need_reinstall $cmd; then
if ! find_pkg_mgr; then
error_and_exit "Can't find compatible package manager. Please manually install $cmd."
fi
cmd_to_pkg
install_pkg_real
fi
done
2023-05-13 00:14:46 +08:00
}
2023-05-17 22:54:26 +08:00
check_ram() {
2024-03-28 00:16:05 +08:00
ram_standard=$(
case "$distro" in
netboot.xyz) echo 0 ;;
2024-05-03 21:37:07 +08:00
alpine | debian | kali | dd) echo 256 ;;
2024-03-31 00:32:05 +08:00
arch | gentoo | windows) echo 512 ;;
redhat | centos | alma | rocky | fedora | oracle | ubuntu | anolis | opencloudos | openeuler) echo 1024 ;;
2024-04-05 22:45:59 +08:00
opensuse) echo -1 ;; # 没有安装模式
2024-03-28 00:16:05 +08:00
esac
)
# 不用检查内存的情况
if [ "$ram_standard" -eq 0 ]; then
return
fi
2024-04-25 22:43:11 +08:00
# 未测试
ram_cloud_image=256
2024-03-28 00:16:05 +08:00
has_cloud_image=$(
case "$distro" in
redhat | centos | alma | rocky | oracle | fedora | debian | ubuntu | opensuse | anolis | openeuler) echo true ;;
2024-05-03 21:37:07 +08:00
netboot.xyz | alpine | dd | arch | gentoo | kali | windows) echo false ;;
2024-03-28 00:16:05 +08:00
esac
)
2023-06-18 21:27:22 +08:00
if is_in_windows; then
ram_size=$(wmic memorychip get capacity | tail +2 | awk '{sum+=$1} END {print sum/1024/1024}')
else
2024-04-03 21:53:52 +08:00
# lsmem最准确但 centos7 arm 和 alpine 不能用debian 9 util-linux 没有 lsmem
2023-06-18 21:27:22 +08:00
# arm 24g dmidecode 显示少了128m
2023-07-05 21:57:27 +08:00
# arm 24g lshw 显示23BiB
# ec2 t4g arm alpine 用 lsmem 和 dmidecode 都无效,要用 lshw但结果和free -m一致其他平台则没问题
2023-07-15 23:25:42 +08:00
install_pkg lsmem
2023-06-18 21:27:22 +08:00
ram_size=$(lsmem -b 2>/dev/null | grep 'Total online memory:' | awk '{ print $NF/1024/1024 }')
2023-07-05 21:57:27 +08:00
2023-06-18 21:27:22 +08:00
if [ -z $ram_size ]; then
install_pkg dmidecode
ram_size=$(dmidecode -t 17 | grep "Size.*[GM]B" | awk '{if ($3=="GB") s+=$2*1024; else s+=$2} END {print s}')
fi
2023-07-05 21:57:27 +08:00
if [ -z $ram_size ]; then
install_pkg lshw
# 不能忽略 -ialpine 显示的是 System memory
ram_str=$(lshw -c memory -short | grep -i 'System Memory' | awk '{print $3}')
ram_size=$(grep <<<$ram_str -o '[0-9]*')
grep <<<$ram_str GiB && ram_size=$((ram_size * 1024))
fi
2023-05-17 22:54:26 +08:00
fi
2023-06-18 21:27:22 +08:00
if [ -z $ram_size ] || [ $ram_size -le 0 ]; then
error_and_exit "Could not detect RAM size."
fi
2023-07-18 00:22:51 +08:00
# ram 足够就用普通方法安装否则如果内存大于512就用 cloud image
# TODO: 测试 256 384 内存
2024-03-28 00:16:05 +08:00
if ! is_use_cloud_image && [ $ram_size -lt $ram_standard ]; then
if $has_cloud_image; then
info "RAM < $ram_standard MB. Fallback to cloud image mode"
2023-07-18 00:22:51 +08:00
cloud_image=1
else
2024-03-28 00:16:05 +08:00
error_and_exit "Could not install $distro: RAM < $ram_standard MB."
2023-07-18 00:22:51 +08:00
fi
2023-06-18 21:27:22 +08:00
fi
2024-03-28 00:16:05 +08:00
if is_use_cloud_image && [ $ram_size -lt $ram_cloud_image ]; then
error_and_exit "Could not install $distro using cloud image: RAM < $ram_cloud_image MB."
fi
2023-06-18 21:27:22 +08:00
}
is_efi() {
if is_in_windows; then
2024-01-27 23:17:50 +08:00
# bcdedit | grep -qi '^path.*\.efi'
mountvol | grep -q --text 'EFI'
2023-06-18 21:27:22 +08:00
else
[ -d /sys/firmware/efi ]
fi
}
is_grub_dir_linked() {
# cloudcone 重装前/重装后(方法1)
[ "$(readlink -f /boot/grub/grub.cfg)" = /boot/grub2/grub.cfg ] ||
[ "$(readlink -f /boot/grub2/grub.cfg)" = /boot/grub/grub.cfg ] ||
# cloudcone 重装后(方法2)
{ [ -f /boot/grub2/grub.cfg ] && [ "$(cat /boot/grub2/grub.cfg)" = 'chainloader (hd0)+1' ]; }
}
2023-12-22 23:45:06 +08:00
is_secure_boot_enabled() {
if is_efi; then
if is_in_windows; then
reg query 'HKLM\SYSTEM\CurrentControlSet\Control\SecureBoot\State' /v UEFISecureBootEnabled 2>/dev/null | grep 0x1
2023-12-22 23:45:06 +08:00
else
2024-02-02 00:49:24 +08:00
# localhost:~# mokutil --sb-state
# SecureBoot disabled
# Platform is in Setup Mode
dmesg | grep -i 'Secure boot enabled'
2023-12-22 23:45:06 +08:00
fi
2024-02-02 00:49:24 +08:00
else
return 1
2023-12-22 23:45:06 +08:00
fi
}
2023-11-09 22:27:07 +08:00
is_use_grub() {
! { is_netboot_xyz && is_efi; }
}
# 只有 linux bios 是用本机的 grub
is_use_local_grub() {
is_use_grub && ! is_in_windows && ! is_efi
}
to_upper() {
tr '[:lower:]' '[:upper:]'
}
to_lower() {
tr '[:upper:]' '[:lower:]'
}
del_cr() {
sed 's/\r//g'
}
2024-03-05 23:01:24 +08:00
del_empty_lines() {
sed '/^[[:space:]]*$/d'
}
2024-02-01 00:59:34 +08:00
# 记录主硬盘
find_main_disk() {
2024-04-03 21:53:52 +08:00
if [ -n "$main_disk" ]; then
return
fi
2024-02-01 00:59:34 +08:00
if is_in_windows; then
# TODO:
# 已测试 vista
# 测试 软raid
# 测试 动态磁盘
# diskpart 命令结果
# 磁盘 ID: E5FDE61C
# 磁盘 ID: {92CF6564-9B2E-4348-A3BD-D84E3507EBD7}
main_disk=$(printf "%s\n%s" "select volume $c" "uniqueid disk" | diskpart |
2024-02-01 00:59:34 +08:00
tail -1 | awk '{print $NF}' | sed 's,[{}],,g' | del_cr)
else
# centos7下测试 lsblk --inverse $mapper | grep -w disk grub2-probe -t disk /
# 跨硬盘btrfs 只显示第一个硬盘 显示两个硬盘
# 跨硬盘lvm 显示两个硬盘 显示/dev/mapper/centos-root
# 跨硬盘软raid 显示两个硬盘 显示/dev/md127
# 还有 findmnt
2024-02-01 00:59:34 +08:00
# 改成先检测 /boot/efi /efi /boot 分区?
install_pkg lsblk
# 查找主硬盘时,优先查找 /boot 分区,再查找 / 分区
2024-02-01 00:59:34 +08:00
# lvm 显示的是 /dev/mapper/xxx-yyy再用第二条命令得到sda
mapper=$(mount | awk '$3=="/boot" {print $1}' | grep . || mount | awk '$3=="/" {print $1}')
2024-02-01 00:59:34 +08:00
xda=$(lsblk -rn --inverse $mapper | grep -w disk | awk '{print $1}' | sort -u)
# 检测主硬盘是否横跨多个磁盘
os_across_disks_count=$(wc -l <<<"$xda")
if [ $os_across_disks_count -eq 1 ]; then
info "Main disk: $xda"
else
error_and_exit "OS across $os_across_disks_count disk: $xda"
fi
# 可以用 dd 找出 guid?
2024-03-01 22:01:22 +08:00
# centos7 blkid lsblk 不显示 PTUUID
2024-02-01 00:59:34 +08:00
# centos7 sfdisk 不显示 Disk identifier
2024-03-01 22:01:22 +08:00
# alpine blkid 不显示 gpt 分区表的 PTUUID
2024-02-01 00:59:34 +08:00
# 因此用 fdisk
# Disk identifier: 0x36778223 # gnu fdisk + mbr
# Disk identifier: D6B17C1A-FA1E-40A1-BDCB-0278A3ED9CFC # gnu fdisk + gpt
# Disk identifier (GUID): d6b17c1a-fa1e-40a1-bdcb-0278a3ed9cfc # busybox fdisk + gpt
# 不显示 Disk identifier # busybox fdisk + mbr
# 获取 xda 的 id
install_pkg fdisk
main_disk=$(fdisk -l /dev/$xda | grep 'Disk identifier' | awk '{print $NF}' | sed 's/0x//')
fi
# 检查 id 格式是否正确
if ! grep -Eix '[0-9a-f]{8}' <<<"$main_disk" &&
! grep -Eix '[0-9a-f-]{36}' <<<"$main_disk"; then
error_and_exit "Disk ID is invalid: $main_disk"
fi
}
is_found_ipv4_netconf() {
[ -n "$ipv4_mac" ] && [ -n "$ipv4_addr" ] && [ -n "$ipv4_gateway" ]
}
is_found_ipv6_netconf() {
[ -n "$ipv6_mac" ] && [ -n "$ipv6_addr" ] && [ -n "$ipv6_gateway" ]
}
# TODO: 单网卡多IP
collect_netconf() {
if is_in_windows; then
convert_net_str_to_array() {
config=$1
key=$2
var=$3
2023-11-09 22:30:21 +08:00
IFS=',' read -r -a "${var?}" <<<"$(grep "$key=" <<<"$config" | cut -d= -f2 | sed 's/[{}\"]//g')"
2023-09-21 00:12:49 +08:00
}
# 部分机器精简了 powershell
# 所以不要用 powershell 获取网络信息
2023-12-23 20:32:32 +08:00
# ids=$(wmic nic where "PhysicalAdapter=true and MACAddress is not null and (PNPDeviceID like '%VEN_%&DEV_%' or PNPDeviceID like '%{F8615163-DF3E-46C5-913F-F2D2F965ED0E}%')" get InterfaceIndex | del_cr | sed '1d')
# 否 手动 0 0.0.0.0/0 19 192.168.1.1
# 否 手动 0 0.0.0.0/0 59 nekoray-tun
# wmic nic:
# 真实网卡
# AdapterType=以太网 802.3
# AdapterTypeId=0
# MACAddress=68:EC:C5:11:11:11
# PhysicalAdapter=TRUE
# PNPDeviceID=PCI\VEN_8086&amp;DEV_095A&amp;SUBSYS_94108086&amp;REV_61\4&amp;295A4BD&amp;1&amp;00E0
# VPN tun 网卡,部分移动云电脑也有
# AdapterType=
# AdapterTypeId=
# MACAddress=
# PhysicalAdapter=TRUE
# PNPDeviceID=SWD\WINTUN\{6A460D48-FB76-6C3F-A47D-EF97D3DC6B0E}
# VMware 网卡
# AdapterType=以太网 802.3
# AdapterTypeId=0
# MACAddress=00:50:56:C0:00:08
# PhysicalAdapter=TRUE
# PNPDeviceID=ROOT\VMWARE\0001
for v in 4 6; do
if [ "$v" = 4 ]; then
# 或者 route print
routes=$(netsh int ipv4 show route | awk '$4 == "0.0.0.0/0"' | del_cr)
else
routes=$(netsh int ipv6 show route | awk '$4 == "::/0"' | del_cr)
fi
if [ -z "$routes" ]; then
continue
fi
while read -r route; do
if false; then
read -r _ _ _ _ id gateway <<<"$route"
else
id=$(awk '{print $5}' <<<"$route")
gateway=$(awk '{print $6}' <<<"$route")
fi
config=$(wmic nicconfig where InterfaceIndex=$id get MACAddress,IPAddress,IPSubnet,DefaultIPGateway /format:list | del_cr)
# 排除 IP/子网/网关/MAC 为空的
if grep -q '=$' <<<"$config"; then
continue
fi
mac_addr=$(grep "MACAddress=" <<<"$config" | cut -d= -f2 | to_lower)
convert_net_str_to_array "$config" IPAddress ips
convert_net_str_to_array "$config" IPSubnet subnets
convert_net_str_to_array "$config" DefaultIPGateway gateways
# IPv4
# shellcheck disable=SC2154
if [ "$v" = 4 ]; then
for ((i = 0; i < ${#ips[@]}; i++)); do
ip=${ips[i]}
subnet=${subnets[i]}
if [[ "$ip" = *.* ]]; then
cidr=$(ipcalc -b "$ip/$subnet" | grep Netmask: | awk '{print $NF}')
ipv4_addr="$ip/$cidr"
ipv4_gateway="$gateway"
ipv4_mac="$mac_addr"
# 只取第一个 IP
break
fi
done
fi
# IPv6
if [ "$v" = 6 ]; then
ipv6_type_list=$(netsh interface ipv6 show address $id normal)
for ((i = 0; i < ${#ips[@]}; i++)); do
ip=${ips[i]}
cidr=${subnets[i]}
if [[ "$ip" = *:* ]]; then
ipv6_type=$(grep "$ip" <<<"$ipv6_type_list" | awk '{print $1}')
# Public 是 slaac
# 还有类型 Temporary不过有 Temporary 肯定还有 Public因此不用
if [ "$ipv6_type" = Public ] ||
[ "$ipv6_type" = Dhcp ] ||
[ "$ipv6_type" = Manual ]; then
ipv6_addr="$ip/$cidr"
ipv6_gateway="$gateway"
ipv6_mac="$mac_addr"
# 只取第一个 IP
break
fi
fi
done
fi
# 网关
# shellcheck disable=SC2154
if false; then
for gateway in "${gateways[@]}"; do
if [ -n "$ipv4_addr" ] && [[ "$gateway" = *.* ]]; then
ipv4_gateway="$gateway"
elif [ -n "$ipv6_addr" ] && [[ "$gateway" = *:* ]]; then
ipv6_gateway="$gateway"
fi
done
fi
# 如果通过本条 route 的网卡找到了 IP 则退出 routes 循环
if is_found_ipv${v}_netconf; then
break
fi
done < <(echo "$routes")
done
else
# linux
# 通过默认网关得到默认网卡
# 多个默认路由下
# ip -6 route show default dev ens3 完全不显示
# ip -6 route show default
# default proto static metric 1024 pref medium
# nexthop via 2a01:1111:262:4940::2 dev ens3 weight 1 onlink
# nexthop via fe80::5054:ff:fed4:5286 dev ens3 weight 1
# ip -6 route show default
# default via 2602:1111:0:80::1 dev eth0 metric 1024 onlink pref medium
for v in 4 6; do
if ethx=$(ip -$v route show default | awk '$4=="dev"' | head -1 | awk '{print $5}' | grep .); then
if ip -$v route show default | awk '$5=="'$ethx'"' | head -1 | grep -q .; then
eval ipv${v}_ethx="$ethx" # can_use_cloud_kernel 要用
eval ipv${v}_mac="$(ip link show dev $ethx | grep link/ether | head -1 | awk '{print $2}')"
eval ipv${v}_gateway="$(ip -$v route show default | awk '$5=="'$ethx'"' | head -1 | awk '{print $3}')"
eval ipv${v}_addr="$(ip -$v -o addr show scope global dev $ethx | head -1 | awk '{print $4}')"
fi
fi
done
fi
2023-09-21 00:12:49 +08:00
if ! is_found_ipv4_netconf && ! is_found_ipv6_netconf; then
error_and_exit "Can not get IP info."
fi
info "Network Info"
echo "IPv4 MAC: $ipv4_mac"
echo "IPv4 Address: $ipv4_addr"
echo "IPv4 Gateway: $ipv4_gateway"
echo "---"
echo "IPv6 MAC: $ipv6_mac"
echo "IPv6 Address: $ipv6_addr"
echo "IPv6 Gateway: $ipv6_gateway"
2024-04-03 21:53:52 +08:00
echo
}
2023-11-09 22:27:07 +08:00
add_efi_entry_in_windows() {
source=$1
# 挂载
if result=$(find /cygdrive/?/EFI/Microsoft/Boot/bootmgfw.efi 2>/dev/null); then
# 已经挂载
x=$(echo $result | cut -d/ -f3)
else
# 找到空盘符并挂载
for x in {a..z}; do
[ ! -e /cygdrive/$x ] && break
done
mountvol $x: /s
fi
# 文件夹命名为reinstall而不是grub因为可能机器已经安装了grubbcdedit名字同理
dist_dir=/cygdrive/$x/EFI/reinstall
basename=$(basename $source)
mkdir -p $dist_dir
cp -f "$source" "$dist_dir/$basename"
# 如果 {fwbootmgr} displayorder 为空
# 执行 bcdedit /copy '{bootmgr}' 会报错
# 例如 azure windows 2016 模板
# 要先设置默认的 {fwbootmgr} displayorder
# https://github.com/hakuna-m/wubiuefi/issues/286
bcdedit /set '{fwbootmgr}' displayorder '{bootmgr}' /addfirst
2023-11-09 22:27:07 +08:00
# 添加启动项
id=$(bcdedit /copy '{bootmgr}' /d "$(get_entry_name)" | grep -o '{.*}')
bcdedit /set $id device partition=$x:
bcdedit /set $id path \\EFI\\reinstall\\$basename
bcdedit /set '{fwbootmgr}' bootsequence $id
}
2023-06-18 21:27:22 +08:00
2023-11-09 22:27:07 +08:00
get_maybe_efi_dirs_in_linux() {
# arch云镜像efi分区挂载在/efi且使用 autofs挂载后会有两个 /efi 条目
2024-01-27 23:17:50 +08:00
mount | awk '$5=="vfat" || $5=="autofs" {print $3}' | grep -E '/boot|/efi' | sort -u
2023-11-09 22:27:07 +08:00
}
get_disk_by_part() {
dev_part=$1
2024-02-01 00:59:34 +08:00
install_pkg lsblk >&2
lsblk -rn --inverse "$dev_part" | grep -w disk | awk '{print $1}'
}
get_part_num_by_part() {
dev_part=$1
grep -oE '[0-9]*$' <<<"$dev_part"
}
grep_efi_index() {
awk -F '*' '{print $1}' | sed 's/Boot//'
}
2023-11-09 22:27:07 +08:00
add_efi_entry_in_linux() {
source=$1
install_pkg efibootmgr
for efi_part in $(get_maybe_efi_dirs_in_linux); do
2024-05-22 22:32:43 +08:00
if find $efi_part -iname "*.efi" >/dev/null; then
2023-11-09 22:27:07 +08:00
dist_dir=$efi_part/EFI/reinstall
basename=$(basename $source)
mkdir -p $dist_dir
if [[ "$source" = http* ]]; then
curl -Lo "$dist_dir/$basename" "$source"
else
cp -f "$source" "$dist_dir/$basename"
fi
2023-11-09 22:27:07 +08:00
if false; then
grub_probe="$(command -v grub-probe grub2-probe)"
dev_part="$("$grub_probe" -t device "$dist_dir")"
else
install_pkg findmnt
# arch findmnt 会得到
# systemd-1
# /dev/sda2
dev_part=$(findmnt -T "$dist_dir" -no SOURCE | grep '^/dev/')
fi
id=$(efibootmgr --create-only \
--disk "/dev/$(get_disk_by_part $dev_part)" \
--part "$(get_part_num_by_part $dev_part)" \
2023-11-09 22:27:07 +08:00
--label "$(get_entry_name)" \
--loader "\\EFI\\reinstall\\$basename" |
tail -1 | grep_efi_index)
2023-11-09 22:27:07 +08:00
efibootmgr --bootnext $id
return
fi
done
error_and_exit "Can't find efi partition."
}
install_grub_linux_efi() {
info 'download grub efi'
if [ "$basearch" = aarch64 ]; then
grub_efi=grubaa64.efi
else
grub_efi=grubx64.efi
fi
2024-02-02 00:49:24 +08:00
# fedora 39 的 efi 无法识别 opensuse tumbleweed 的 xfs
efi_distro=opensuse
2023-12-23 21:03:56 +08:00
2024-01-27 23:17:50 +08:00
# 不要用 download.opensuse.org 和 download.fedoraproject.org
# 因为 ipv6 访问有时跳转到 ipv4 地址,造成 ipv6 only 机器无法下载
2024-05-22 22:32:43 +08:00
# 日韩机器有时得到国内镜像源,但镜像源屏蔽了国外 IP 导致连不上
# https://mirrors.bfsu.edu.cn/opensuse/ports/aarch64/tumbleweed/repo/oss/EFI/BOOT/grub.efi
2023-12-23 21:03:56 +08:00
if [ "$efi_distro" = fedora ]; then
2024-05-22 22:32:43 +08:00
fedora_ver=40
if is_in_china; then
2024-06-06 20:46:07 +08:00
mirror=https://mirrors.ustc.edu.cn/fedora
else
2024-01-27 23:17:50 +08:00
mirror=https://mirror.fcix.net/fedora/linux
fi
curl -Lo $tmp/$grub_efi $mirror/releases/$fedora_ver/Everything/$basearch/os/EFI/BOOT/$grub_efi
else
if is_in_china; then
mirror=https://mirror.sjtu.edu.cn/opensuse
else
2024-01-27 23:17:50 +08:00
mirror=https://mirror.fcix.net/opensuse
fi
2024-02-02 00:49:24 +08:00
[ "$basearch" = x86_64 ] && ports='' || ports=/ports/$basearch
curl -Lo $tmp/$grub_efi $mirror$ports/tumbleweed/repo/oss/EFI/BOOT/grub.efi
fi
add_efi_entry_in_linux $tmp/$grub_efi
}
2023-11-09 22:27:07 +08:00
install_grub_win() {
2023-06-18 21:27:22 +08:00
# 下载 grub
info download grub
grub_ver=2.06
2024-05-22 22:32:43 +08:00
# ftpmirror.gnu.org 是 geoip 重定向,不是 cdn
# 有可能重定义到一个拉黑了部分 IP 的服务器
2024-06-06 20:46:07 +08:00
is_in_china && grub_url=https://mirrors.ustc.edu.cn/gnu/grub/grub-$grub_ver-for-windows.zip ||
2023-12-04 22:19:59 +08:00
grub_url=https://ftpmirror.gnu.org/gnu/grub/grub-$grub_ver-for-windows.zip
curl -Lo $tmp/grub.zip $grub_url
# unzip -qo $tmp/grub.zip
7z x $tmp/grub.zip -o$tmp -r -y -xr!i386-efi -xr!locale -xr!themes -bso0
grub_dir=$tmp/grub-$grub_ver-for-windows
2023-11-09 22:27:07 +08:00
grub=$grub_dir/grub
2023-06-18 21:27:22 +08:00
2024-01-27 23:08:34 +08:00
# 设置 grub 包含的模块
2023-12-22 23:59:46 +08:00
# 原系统是 windows因此不需要 ext2 lvm xfs btrfs
2024-01-27 23:08:34 +08:00
grub_modules+=" normal minicmd serial ls echo test cat reboot halt linux chain search all_video configfile"
2023-12-22 23:59:46 +08:00
grub_modules+=" scsi part_msdos part_gpt fat ntfs ntfscomp lzopio xzio gzio zstd"
2023-06-18 21:27:22 +08:00
if ! is_efi; then
2024-01-27 23:08:34 +08:00
grub_modules+=" biosdisk linux16"
2023-06-18 21:27:22 +08:00
fi
# 设置 grub prefix 为c盘根目录
# 运行 grub-probe 会改变cmd窗口字体
2024-01-27 23:17:50 +08:00
prefix=$($grub-probe -t drive $c: | sed 's|.*PhysicalDrive|(hd|' | del_cr)/
2023-06-18 21:27:22 +08:00
echo $prefix
# 安装 grub
if is_efi; then
# efi
info install grub for efi
2024-01-27 23:08:34 +08:00
if [ "$basearch" = aarch64 ]; then
2024-05-22 22:32:43 +08:00
# 3.20 是 grub 2.12,可能会有问题
2024-01-27 23:08:34 +08:00
alpine_ver=3.19
2024-06-06 20:46:07 +08:00
is_in_china && mirror=http://mirrors.ustc.edu.cn/alpine || mirror=https://dl-cdn.alpinelinux.org/alpine
2024-01-27 23:08:34 +08:00
grub_efi_apk=$(curl -L $mirror/v$alpine_ver/main/aarch64/ | grep -oP 'grub-efi-.*?apk' | head -1)
mkdir -p $tmp/grub-efi
curl -L "$mirror/v$alpine_ver/main/aarch64/$grub_efi_apk" | tar xz --warning=no-unknown-keyword -C $tmp/grub-efi/
cp -r $tmp/grub-efi/usr/lib/grub/arm64-efi/ $grub_dir
2024-01-27 23:08:34 +08:00
$grub-mkimage -p $prefix -O arm64-efi -o "$(cygpath -w $grub_dir/grubaa64.efi)" $grub_modules
add_efi_entry_in_windows $grub_dir/grubaa64.efi
else
$grub-mkimage -p $prefix -O x86_64-efi -o "$(cygpath -w $grub_dir/grubx64.efi)" $grub_modules
add_efi_entry_in_windows $grub_dir/grubx64.efi
fi
2023-06-18 21:27:22 +08:00
else
# bios
info install grub for bios
2024-01-27 23:19:14 +08:00
# bootmgr 加载 g2ldr 有大小限制
# 超过大小会报错 0xc000007b
2023-10-07 22:36:40 +08:00
# 解决方法1 g2ldr.mbr + g2ldr
2023-06-18 21:27:22 +08:00
# 解决方法2 生成少于64K的 g2ldr + 动态模块
2024-01-27 23:19:14 +08:00
if false; then
# g2ldr.mbr
# 部分国内机无法访问 ftp.cn.debian.org
2024-06-06 20:46:07 +08:00
is_in_china && host=mirrors.ustc.edu.cn || host=deb.debian.org
2024-01-27 23:19:14 +08:00
curl -LO http://$host/debian/tools/win32-loader/stable/win32-loader.exe
7z x win32-loader.exe 'g2ldr.mbr' -o$tmp/win32-loader -r -y -bso0
find $tmp/win32-loader -name 'g2ldr.mbr' -exec cp {} /cygdrive/$c/ \;
2024-01-27 23:19:14 +08:00
# g2ldr
# 配置文件 c:\grub.cfg
$grub-mkimage -p "$prefix" -O i386-pc -o "$(cygpath -w $grub_dir/core.img)" $grub_modules
cat $grub_dir/i386-pc/lnxboot.img $grub_dir/core.img >/cygdrive/$c/g2ldr
else
# grub-install 无法设置 prefix
# 配置文件 c:\grub\grub.cfg
$grub-install $c \
--target=i386-pc \
--boot-directory=$c: \
--install-modules="$grub_modules" \
--themes= \
--fonts= \
--no-bootsector
cat $grub_dir/i386-pc/lnxboot.img /cygdrive/$c/grub/i386-pc/core.img >/cygdrive/$c/g2ldr
fi
2023-06-18 21:27:22 +08:00
# 添加引导
# 脚本可能不是首次运行,所以先删除原来的
id='{1c41f649-1637-52f1-aea8-f96bfebeecc8}'
bcdedit /enum all | grep --text $id && bcdedit /delete $id
2023-11-09 22:27:07 +08:00
bcdedit /create $id /d "$(get_entry_name)" /application bootsector
2023-06-18 21:27:22 +08:00
bcdedit /set $id device partition=$c:
2024-01-27 23:19:14 +08:00
bcdedit /set $id path \\g2ldr
2023-06-18 21:27:22 +08:00
bcdedit /displayorder $id /addlast
bcdedit /bootsequence $id /addfirst
2023-05-17 22:54:26 +08:00
fi
}
2023-05-03 22:22:21 +08:00
# 转换 finalos_a=1 为 finalos.a=1 ,排除 finalos_mirrorlist
build_finalos_cmdline() {
2023-06-18 21:27:22 +08:00
if vars=$(compgen -v finalos_); then
for key in $vars; do
value=${!key}
key=${key#finalos_}
if [ -n "$value" ] && [ $key != "mirrorlist" ]; then
finalos_cmdline+=" finalos.$key='$value'"
fi
done
fi
2023-05-03 22:22:21 +08:00
}
build_extra_cmdline() {
2024-07-09 23:37:51 +08:00
for key in confhome hold force cloud_image main_disk; do
2023-05-13 00:14:46 +08:00
value=${!key}
if [ -n "$value" ]; then
2023-05-25 20:15:12 +08:00
extra_cmdline+=" extra.$key='$value'"
2023-05-03 22:22:21 +08:00
fi
2023-05-13 00:14:46 +08:00
done
# 指定最终安装系统的 mirrorlist链接有&在grub中是特殊字符所以要加引号
if [ -n "$finalos_mirrorlist" ]; then
extra_cmdline+=" extra.mirrorlist='$finalos_mirrorlist'"
elif [ -n "$nextos_mirrorlist" ]; then
extra_cmdline+=" extra.mirrorlist='$nextos_mirrorlist'"
2023-05-03 22:22:21 +08:00
fi
# cloudcone 特殊处理
if is_grub_dir_linked; then
finalos_cmdline+=" extra.link_grub_dir=1"
fi
2023-05-03 22:22:21 +08:00
}
2022-09-25 21:57:47 +08:00
2023-09-03 19:35:01 +08:00
echo_tmp_ttys() {
2023-10-08 21:58:16 +08:00
curl -L $confhome/ttys.sh | sh -s "console="
2023-09-03 19:35:01 +08:00
}
2023-11-09 22:27:07 +08:00
get_entry_name() {
2024-01-27 23:17:50 +08:00
printf 'reinstall ('
printf '%s' "$distro"
[ -n "$releasever" ] && printf ' %s' "$releasever"
[ "$distro" = alpine ] && [ "$hold" = 1 ] && printf ' Live OS'
printf ')'
2023-11-09 22:27:07 +08:00
}
2023-06-18 21:27:22 +08:00
# shellcheck disable=SC2154
2023-12-02 22:46:16 +08:00
build_nextos_cmdline() {
if [ $nextos_distro = alpine ]; then
nextos_cmdline="alpine_repo=$nextos_repo modloop=$nextos_modloop"
2024-05-08 22:17:33 +08:00
elif is_distro_like_debian $nextos_distro; then
2024-05-03 21:37:07 +08:00
nextos_cmdline="lowmem/low=1 auto=true priority=critical"
nextos_cmdline+=" url=$nextos_ks"
nextos_cmdline+=" mirror/http/hostname=$nextos_deb_hostname"
nextos_cmdline+=" mirror/http/directory=/$nextos_distro"
nextos_cmdline+=" base-installer/kernel/image=$nextos_kernel"
# kali 安装好后网卡是 eth0 这种格式,但安装时不是
if [ "$nextos_distro" = kali ]; then
nextos_cmdline+=" net.ifnames=0"
nextos_cmdline+=" simple-cdd/profiles=kali"
2024-05-03 21:37:07 +08:00
fi
2024-05-08 22:17:33 +08:00
elif is_distro_like_redhat $nextos_distro; then
2023-12-02 22:46:16 +08:00
# redhat
nextos_cmdline="root=live:$nextos_squashfs inst.ks=$nextos_ks"
fi
2024-05-08 22:17:33 +08:00
if is_distro_like_debian $nextos_distro; then
2024-03-17 00:08:48 +08:00
if [ "$basearch" = "x86_64" ]; then
# debian installer 好像第一个 tty 是主 tty
2024-03-18 23:56:10 +08:00
# 设置ttyS0,tty0,安装界面还是显示在ttyS0
2024-03-17 00:08:48 +08:00
:
else
2024-03-18 23:56:10 +08:00
# debian arm 在没有ttyAMA0的机器上aws t4g最少要设置一个tty才能启动
# 只设置tty0也行但安装过程ttyS0没有显示
nextos_cmdline+=" console=ttyAMA0,115200 console=ttyS0,115200 console=tty0"
2024-03-17 00:08:48 +08:00
fi
else
2024-03-18 23:56:10 +08:00
# nextos_cmdline+=" $(echo_tmp_ttys)"
nextos_cmdline+=" console=ttyAMA0,115200 console=ttyS0,115200 console=tty0"
2024-03-17 00:08:48 +08:00
fi
2023-12-02 22:46:16 +08:00
# nextos_cmdline+=" mem=256M"
2024-05-03 21:37:07 +08:00
# nextos_cmdline+=" lowmem=+1"
2023-12-02 22:46:16 +08:00
}
2023-06-18 21:27:22 +08:00
build_cmdline() {
2023-12-02 22:46:16 +08:00
# nextos
build_nextos_cmdline
2023-06-18 21:27:22 +08:00
2023-12-02 22:46:16 +08:00
# finalos
# trans 需要 finalos_distro 识别是安装 alpine 还是其他系统
if [ "$distro" = alpine ]; then
finalos_distro=alpine
2023-06-18 21:27:22 +08:00
fi
2023-12-02 22:46:16 +08:00
if [ -n "$finalos_distro" ]; then
build_finalos_cmdline
fi
# extra
build_extra_cmdline
cmdline="$nextos_cmdline $finalos_cmdline $extra_cmdline"
2023-06-18 21:27:22 +08:00
}
2023-05-13 00:14:46 +08:00
# 脚本可能多次运行,先清理之前的残留
mkdir_clear() {
dir=$1
if [ -z "$dir" ] || [ "$dir" = / ]; then
return
fi
# alpine 没有 -R
2023-07-23 11:44:47 +08:00
# { umount $dir || umount -R $dir || true; } 2>/dev/null
rm -rf $dir
mkdir -p $dir
}
mod_initrd_debian_kali() {
# hack 1
# 允许设置 ipv4 onlink 网关
sed -Ei 's,&&( onlink=),||\1,' etc/udhcpc/default.script
# hack 2
# 修改 /var/lib/dpkg/info/netcfg.postinst 运行我们的脚本
# shellcheck disable=SC1091,SC2317
netcfg() {
#!/bin/sh
. /usr/share/debconf/confmodule
db_progress START 0 5 debian-installer/netcfg/title
: get_ip_conf_cmd
# 运行 trans.sh保存配置
db_progress INFO base-installer/progress/netcfg
2024-04-03 21:53:52 +08:00
sh /trans.sh
db_progress STEP 1
}
2023-05-13 00:14:46 +08:00
# 直接覆盖 net-retriever方便调试
# curl -Lo /usr/lib/debian-installer/retriever/net-retriever $confhome/net-retriever
postinst=var/lib/dpkg/info/netcfg.postinst
get_function_content netcfg >$postinst
get_ip_conf_cmd | insert_into_file $postinst after ": get_ip_conf_cmd"
# cat $postinst
2024-04-03 21:53:52 +08:00
# shellcheck disable=SC2317
expand_packages() {
2024-05-03 21:37:07 +08:00
expand_packages_real "$@" | while IFS= read -r line; do
key_=$(echo "$line" | cut -d' ' -f1)
value=$(echo "$line" | cut -d' ' -f2-)
case "$key_" in
Package:)
package="$value"
2024-04-03 21:53:52 +08:00
;;
2024-05-03 21:37:07 +08:00
Priority:)
2024-04-03 21:53:52 +08:00
# shellcheck disable=SC2154
2024-05-03 21:37:07 +08:00
if [ "$value" = standard ] && echo "$disabled_list" | grep -qx "$package"; then
line="Priority: optional"
elif [[ "$package" = ata-modules* ]]; then
# 改成强制安装
# 因为是 pata-modules sata-modules scsi-modules 的依赖
# 但我们没安装它们,也就不会自动安装 ata-modules
line="Priority: standard"
2024-04-03 21:53:52 +08:00
fi
;;
esac
2024-05-03 21:37:07 +08:00
echo "$line"
2024-04-03 21:53:52 +08:00
done
}
# shellcheck disable=SC2012
kver=$(ls -d lib/modules/* | awk -F/ '{print $NF}')
net_retriever=usr/lib/debian-installer/retriever/net-retriever
sed -i 's/^expand_packages()/expand_packages_real()/' $net_retriever
insert_into_file $net_retriever after '#!/bin/sh' <<EOF
disabled_list="
depthcharge-tools-installer
kickseed-common
nobootloader
partman-btrfs
partman-cros
partman-iscsi
partman-jfs
partman-md
partman-xfs
rescue-check
wpasupplicant-udeb
nic-modules-$kver-di
nic-pcmcia-modules-$kver-di
nic-usb-modules-$kver-di
nic-wireless-modules-$kver-di
nic-shared-modules-$kver-di
pcmcia-modules-$kver-di
pcmcia-storage-modules-$kver-di
cdrom-core-modules-$kver-di
firewire-core-modules-$kver-di
usb-storage-modules-$kver-di
isofs-modules-$kver-di
jfs-modules-$kver-di
xfs-modules-$kver-di
loop-modules-$kver-di
pata-modules-$kver-di
sata-modules-$kver-di
scsi-modules-$kver-di
"
expand_packages() {
$(get_function_content expand_packages)
}
EOF
# https://github.com/linuxhw/LsPCI?tab=readme-ov-file#storageata-pci
2024-04-03 21:53:52 +08:00
# https://debian.pkgs.org/12/debian-main-amd64/linux-image-6.1.0-18-cloud-amd64_6.1.76-1_amd64.deb.html
# https://deb.debian.org/debian/pool/main/l/linux-signed-amd64/
# https://deb.debian.org/debian/dists/bookworm/main/debian-installer/binary-all/Packages.xz
# https://deb.debian.org/debian/dists/bookworm/main/debian-installer/binary-amd64/Packages.xz
# 以下是 debian-installer 有的驱动,这些驱动云内核不一定都有,(+)表示云内核有
# scsi-core-modules 默认安装(不用修改),是 ata-modules 的依赖
# 包含 sd_mod.ko(+) scsi_mod.ko(+) scsi_transport_fc.ko(+) scsi_transport_sas.ko(+) scsi_transport_spi.ko(+)
# ata-modules 默认可选(改成必装),是下方模块的依赖。只有 ata_generic.ko(+) 和 libata.ko(+) 两个驱动
# pata-modules 默认安装(改成可选),里面的驱动都是 pata_ 开头,但只有 pata_legacy.ko(+) 在云内核中
# sata-modules 默认安装(改成可选),里面的驱动大部分是 sata_ 开头的,其他重要的还有 ahci.ko libahci.ko ata_piix.ko(+)
2024-04-03 21:53:52 +08:00
# 云内核没有 sata 模块,也没有内嵌,有一个 CONFIG_SATA_HOST=ylibata-$(CONFIG_SATA_HOST) += libata-sata.o
# scsi-modules 默认安装(改成可选),包含 nvme.ko(+) 和各种虚拟化驱动(+)
2024-04-03 21:53:52 +08:00
download_and_extract_udeb() {
package=$1
extract_dir=$2
# 获取 udeb 列表
udeb_list=$tmp/udeb_list
if ! [ -f $udeb_list ]; then
2024-05-03 21:37:07 +08:00
# shellcheck disable=SC2154
curl -L http://$nextos_deb_hostname/$distro/dists/$nextos_codename/main/debian-installer/binary-$basearch_alt/Packages.gz |
2024-04-03 21:53:52 +08:00
zcat | grep 'Filename:' | awk '{print $2}' >$udeb_list
fi
# 下载 udeb
2024-05-03 21:37:07 +08:00
curl -Lo $tmp/tmp.udeb http://$nextos_deb_hostname/$distro/"$(grep /$package $udeb_list)"
2024-04-03 21:53:52 +08:00
if false; then
# 使用 dpkg
# cygwin 没有 dpkg
install_pkg dpkg
dpkg -x $tmp/tmp.udeb $extract_dir
else
# 使用 ar tar xz
# cygwin 需安装 binutils
# centos7 ar 不支持 --output
install_pkg ar tar xz
(cd $tmp && ar x $tmp/tmp.udeb)
tar xf $tmp/data.tar.xz -C $extract_dir
fi
}
# 不用在 windows 判断是哪种硬盘控制器,因为 256M 运行 windows 只可能是 xp而脚本本来就不支持 xp
# 在 debian installer 中判断能否用云内核
create_can_use_cloud_kernel_sh can_use_cloud_kernel.sh
# 最近 kali initrd 删除了原版 wget
# 但 initrd 的 busybox wget 又不支持 https
# 因此改成在这里下载
curl -LO "$confhome/get-xda.sh"
curl -LO "$confhome/ttys.sh"
# 可以节省一点内存?
echo 'export DEBCONF_DROP_TRANSLATIONS=1' |
insert_into_file lib/debian-installer/menu before 'exec debconf'
# 还原 kali netinst.iso 的 simple-cdd 机制
# 主要用于调用 kali.postinst 设置 zsh 为默认 shell
# 但 mini.iso 又没有这种机制
# https://gitlab.com/kalilinux/build-scripts/live-build-config/-/raw/master/kali-config/common/includes.installer/kali-finish-install?ref_type=heads
# https://salsa.debian.org/debian/simple-cdd/-/blob/master/debian/14simple-cdd?ref_type=heads
# https://http.kali.org/pool/main/s/simple-cdd/simple-cdd-profiles_0.6.9_all.udeb
if [ "$distro" = kali ]; then
# 但我们没有使用 iso因此没有 kali.postinst需要另外下载
mkdir -p cdrom/simple-cdd
curl -Lo cdrom/simple-cdd/kali.postinst https://gitlab.com/kalilinux/build-scripts/live-build-config/-/raw/master/kali-config/common/includes.installer/kali-finish-install?ref_type=heads
chmod a+x cdrom/simple-cdd/kali.postinst
fi
2024-04-03 21:53:52 +08:00
# 提前下载 fdisk
# 因为 fdisk-udeb 包含 fdisk 和 sfdisk提前下载可减少占用
mkdir_clear $tmp/fdisk
download_and_extract_udeb fdisk-udeb $tmp/fdisk
cp -f $tmp/fdisk/usr/sbin/fdisk usr/sbin/
# >256M 或者当前系统是 windows
if [ $ram_size -gt 256 ] || is_in_windows; then
2024-04-03 21:53:52 +08:00
sed -i '/^pata-modules/d' $net_retriever
sed -i '/^sata-modules/d' $net_retriever
sed -i '/^scsi-modules/d' $net_retriever
else
# <=256M 极限优化
find_main_disk
extra_drivers=
for driver in $(get_disk_drivers); do
2024-04-03 21:53:52 +08:00
echo "using driver: $driver"
case $driver in
nvme | virtio_blk | virtio_scsi | xen_blkfront | xen_scsifront | hv_storvsc | vmw_pvscsi) extra_drivers+=" $driver" ;;
pata_legacy) sed -i '/^pata-modules/d' $net_retriever ;; # 属于 pata-modules
ata_piix) sed -i '/^sata-modules/d' $net_retriever ;; # 属于 sata-modules
ata_generic) ;; # 属于 ata-modules不用处理因为我们设置强制安装了 ata-modules
2024-04-03 21:53:52 +08:00
esac
done
# extra drivers
# xen 还需要以下两个?
# kernel/drivers/xen/xen-scsiback.ko
# kernel/drivers/block/xen-blkback/xen-blkback.ko
# 但反查也找不到 curl https://deb.debian.org/debian/dists/bookworm/main/Contents-udeb-amd64.gz | zcat | grep xen
2024-04-03 21:53:52 +08:00
if [ -n "$extra_drivers" ]; then
mkdir_clear $tmp/scsi
download_and_extract_udeb scsi-modules-$kver-di $tmp/scsi
udeb_drivers_dir=$tmp/scsi/lib/modules/$kver/kernel/drivers
2024-04-03 21:53:52 +08:00
(
cd lib/modules/*/kernel/drivers/
mkdir -p block scsi nvme/host
2024-04-03 21:53:52 +08:00
for driver in $extra_drivers; do
echo "adding driver: $driver"
case $driver in
# debian 模块没有压缩
# kali 模块有压缩
# 因此要有 *
2024-04-03 21:53:52 +08:00
nvme)
cp -fv $udeb_drivers_dir/nvme/host/nvme.ko* nvme/host/
cp -fv $udeb_drivers_dir/nvme/host/nvme-core.ko* nvme/host/
2024-04-03 21:53:52 +08:00
;;
virtio_blk) cp -fv $udeb_drivers_dir/block/virtio_blk.ko* block/ ;;
virtio_scsi) cp -fv $udeb_drivers_dir/scsi/virtio_scsi.ko* scsi/ ;;
# xen 的横杠特别不同
xen_blkfront) cp -fv $udeb_drivers_dir/block/xen-blkfront.ko* block/ ;;
xen_scsifront) cp -fv $udeb_drivers_dir/scsi/xen-scsifront.ko* scsi/ ;;
hv_storvsc) cp -fv $udeb_drivers_dir/scsi/hv_storvsc.ko* scsi/ ;;
vmw_pvscsi) cp -fv $udeb_drivers_dir/scsi/vmw_pvscsi.ko* scsi/ ;;
2024-04-03 21:53:52 +08:00
esac
done
)
fi
fi
# amd64)
# level1=737 # MT=754108, qemu: -m 780
# level2=424 # MT=433340, qemu: -m 460
# min=316 # MT=322748, qemu: -m 350
2024-03-31 01:07:33 +08:00
# 将 use_level 2 9 修改为 use_level 1
2024-04-03 21:53:52 +08:00
# x86 use_level 2 会出现 No root file system is defined.
# arm 即使 use_level 1 也会出现 No root file system is defined.
2024-03-31 01:07:33 +08:00
sed -i 's/use_level=[29]/use_level=1/' lib/debian-installer-startup.d/S15lowmem
# hack 3
# 修改 trans.sh
# 1. 直接调用 create_ifupdown_config
insert_into_file $tmp_dir/trans.sh after ': main' <<EOF
2024-05-03 21:37:07 +08:00
distro=$nextos_distro
create_ifupdown_config /etc/network/interfaces
exit
EOF
# 2. 删除 debian busybox 无法识别的语法
# 3. 删除 apk 语句
# 4. debian 11/12 initrd 无法识别 > >
# 5. debian 11/12 initrd 无法识别 < <
# 6. debian 11 initrd 无法识别 set -E
# 7. debian 11 initrd 无法识别 trap ERR
# 删除或注释,可能会导致空方法而报错,因此改为替换成'\n: #'
replace='\n: #'
sed -Ei "s/> >/$replace/" $tmp_dir/trans.sh
sed -Ei "s/< </$replace/" $tmp_dir/trans.sh
sed -Ei "s/(^[[:space:]]*set[[:space:]].*)E/\1/" $tmp_dir/trans.sh
sed -Ei "s/^[[:space:]]*apk[[:space:]]/$replace/" $tmp_dir/trans.sh
sed -Ei "s/^[[:space:]]*trap[[:space:]]/$replace/" $tmp_dir/trans.sh
}
get_disk_drivers() {
get_drivers "/sys/block/$1"
}
get_net_drivers() {
get_drivers "/sys/class/net/$1"
}
# 不用在 windows 判断是哪种硬盘/网络驱动,因为 256M 运行 windows 只可能是 xp而脚本本来就不支持 xp
# 而且安装过程也有二次判断
get_drivers() {
# 有以下结果组合出现
# sd_mod
# virtio_blk
# virtio_scsi
# virtio_pci
# pcieport
# xen_blkfront
# ahci
# nvme
2024-05-29 21:30:24 +08:00
# mptspi
# mptsas
# vmw_pvscsi
(
cd "$(readlink -f $1)"
while ! [ "$(pwd)" = / ]; do
if [ -d driver ]; then
if [ -d driver/module ]; then
# 显示全名,例如 xen_blkfront sd_mod
# 但 ahci 没有这个文件,所以 else 不能省略
basename "$(readlink -f driver/module)"
else
# 不显示全名,例如 vbd sd
basename "$(readlink -f driver)"
fi
fi
cd ..
done
)
}
exit_if_cant_use_cloud_kernel() {
find_main_disk
collect_netconf
# shellcheck disable=SC2154
if ! can_use_cloud_kernel "$xda" $ipv4_ethx $ipv6_ethx; then
error_and_exit "Can't use cloud kernel. And not enough RAM to run normal kernel."
fi
}
can_use_cloud_kernel() {
# initrd 下也要使用,不要用 <<<
# 有些虚拟机用了 ahci但云内核没有 ahci 驱动
cloud_eth_modules='ena|gve|mana|virtio_net|xen_netfront|hv_netvsc|vmxnet3|mlx4_en|mlx4_core|mlx5_core|ixgbevf'
cloud_blk_modules='ata_generic|ata_piix|pata_legacy|nvme|virtio_blk|virtio_scsi|xen_blkfront|xen_scsifront|hv_storvsc|vmw_pvscsi'
# disk
drivers="$(get_disk_drivers $1)"
shift
for driver in $drivers; do
echo "using disk driver: $driver"
done
echo "$drivers" | grep -Ewq "$cloud_blk_modules" || return 1
# net
# v4 v6 eth 相同,只检查一次
if [ "$1" = "$2" ]; then
shift
fi
while [ $# -gt 0 ]; do
drivers="$(get_net_drivers $1)"
shift
for driver in $drivers; do
echo "using net driver: $driver"
done
echo "$drivers" | grep -Ewq "$cloud_eth_modules" || return 1
done
}
create_can_use_cloud_kernel_sh() {
cat <<EOF >$1
$(get_function get_drivers)
$(get_function get_net_drivers)
$(get_function get_disk_drivers)
$(get_function can_use_cloud_kernel)
can_use_cloud_kernel "\$@"
EOF
}
get_ip_conf_cmd() {
collect_netconf >&2
is_in_china && is_in_china=true || is_in_china=false
sh=/alpine-network.sh
if is_found_ipv4_netconf && is_found_ipv6_netconf && [ "$ipv4_mac" = "$ipv6_mac" ]; then
echo "'$sh' '$ipv4_mac' '$ipv4_addr' '$ipv4_gateway' '$ipv6_addr' '$ipv6_gateway' '$is_in_china'"
else
if is_found_ipv4_netconf; then
echo "'$sh' '$ipv4_mac' '$ipv4_addr' '$ipv4_gateway' '' '' '$is_in_china'"
fi
if is_found_ipv6_netconf; then
echo "'$sh' '$ipv6_mac' '' '' '$ipv6_addr' '$ipv6_gateway' '$is_in_china'"
fi
fi
}
mod_initrd_alpine() {
2024-05-22 21:53:32 +08:00
# hack 1 v3.19 和之前的 virt 内核需添加 ipv6 模块
if virt_dir=$(ls -d $tmp_dir/lib/modules/*-virt 2>/dev/null); then
ipv6_dir=$virt_dir/kernel/net/ipv6
2024-05-22 21:53:32 +08:00
if ! [ -f $ipv6_dir/ipv6.ko ] || ! grep -q ipv6 $tmp_dir/lib/modules/*/modules.builtin; then
2024-03-31 00:47:13 +08:00
mkdir -p $ipv6_dir
modloop_file=$tmp/modloop_file
modloop_dir=$tmp/modloop_dir
2024-03-31 00:47:13 +08:00
curl -Lo $modloop_file $nextos_modloop
if is_in_windows; then
# cygwin 没有 unsquashfs
7z e $modloop_file ipv6.ko -r -y -o$ipv6_dir
else
install_pkg unsquashfs
mkdir_clear $modloop_dir
unsquashfs -f -d $modloop_dir $modloop_file 'modules/*/kernel/net/ipv6/ipv6.ko'
find $modloop_dir -name ipv6.ko -exec cp {} $ipv6_dir/ \;
fi
fi
fi
2024-01-20 22:27:01 +08:00
# hack 2 /usr/share/udhcpc/default.script
2023-08-23 14:17:45 +08:00
# 脚本被调用的顺序
# udhcpc: deconfig
# udhcpc: bound
# udhcpc6: deconfig
# udhcpc6: bound
# shellcheck disable=SC2317
2023-10-22 19:07:12 +08:00
udhcpc() {
if [ "$1" = deconfig ]; then
2023-08-23 14:17:45 +08:00
return
fi
2023-10-22 19:07:12 +08:00
if [ "$1" = bound ] && [ -n "$ipv6" ]; then
# shellcheck disable=SC2154
2023-10-22 19:07:12 +08:00
ip -6 addr add "$ipv6" dev "$interface"
ip link set dev "$interface" up
2023-08-23 14:17:45 +08:00
return
fi
2023-10-22 19:07:12 +08:00
}
get_function_content udhcpc |
insert_into_file usr/share/udhcpc/default.script after 'deconfig\|renew\|bound'
# 允许设置 ipv4 onlink 网关
sed -Ei 's,(0\.0\.0\.0\/0),"\1 onlink",' usr/share/udhcpc/default.script
# hack 3 网络配置
# alpine 根据 MAC_ADDRESS 判断是否有网络
# https://github.com/alpinelinux/mkinitfs/blob/c4c0115f9aa5aa8884c923dc795b2638711bdf5c/initramfs-init.in#L914
insert_into_file init after 'configure_ip\(\)' <<EOF
depmod
[ -d /sys/module/ipv6 ] || modprobe ipv6
$(get_ip_conf_cmd)
MAC_ADDRESS=1
return
EOF
# grep -E -A5 'configure_ip\(\)' init
# hack 4 运行 trans.start
2023-05-13 00:14:46 +08:00
# exec /bin/busybox switch_root $switch_root_opts $sysroot $chart_init "$KOPT_init" $KOPT_init_args # 3.17
# exec switch_root $switch_root_opts $sysroot $chart_init "$KOPT_init" $KOPT_init_args # 3.18
# 1. alpine arm initramfs 时间问题 要添加 --no-check-certificate
# 2. aws t4g arm 如果没设置console=ttyx在initramfs里面wget https会出现bad header错误chroot后正常
# Connecting to raw.githubusercontent.com (185.199.108.133:443)
# 60C0BB2FFAFF0000:error:0A00009C:SSL routines:ssl3_get_record:http request:ssl/record/ssl3_record.c:345:
# ssl_client: SSL_connect
# wget: bad header line: <20>
2023-09-16 20:00:05 +08:00
insert_into_file init before '^exec (/bin/busybox )?switch_root' <<EOF
# echo "wget --no-check-certificate -O- $confhome/trans.sh | /bin/ash" >\$sysroot/etc/local.d/trans.start
# wget --no-check-certificate -O \$sysroot/etc/local.d/trans.start $confhome/trans.sh
cp /trans.sh \$sysroot/etc/local.d/trans.start
2023-05-13 00:14:46 +08:00
chmod a+x \$sysroot/etc/local.d/trans.start
ln -s /etc/init.d/local \$sysroot/etc/runlevels/default/
EOF
# 判断云镜像 debain 能否用云内核
if is_distro_like_debian; then
create_can_use_cloud_kernel_sh can_use_cloud_kernel.sh
insert_into_file init before '^exec (/bin/busybox )?switch_root' <<EOF
cp /can_use_cloud_kernel.sh \$sysroot/
chmod a+x \$sysroot/can_use_cloud_kernel.sh
EOF
fi
}
mod_initrd() {
info "mod $nextos_distro initrd"
install_pkg gzip cpio
# 解压
# 先删除临时文件,避免之前运行中断有残留文件
2024-05-22 22:32:43 +08:00
tmp_dir=$tmp/initrd
mkdir_clear $tmp_dir
cd $tmp_dir
# cygwin 下处理 debian initrd 时
# 解压/重新打包/删除 initrd 的 /dev/console /dev/null 都会报错
# cpio: dev/console: Cannot utime: Invalid argument
# cpio: ./dev/console: Cannot stat: Bad address
# 用 windows 文件管理器可删除
# 但同样运行 zcat /reinstall-initrd | cpio -idm
# 打开 C:\cygwin\Cygwin.bat ,运行报错
# 打开桌面的 Cygwin 图标,运行就没问题
# shellcheck disable=SC2046
# nonmatching 是精确匹配路径
zcat /reinstall-initrd | cpio -idm \
$(is_in_windows && echo --nonmatching 'dev/console' --nonmatching 'dev/null')
curl -Lo $tmp_dir/trans.sh $confhome/trans.sh
curl -Lo $tmp_dir/alpine-network.sh $confhome/alpine-network.sh
chmod a+x $tmp_dir/trans.sh $tmp_dir/alpine-network.sh
2024-05-08 22:17:33 +08:00
if is_distro_like_debian $nextos_distro; then
mod_initrd_debian_kali
2024-05-03 21:37:07 +08:00
else
mod_initrd_$nextos_distro
fi
# alpine live 不精简 initrd
# 因为不知道用户想干什么,可能会用到精简的文件
2024-03-31 01:07:33 +08:00
if is_virt && ! is_alpine_live; then
remove_useless_initrd_files
2024-03-31 01:07:33 +08:00
fi
2023-05-13 00:14:46 +08:00
# 重建
# 注意要用 cpio -H newc 不要用 cpio -c ,不同版本的 -c 作用不一样,很坑
# -c Use the old portable (ASCII) archive format
# -c Identical to "-H newc", use the new (SVR4)
# portable format.If you wish the old portable
# (ASCII) archive format, use "-H odc" instead.
2023-12-02 22:46:16 +08:00
find . | cpio --quiet -o -H newc | gzip -1 >/reinstall-initrd
cd - >/dev/null
2023-06-18 21:27:22 +08:00
}
2023-05-13 00:14:46 +08:00
remove_useless_initrd_files() {
# 显示精简前的大小
du -sh .
# 删除 initrd 里面没用的文件/驱动
rm -rf bin/brltty
rm -rf etc/brltty
rm -rf sbin/wpa_supplicant
rm -rf usr/lib/libasound.so.*
rm -rf usr/share/alsa
(
cd lib/modules/*/kernel/drivers/net/ethernet/
for item in *; do
case "$item" in
intel | amazon | google) ;;
*) rm -rf $item ;;
esac
done
)
(
cd lib/modules/*/kernel
for item in \
net/mac80211 \
net/wireless \
net/bluetooth \
drivers/hid \
drivers/mmc \
drivers/mtd \
drivers/usb \
drivers/ssb \
drivers/mfd \
drivers/bcma \
drivers/pcmcia \
drivers/parport \
drivers/platform \
drivers/staging \
drivers/net/usb \
drivers/net/bonding \
drivers/net/wireless \
drivers/input/rmi4 \
drivers/input/keyboard \
drivers/input/touchscreen \
drivers/bus/mhi \
drivers/char/pcmcia \
drivers/misc/cardreader; do
rm -rf $item
done
)
# 显示精简后的大小
du -sh .
}
2023-11-09 22:27:07 +08:00
# 脚本入口
if is_in_windows; then
# win系统盘
c=$(echo $SYSTEMDRIVE | cut -c1)
# 64位系统 + 32位cmd/cygwin需要添加 PATH否则找不到64位系统程序例如bcdedit
sysnative=$(cygpath -u $WINDIR\\Sysnative)
if [ -d $sysnative ]; then
PATH=$PATH:$sysnative
fi
# 更改 windows 命令输出语言为英文
2024-03-22 20:05:53 +08:00
# chcp 会清屏
mode.com con cp select=437 >/dev/null
fi
2023-11-09 22:27:07 +08:00
# 检查 root
2024-01-27 23:17:50 +08:00
if is_in_windows; then
# 64位系统 + 32位cmd/cygwin运行 openfiles 报错:目标系统必须运行 32 位的操作系统
if ! fltmc >/dev/null 2>&1; then
2024-01-27 23:17:50 +08:00
error_and_exit "Please run as administrator."
fi
else
2023-11-09 22:27:07 +08:00
if [ "$EUID" -ne 0 ]; then
2024-01-27 23:17:50 +08:00
error_and_exit "Please run as root."
2023-11-09 22:27:07 +08:00
fi
fi
# 整理参数
2024-07-09 23:37:51 +08:00
if ! opts=$(getopt -n $0 -o "" --long ci,debug,hold:,sleep:,iso:,image-name:,img:,lang:,commit:,force: -- "$@"); then
2023-11-09 22:27:07 +08:00
usage_and_exit
fi
eval set -- "$opts"
# shellcheck disable=SC2034
while true; do
case "$1" in
2024-05-22 22:32:43 +08:00
--commit)
commit=$2
shift 2
;;
2023-11-09 22:27:07 +08:00
--debug)
set -x
shift
;;
2024-04-05 22:45:59 +08:00
--ci)
2023-11-09 22:27:07 +08:00
cloud_image=1
shift
;;
--hold | --sleep)
hold=$2
2024-04-05 22:45:59 +08:00
if ! { [ "$hold" = 1 ] || [ "$hold" = 2 ]; }; then
error_and_exit "Invalid --hold value: $hold."
fi
2023-11-09 22:27:07 +08:00
shift 2
;;
2024-07-09 23:37:51 +08:00
--force)
force=$2
if ! { [ "$force" = bios ] || [ "$force" = efi ]; }; then
error_and_exit "Invalid --force value: $force."
fi
shift 2
;;
2023-11-09 22:27:07 +08:00
--img)
img=$2
shift 2
;;
--iso)
iso=$2
shift 2
;;
--image-name)
2024-04-05 22:45:59 +08:00
image_name=$(echo "$2" | to_lower)
shift 2
;;
--lang)
lang=$(echo "$2" | to_lower)
2023-11-09 22:27:07 +08:00
shift 2
;;
--)
shift
break
;;
*)
echo "Unexpected option: $1."
usage_and_exit
;;
esac
done
# 检查目标系统名
verify_os_name "$@"
# 检查必须的参数
verify_os_args
2023-12-22 23:45:06 +08:00
# 不支持容器虚拟化
assert_not_in_container
# 不支持安全启动
if is_secure_boot_enabled; then
error_and_exit "Please disable secure boot first."
2023-12-22 23:45:06 +08:00
fi
2023-11-09 22:27:07 +08:00
# 必备组件
2024-02-01 00:59:34 +08:00
install_pkg curl grep
2023-11-09 22:27:07 +08:00
# /tmp 挂载在内存的话,可能不够空间
tmp=/reinstall-tmp
2024-05-03 21:37:07 +08:00
mkdir_clear "$tmp"
2024-03-28 00:16:05 +08:00
# 强制忽略/强制添加 --ci 参数
# debian 不强制忽略 ci 留作测试
2024-03-28 00:16:05 +08:00
case "$distro" in
dd | windows | netboot.xyz | kali | alpine | arch | gentoo)
2024-03-28 00:16:05 +08:00
if is_use_cloud_image; then
echo "ignored --ci"
cloud_image=0
fi
;;
redhat | centos | alma | rocky | oracle | ubuntu | fedora | opensuse | anolis | opencloudos | openeuler)
2024-03-28 00:16:05 +08:00
cloud_image=1
;;
esac
2023-11-09 22:27:07 +08:00
2024-03-28 00:16:05 +08:00
# 检查内存
check_ram
2023-11-09 22:27:07 +08:00
# 检查硬件架构
2024-01-27 23:08:34 +08:00
if is_in_windows; then
# x86-based PC
# x64-based PC
# ARM-based PC
# ARM64-based PC
basearch=$(wmic ComputerSystem get SystemType /format:list |
grep '=' | cut -d= -f2 | cut -d- -f1)
else
# archlinux 云镜像没有 arch 命令
# https://en.wikipedia.org/wiki/Uname
basearch=$(uname -m)
fi
# 统一架构名称,并强制 64 位
case "$(echo $basearch | to_lower)" in
i?86 | x64 | x86* | amd64)
basearch=x86_64
basearch_alt=amd64
;;
arm* | aarch64)
basearch=aarch64
basearch_alt=arm64
;;
*) error_and_exit "Unsupported arch: $basearch" ;;
2023-11-09 22:27:07 +08:00
esac
# 设置国内代理
# gitee 不支持ipv6
# jsdelivr 有12小时缓存
# https://github.com/XIU2/UserScript/blob/master/GithubEnhanced-High-Speed-Download.user.js#L31
2024-05-22 22:32:43 +08:00
if [ -n "$github_proxy" ] && [[ "$confhome" = http*://raw.githubusercontent.com/* ]]; then
# 未测试
if false; then
repo=$(echo $confhome | cut -d/ -f4,5)
branch=$(echo $confhome | cut -d/ -f6)
# 避免脚本更新时,文件不同步造成错误
if [ -z "$commit" ]; then
commit=$(curl -L https://api.github.com/repos/$repo/git/refs/heads/$branch |
grep '"sha"' | grep -Eo '[0-9a-f]{40}')
fi
# shellcheck disable=SC2001
confhome=$(echo "$confhome" | sed "s/main$/$commit/")
fi
if is_in_china; then
confhome=${confhome/http:\/\//https:\/\/}
confhome=${confhome/https:\/\/raw.githubusercontent.com/$github_proxy}
fi
2023-11-09 22:27:07 +08:00
fi
2023-12-02 22:46:16 +08:00
# 以下目标系统不需要两步安装
# alpine
2023-11-09 22:27:07 +08:00
# debian
# el7 x86_64 >=1g
# el7 aarch64 >=1.5g
# el8/9/fedora 任何架构 >=2g
if is_netboot_xyz ||
{ ! is_use_cloud_image && {
2024-05-03 21:37:07 +08:00
[ "$distro" = "alpine" ] || is_distro_like_debian ||
{ is_distro_like_redhat && [ $releasever -eq 7 ] && [ $ram_size -ge 1024 ] && [ $basearch = "x86_64" ]; } ||
{ is_distro_like_redhat && [ $releasever -eq 7 ] && [ $ram_size -ge 1536 ] && [ $basearch = "aarch64" ]; } ||
{ is_distro_like_redhat && [ $releasever -ge 8 ] && [ $ram_size -ge 2048 ]; }
2023-11-09 22:27:07 +08:00
}; }; then
setos nextos $distro $releasever
else
2024-05-22 21:53:32 +08:00
# alpine 作为中间系统时,使用 3.20
alpine_ver_for_trans=3.20
2023-11-09 22:27:07 +08:00
setos finalos $distro $releasever
2023-12-02 22:46:16 +08:00
setos nextos alpine $alpine_ver_for_trans
2023-11-09 22:27:07 +08:00
fi
# 删除之前的条目
# 防止第一次运行 netboot.xyz第二次运行其他但还是进入 netboot.xyz
# 防止第一次运行其他,第二次运行 netboot.xyz但还有第一次的菜单
# bios 无论什么情况都用到 grub所以不用处理
if is_efi; then
if is_in_windows; then
rm -f /cygdrive/$c/grub.cfg
bcdedit /set '{fwbootmgr}' bootsequence '{bootmgr}'
bcdedit /enum bootmgr | grep --text -B3 'reinstall' | awk '{print $2}' | grep '{.*}' |
2023-11-09 22:27:07 +08:00
xargs -I {} cmd /c bcdedit /delete {}
2023-09-10 22:23:01 +08:00
else
# shellcheck disable=SC2046
find $(get_maybe_efi_dirs_in_linux) /boot -type f -name 'custom.cfg' -exec rm -f {} \;
2023-11-09 22:27:07 +08:00
install_pkg efibootmgr
efibootmgr | grep -q 'BootNext:' && efibootmgr --quiet --delete-bootnext
efibootmgr | grep 'reinstall' | grep_efi_index |
2023-11-09 22:27:07 +08:00
xargs -I {} efibootmgr --quiet --bootnum {} --delete-bootnum
fi
fi
# 有的机器开启了 kexec例如腾讯云轻量 debian要禁用
if ! is_in_windows && [ -f /etc/default/kexec ]; then
sed -i 's/LOAD_KEXEC=true/LOAD_KEXEC=false/' /etc/default/kexec
2023-11-09 22:27:07 +08:00
fi
# 下载 netboot.xyz / 内核
# shellcheck disable=SC2154
if is_netboot_xyz; then
if is_efi; then
curl -Lo /netboot.xyz.efi $nextos_efi
if is_in_windows; then
add_efi_entry_in_windows /netboot.xyz.efi
else
add_efi_entry_in_linux /netboot.xyz.efi
2023-09-10 22:23:01 +08:00
fi
2023-11-09 22:27:07 +08:00
else
curl -Lo /reinstall-vmlinuz $nextos_vmlinuz
2023-07-28 21:27:16 +08:00
fi
2023-11-09 22:27:07 +08:00
else
# 下载 nextos 内核
info download vmlnuz and initrd
curl -Lo /reinstall-vmlinuz $nextos_vmlinuz
curl -Lo /reinstall-initrd $nextos_initrd
2024-05-03 21:37:07 +08:00
if is_use_firmware; then
curl -Lo /reinstall-firmware $nextos_firmware
fi
fi
2023-06-18 21:27:22 +08:00
2024-05-03 21:37:07 +08:00
# 修改 alpine debian kali initrd
2024-05-08 22:17:33 +08:00
if [ "$nextos_distro" = alpine ] || is_distro_like_debian "$nextos_distro"; then
mod_initrd
fi
# 将内核/netboot.xyz.lkrn 放到正确的位置
if false && is_use_grub; then
if is_in_windows; then
2024-01-27 23:17:50 +08:00
cp -f /reinstall-vmlinuz /cygdrive/$c/
is_have_initrd && cp -f /reinstall-initrd /cygdrive/$c/
else
if is_os_in_btrfs && is_os_in_subvol; then
cp_to_btrfs_root /reinstall-vmlinuz
is_have_initrd && cp_to_btrfs_root /reinstall-initrd
fi
2023-12-02 22:46:16 +08:00
fi
2023-05-13 00:14:46 +08:00
fi
# grub
2023-11-09 22:27:07 +08:00
if is_use_grub; then
# win 使用外部 grub
if is_in_windows; then
install_grub_win
else
2024-01-27 23:17:50 +08:00
# linux aarch64 原系统的 grub 可能无法启动 alpine 3.19 的内核
# 要用去除了内核 magic number 校验的 grub
# 为了方便测试linux x86 efi 也采用外部 grub
if is_efi; then
install_grub_linux_efi
fi
fi
2023-11-09 22:27:07 +08:00
info 'create grub config'
# 寻找 grub.cfg
if is_in_windows; then
2024-01-27 23:19:14 +08:00
if is_efi; then
grub_cfg=/cygdrive/$c/grub.cfg
else
grub_cfg=/cygdrive/$c/grub/grub.cfg
fi
else
# linux
if is_efi; then
# 现在 linux-efi 是使用 reinstall 目录下的 grub
# shellcheck disable=SC2046
efi_reinstall_dir=$(find $(get_maybe_efi_dirs_in_linux) -type d -name "reinstall" | head -1)
grub_cfg=$efi_reinstall_dir/grub.cfg
2023-11-09 22:27:07 +08:00
else
if is_have_cmd update-grub; then
# alpine debian ubuntu
grub_cfg=$(grep -o '[^ ]*grub.cfg' "$(get_cmd_path update-grub)" | head -1)
else
# 找出主配置文件含有menuentry|blscfg
# 现在 efi 用下载的 grub因此不需要查找 efi 目录
grub_cfg=$(
find /boot/grub* \
-type f -name grub.cfg \
-exec grep -E -l 'menuentry|blscfg' {} \;
)
if [ "$(wc -l <<<"$grub_cfg")" -gt 1 ]; then
error_and_exit 'find multi grub.cfg files.'
fi
2023-11-09 22:27:07 +08:00
fi
fi
fi
2023-11-09 22:27:07 +08:00
2024-01-27 23:17:50 +08:00
# 判断用 linux 还是 linuxefi主要是红帽系
# 现在 efi 用下载的 grub因此不需要判断 linux 或 linuxefi
if false && is_use_local_grub; then
2023-11-09 22:27:07 +08:00
# 在x86 efi机器上不同版本的 grub 可能用 linux 或 linuxefi 加载内核
# 通过检测原有的条目有没有 linuxefi 字样就知道当前 grub 用哪一种
2024-01-27 23:17:50 +08:00
# 也可以检测 /etc/grub.d/10_linux
2023-11-09 22:27:07 +08:00
if [ -d /boot/loader/entries/ ]; then
entries="/boot/loader/entries/"
fi
2024-03-01 22:01:22 +08:00
if grep -q -r -E '^[[:space:]]*linuxefi[[:space:]]' $grub_cfg $entries; then
2023-11-09 22:27:07 +08:00
efi=efi
fi
fi
# 找到 grub 程序的前缀
# 并重新生成 grub.cfg
# 因为有些机子例如hython debian的grub.cfg少了40_custom 41_custom
if is_use_local_grub; then
if is_have_cmd grub2-mkconfig; then
grub=grub2
elif is_have_cmd grub-mkconfig; then
grub=grub
else
error_and_exit "grub not found"
fi
$grub-mkconfig -o $grub_cfg
fi
2023-11-09 22:27:07 +08:00
# 选择用 custom.cfg (linux-bios) 还是 grub.cfg (win/linux-efi)
if is_use_local_grub; then
target_cfg=$(dirname $grub_cfg)/custom.cfg
else
target_cfg=$grub_cfg
fi
# 找到 /reinstall-vmlinuz /reinstall-initrd 的绝对路径
if is_in_windows; then
# dir=/cygwin/
dir=$(cygpath -m / | cut -d: -f2-)/
else
# 获取当前系统根目录在 btrfs 中的绝对路径
if is_os_in_btrfs; then
# btrfs subvolume show /
# 输出可能是 / 或 root 或 @/.snapshots/1/snapshot
dir=$(btrfs subvolume show / | head -1)
if ! [ "$dir" = / ]; then
dir="/$dir/"
fi
else
dir=/
fi
fi
vmlinuz=${dir}reinstall-vmlinuz
initrd=${dir}reinstall-initrd
2024-05-03 21:37:07 +08:00
firmware=${dir}reinstall-firmware
# 生成 linux initrd 命令
2023-11-09 22:27:07 +08:00
if is_netboot_xyz; then
linux_cmd="linux16 $vmlinuz"
2023-11-09 22:27:07 +08:00
else
2024-02-01 00:59:34 +08:00
find_main_disk
2023-12-02 22:46:16 +08:00
build_cmdline
linux_cmd="linux$efi $vmlinuz $cmdline"
initrd_cmd="initrd$efi $initrd"
2024-05-03 21:37:07 +08:00
if is_use_firmware; then
initrd_cmd+=" $firmware"
fi
2023-11-09 22:27:07 +08:00
fi
2023-12-02 22:46:16 +08:00
# cloudcone 从光驱的 grub 启动,再加载硬盘的 grub.cfg
# menuentry "Grub 2" --id grub2 {
# set root=(hd0,msdos1)
# configfile /boot/grub2/grub.cfg
# }
# 加载后 $prefix 依然是光驱的 (hd96)/boot/grub
# 导致找不到 $prefix 目录的 grubenv因此读取不到 next_entry
# 以下方法为 cloudcone 重新加载 grubenv
# 需查找 2*2 个文件夹
# 分区:系统 / boot
# 文件夹grub / grub2
# shellcheck disable=SC2121,SC2154
# cloudcone debian 能用但 ubuntu 模板用不了
# ubuntu 模板甚至没显示 reinstall menuentry
load_grubenv_if_not_loaded() {
if ! [ -s $prefix/grubenv ]; then
for dir in /boot/grub /boot/grub2 /grub /grub2; do
set grubenv="($root)$dir/grubenv"
if [ -s $grubenv ]; then
load_env --file $grubenv
if [ "${next_entry}" ]; then
set default="${next_entry}"
set next_entry=
save_env --file $grubenv next_entry
else
set default="0"
fi
return
fi
done
fi
}
# 生成 grub 配置
2024-01-03 21:28:10 +08:00
# 实测 centos 7 lvm 要手动加载 lvm 模块
echo $target_cfg
get_function_content load_grubenv_if_not_loaded >$target_cfg
2024-07-09 23:37:51 +08:00
# 原系统为 openeuler 云镜像,需要添加 --unrestricted否则要输入密码
del_empty_lines <<EOF | tee -a $target_cfg
set timeout_style=menu
2023-06-18 21:27:22 +08:00
set timeout=5
2024-07-09 23:37:51 +08:00
menuentry "$(get_entry_name)" --unrestricted {
$(! is_in_windows && echo 'insmod lvm')
$(is_os_in_btrfs && echo 'set btrfs_relative_path=n')
2023-11-09 22:27:07 +08:00
insmod all_video
search --no-floppy --file --set=root $vmlinuz
2023-11-09 22:27:07 +08:00
$linux_cmd
$initrd_cmd
2023-05-03 22:22:21 +08:00
}
2022-09-25 21:57:47 +08:00
EOF
# 设置重启引导项
if is_use_local_grub; then
2023-11-09 22:27:07 +08:00
$grub-reboot "$(get_entry_name)"
fi
2023-06-18 21:27:22 +08:00
fi
2023-12-04 22:19:59 +08:00
info 'info'
echo "$distro $releasever"
if ! { is_netboot_xyz || is_use_dd; }; then
2023-12-04 22:19:59 +08:00
if [ "$distro" = windows ]; then
username="administrator"
else
username="root"
fi
echo "Username: $username"
echo "Password: 123@@@"
fi
if is_netboot_xyz; then
echo 'Reboot to start netboot.xyz.'
elif is_alpine_live; then
echo 'Reboot to start Alpine Live OS.'
elif is_use_dd; then
echo 'Reboot to start DD.'
else
2023-12-04 22:19:59 +08:00
echo "Reboot to start the installation."
fi
2024-03-31 00:47:13 +08:00
if is_in_windows; then
echo 'You can run this command to reboot:'
2024-03-31 00:47:13 +08:00
echo 'shutdown /r /t 0'
fi