dd: 修复 alpine 的 file -z 无法看透部分镜像导致文件类型验证失败
This commit is contained in:
parent
a976efffb7
commit
8d1ee541e3
@ -199,7 +199,7 @@ bash reinstall.sh ubuntu --installer
|
|||||||
|
|
||||||
### Feature 2: DD
|
### Feature 2: DD
|
||||||
|
|
||||||
- Supports `raw`, `vhd` images or those compressed with `xz` or `gzip`.
|
- Supports `raw` and `vhd` image formats (either uncompressed or compressed as `.gz`, `.xz`, `.zst`, `.tar`, `.tar.gz`, `.tar.xz`, `.tar.zst`).
|
||||||
- When deploy a Windows image, the system disk will be automatically expanded, and machines with a static IP will have their IP configured, and may take a few minutes after the first boot for the configuration to take effect.
|
- When deploy a Windows image, the system disk will be automatically expanded, and machines with a static IP will have their IP configured, and may take a few minutes after the first boot for the configuration to take effect.
|
||||||
- When deploy a Linux image, will **NOT** modify any contents of the image.
|
- When deploy a Linux image, will **NOT** modify any contents of the image.
|
||||||
|
|
||||||
|
@ -199,7 +199,7 @@ bash reinstall.sh ubuntu --installer
|
|||||||
|
|
||||||
### 功能 2: DD
|
### 功能 2: DD
|
||||||
|
|
||||||
- 支持 `raw` `vhd` 或者经过 `xz` `gzip` 压缩的镜像
|
- 支持 `raw` `vhd` 格式的镜像(未压缩,或者压缩成 `.gz` `.xz` `.zst` `.tar` `.tar.gz` `.tar.xz` `.tar.zst`)
|
||||||
- DD Windows 镜像时,会自动扩展系统盘,静态 IP 的机器会配置好 IP,可能首次开机几分钟后才生效
|
- DD Windows 镜像时,会自动扩展系统盘,静态 IP 的机器会配置好 IP,可能首次开机几分钟后才生效
|
||||||
- DD Linux 镜像时,**不会**修改镜像的任何内容
|
- DD Linux 镜像时,**不会**修改镜像的任何内容
|
||||||
|
|
||||||
|
145
reinstall.sh
145
reinstall.sh
@ -292,24 +292,21 @@ test_url_real() {
|
|||||||
if [ -n "$expect_types" ]; then
|
if [ -n "$expect_types" ]; then
|
||||||
install_pkg file
|
install_pkg file
|
||||||
real_type=$(file_enhanced $tmp_file)
|
real_type=$(file_enhanced $tmp_file)
|
||||||
echo "$real_type"
|
echo "File type: $real_type"
|
||||||
|
|
||||||
# 期待值没有.表示要只需判断外侧
|
for type in $expect_types; do
|
||||||
if ! grep -Fq . <<<"$expect_types"; then
|
if [[ ."$real_type" = *."$type" ]]; then
|
||||||
real_type=$(echo "$real_type" | cut -d. -f2-)
|
# 如果要设置变量
|
||||||
fi
|
if [ -n "$var_to_eval" ]; then
|
||||||
|
IFS=. read -r "${var_to_eval?}" "${var_to_eval}_warp" <<<"$real_type"
|
||||||
|
fi
|
||||||
|
return
|
||||||
|
fi
|
||||||
|
done
|
||||||
|
|
||||||
# 检查
|
failed "$url
|
||||||
if ! grep -Foq "|$real_type|" <<<"|$expect_types|"; then
|
Expected type: $expect_types
|
||||||
failed "$url
|
Actually type: $real_type"
|
||||||
expected: $expect_types
|
|
||||||
actually: $real_type"
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
|
|
||||||
# 如果要设置变量
|
|
||||||
if [ -n "$var_to_eval" ]; then
|
|
||||||
IFS=. read -r "${var_to_eval?}" "${var_to_eval}_warp" <<<"$real_type"
|
|
||||||
fi
|
fi
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -319,45 +316,78 @@ fix_file_type() {
|
|||||||
# 所以不用mime判断
|
# 所以不用mime判断
|
||||||
# https://www.digipres.org/formats/sources/tika/formats/#application/gzip
|
# https://www.digipres.org/formats/sources/tika/formats/#application/gzip
|
||||||
|
|
||||||
|
# centos 7 上的 file 显示 qcow2 的 mime 为 application/octet-stream
|
||||||
|
# file debian-12-genericcloud-amd64.qcow2
|
||||||
|
# debian-12-genericcloud-amd64.qcow2: QEMU QCOW Image (v3), 2147483648 bytes
|
||||||
|
# file --mime debian-12-genericcloud-amd64.qcow2
|
||||||
|
# debian-12-genericcloud-amd64.qcow2: application/octet-stream; charset=binary
|
||||||
|
|
||||||
# --extension 不靠谱
|
# --extension 不靠谱
|
||||||
# file -b /reinstall-tmp/img-test --mime-type
|
# file -b /reinstall-tmp/img-test --mime-type
|
||||||
# application/x-qemu-disk
|
# application/x-qemu-disk
|
||||||
# file -b /reinstall-tmp/img-test --extension
|
# file -b /reinstall-tmp/img-test --extension
|
||||||
# ???
|
# ???
|
||||||
|
|
||||||
# 有些 file 版本输出的是 # ISO 9660 CD-ROM filesystem data ,要去掉开头的井号
|
# 1. 删除,;#
|
||||||
|
# DOS/MBR boot sector; partition 1: ...
|
||||||
|
# gzip compressed data, was ...
|
||||||
|
# # ISO 9660 CD-ROM filesystem data... (有些 file 版本开头输出有井号)
|
||||||
|
|
||||||
# 下面两种都是 raw
|
# 2. 删除开头的空格
|
||||||
|
|
||||||
|
# 3. 删除无意义的单词 POSIX, Unicode, UTF-8, ASCII
|
||||||
|
# POSIX tar archive (GNU)
|
||||||
|
# Unicode text, UTF-8 text
|
||||||
|
# UTF-8 Unicode text, with very long lines
|
||||||
|
# ASCII text
|
||||||
|
|
||||||
|
# 4. 下面两种都是 raw
|
||||||
# DOS/MBR boot sector
|
# DOS/MBR boot sector
|
||||||
# x86 boot sector; partition 1: ...
|
# x86 boot sector; partition 1: ...
|
||||||
|
sed -E \
|
||||||
sed 's/^# //' | awk '{print $1}' | to_lower |
|
-e 's/[,;#]//g' \
|
||||||
sed -e 's,dos/mbr,raw,' \
|
-e 's/^[[:space:]]*//' \
|
||||||
-e 's,x86,raw,' \
|
-e 's/(POSIX|Unicode|UTF-8|ASCII)//gi' \
|
||||||
-e 's,windows,wim,'
|
-e 's/DOS\/MBR boot sector/raw/i' \
|
||||||
|
-e 's/x86 boot sector/raw/i' \
|
||||||
|
-e 's/Zstandard/zstd/i' \
|
||||||
|
-e 's/Windows imaging \(WIM\) image/wim/i' |
|
||||||
|
awk '{print $1}' | to_lower
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# 不用 file -z,因为
|
||||||
|
# 1. file -z 只能看透一层
|
||||||
|
# 2. alpine file -z 无法看透部分镜像(前1M),例如:
|
||||||
|
# guajibao-win10-ent-ltsc-2021-x64-cn-efi.vhd.gz
|
||||||
|
# guajibao-win7-sp1-ent-x64-cn-efi.vhd.gz
|
||||||
|
# win7-ent-sp1-x64-cn-efi.vhd.gz
|
||||||
|
# 还要注意 centos 7 没有 -Z 只有 -z
|
||||||
file_enhanced() {
|
file_enhanced() {
|
||||||
local file=$1
|
file=$1
|
||||||
local outside inside
|
|
||||||
|
|
||||||
outside=$(file -b $file | fix_file_type)
|
full_type=
|
||||||
|
while true; do
|
||||||
if [ "$outside" = "xz" ] || [ "$outside" = "gzip" ]; then
|
type="$(file -b $file | fix_file_type)"
|
||||||
# 要安装 xz 或者 gzip,不然会报错
|
full_type="$type.$full_type"
|
||||||
# ERROR:[xz: Wait failed, No child process]
|
case "$type" in
|
||||||
install_pkg "$outside"
|
xz | gzip | zstd)
|
||||||
|
install_pkg "$type"
|
||||||
# 加 if 是为了避免以下情况(外面是xz,但是识别不到里面的东西,即使装了xz),
|
$type -dc <"$file" | head -c 1048576 >"$file.inside"
|
||||||
# 即使 file 报错返回值也是 0
|
mv -f "$file.inside" "$file"
|
||||||
# [root@localhost ~]# file -bZ /reinstall-tmp/img-test
|
;;
|
||||||
# ERROR:[xz: Unexpected end of input]
|
tar)
|
||||||
if inside="$(file -bZ $file | fix_file_type)" && ! grep -iq "^Error" <<<"$inside"; then
|
install_pkg "$type"
|
||||||
echo "$inside.$outside"
|
# 隐藏 gzip: unexpected end of file 提醒
|
||||||
return
|
tar xf "$file" -O 2>/dev/null | head -c 1048576 >"$file.inside"
|
||||||
fi
|
mv -f "$file.inside" "$file"
|
||||||
fi
|
;;
|
||||||
echo "$outside"
|
*)
|
||||||
|
break
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
done
|
||||||
|
# shellcheck disable=SC2001
|
||||||
|
echo "$full_type" | sed 's/\.$//'
|
||||||
}
|
}
|
||||||
|
|
||||||
add_community_repo_for_alpine() {
|
add_community_repo_for_alpine() {
|
||||||
@ -433,7 +463,7 @@ is_virt() {
|
|||||||
if [ -z "$_is_virt" ]; then
|
if [ -z "$_is_virt" ]; then
|
||||||
_is_virt=false
|
_is_virt=false
|
||||||
fi
|
fi
|
||||||
echo "vm: $_is_virt"
|
echo "VM: $_is_virt"
|
||||||
fi
|
fi
|
||||||
$_is_virt
|
$_is_virt
|
||||||
}
|
}
|
||||||
@ -1037,7 +1067,7 @@ Continue?
|
|||||||
filename=$(curl -L $mirror | grep -oP "ubuntu-$releasever.*?-live-server-$basearch_alt.iso" | head -1)
|
filename=$(curl -L $mirror | grep -oP "ubuntu-$releasever.*?-live-server-$basearch_alt.iso" | head -1)
|
||||||
iso=$mirror/$filename
|
iso=$mirror/$filename
|
||||||
# 在 ubuntu 20.04 上,file 命令检测 ubuntu 22.04 iso 结果是 DOS/MBR boot sector
|
# 在 ubuntu 20.04 上,file 命令检测 ubuntu 22.04 iso 结果是 DOS/MBR boot sector
|
||||||
test_url $iso 'iso|raw'
|
test_url $iso 'iso raw'
|
||||||
eval ${step}_iso=$iso
|
eval ${step}_iso=$iso
|
||||||
|
|
||||||
# ks
|
# ks
|
||||||
@ -1088,8 +1118,8 @@ Continue?
|
|||||||
else
|
else
|
||||||
# 传统安装
|
# 传统安装
|
||||||
# 该服务器文件缓存 miss 时会响应 206 + Location 头
|
# 该服务器文件缓存 miss 时会响应 206 + Location 头
|
||||||
# 但 curl 这种情况不会重定向,所以添加 ascii 类型让它不要报错
|
# 但 curl 这种情况不会重定向,所以添加 text 类型让它不要报错
|
||||||
test_url $mirror/nixos-$releasever/store-paths.xz 'xz|ascii'
|
test_url $mirror/nixos-$releasever/store-paths.xz 'xz text'
|
||||||
eval ${step}_mirror=$mirror
|
eval ${step}_mirror=$mirror
|
||||||
fi
|
fi
|
||||||
}
|
}
|
||||||
@ -1113,7 +1143,7 @@ Continue?
|
|||||||
dir=releases/$basearch_alt/autobuilds/current-$prefix
|
dir=releases/$basearch_alt/autobuilds/current-$prefix
|
||||||
file=$(curl -L $mirror/$dir/latest-$prefix.txt | grep '.tar.xz' | awk '{print $1}')
|
file=$(curl -L $mirror/$dir/latest-$prefix.txt | grep '.tar.xz' | awk '{print $1}')
|
||||||
stage3=$mirror/$dir/$file
|
stage3=$mirror/$dir/$file
|
||||||
test_url $stage3 'xz'
|
test_url $stage3 'tar.xz'
|
||||||
eval ${step}_img=$stage3
|
eval ${step}_img=$stage3
|
||||||
fi
|
fi
|
||||||
}
|
}
|
||||||
@ -1165,7 +1195,7 @@ Continue?
|
|||||||
# 注意 windows server 2008 r2 serverdatacenter 不用改
|
# 注意 windows server 2008 r2 serverdatacenter 不用改
|
||||||
image_name=${image_name/windows server 2008 server/windows longhorn server}
|
image_name=${image_name/windows server 2008 server/windows longhorn server}
|
||||||
|
|
||||||
test_url "$iso" 'iso|raw'
|
test_url "$iso" 'iso raw'
|
||||||
[ -n "$boot_wim" ] && test_url "$boot_wim" 'wim'
|
[ -n "$boot_wim" ] && test_url "$boot_wim" 'wim'
|
||||||
eval "${step}_iso='$iso'"
|
eval "${step}_iso='$iso'"
|
||||||
eval "${step}_boot_wim='$boot_wim'"
|
eval "${step}_boot_wim='$boot_wim'"
|
||||||
@ -1175,24 +1205,11 @@ Continue?
|
|||||||
# shellcheck disable=SC2154
|
# shellcheck disable=SC2154
|
||||||
setos_dd() {
|
setos_dd() {
|
||||||
# raw 包含 vhd
|
# raw 包含 vhd
|
||||||
test_url $img 'raw|raw.gzip|raw.xz' img_type
|
test_url $img 'raw raw.gzip raw.xz raw.zstd raw.tar.gzip raw.tar.xz raw.tar.zstd' img_type
|
||||||
|
|
||||||
if is_efi; then
|
if is_efi; then
|
||||||
install_pkg hexdump
|
install_pkg hexdump
|
||||||
|
|
||||||
extract() {
|
|
||||||
case "$img_type_warp" in
|
|
||||||
'') cat "$1" ;;
|
|
||||||
xz | gzip)
|
|
||||||
install_pkg $img_type_warp
|
|
||||||
# xz/gzip -d 文件必须有正确的扩展名,否则报扩展名错误
|
|
||||||
# 因此用 stdin
|
|
||||||
"$img_type_warp" -dc <"$1"
|
|
||||||
;;
|
|
||||||
*) error_and_exit "warp type $img_type_warp not support." ;;
|
|
||||||
esac
|
|
||||||
}
|
|
||||||
|
|
||||||
# openwrt 镜像 efi part type 不是 esp
|
# openwrt 镜像 efi part type 不是 esp
|
||||||
# 因此改成检测 fat?
|
# 因此改成检测 fat?
|
||||||
# https://downloads.openwrt.org/releases/23.05.3/targets/x86/64/openwrt-23.05.3-x86-64-generic-ext4-combined-efi.img.gz
|
# https://downloads.openwrt.org/releases/23.05.3/targets/x86/64/openwrt-23.05.3-x86-64-generic-ext4-combined-efi.img.gz
|
||||||
@ -1204,7 +1221,7 @@ Continue?
|
|||||||
|
|
||||||
# 仅打印前34个扇区 * 4096字节(按最大的算)
|
# 仅打印前34个扇区 * 4096字节(按最大的算)
|
||||||
# 每行128字节
|
# 每行128字节
|
||||||
extract "$tmp/img-test" | hexdump -n $((34 * 4096)) -e '128/1 "%02x" "\n"' -v >$tmp/img-test-hex
|
hexdump -n $((34 * 4096)) -e '128/1 "%02x" "\n"' -v "$tmp/img-test" >$tmp/img-test-hex
|
||||||
if grep -q '^28732ac11ff8d211ba4b00a0c93ec93b' $tmp/img-test-hex; then
|
if grep -q '^28732ac11ff8d211ba4b00a0c93ec93b' $tmp/img-test-hex; then
|
||||||
echo 'DD: Image is EFI.'
|
echo 'DD: Image is EFI.'
|
||||||
else
|
else
|
||||||
@ -1392,7 +1409,7 @@ Continue with DD?
|
|||||||
# 集中测试云镜像格式
|
# 集中测试云镜像格式
|
||||||
if is_use_cloud_image && [ "$step" = finalos ]; then
|
if is_use_cloud_image && [ "$step" = finalos ]; then
|
||||||
# shellcheck disable=SC2154
|
# shellcheck disable=SC2154
|
||||||
test_url $finalos_img 'qemu|qemu.gzip|qemu.xz' finalos_img_type
|
test_url $finalos_img 'qemu qemu.gzip qemu.xz qemu.zstd' finalos_img_type
|
||||||
fi
|
fi
|
||||||
}
|
}
|
||||||
|
|
||||||
|
33
trans.sh
33
trans.sh
@ -1736,6 +1736,9 @@ EOF
|
|||||||
# 安装系统
|
# 安装系统
|
||||||
install_$distro
|
install_$distro
|
||||||
|
|
||||||
|
# 安装 arch 有 gpg-agent 进程驻留
|
||||||
|
pkill gpg-agent || true
|
||||||
|
|
||||||
# 初始化
|
# 初始化
|
||||||
if false; then
|
if false; then
|
||||||
# preset-all 后多了很多服务,内存占用多了几十M
|
# preset-all 后多了很多服务,内存占用多了几十M
|
||||||
@ -1825,20 +1828,26 @@ get_http_file_size() {
|
|||||||
pipe_extract() {
|
pipe_extract() {
|
||||||
# alpine busybox 自带 gzip,但官方版也许性能更好
|
# alpine busybox 自带 gzip,但官方版也许性能更好
|
||||||
case "$img_type_warp" in
|
case "$img_type_warp" in
|
||||||
xz | gzip) apk add $img_type_warp ;;
|
xz | gzip | zstd)
|
||||||
'') ;;
|
apk add $img_type_warp
|
||||||
|
"$img_type_warp" -dc
|
||||||
|
;;
|
||||||
|
tar)
|
||||||
|
apk add tar
|
||||||
|
tar x -O
|
||||||
|
;;
|
||||||
|
tar.*)
|
||||||
|
type=$(echo "$img_type_warp" | cut -d. -f2)
|
||||||
|
apk add tar "$type"
|
||||||
|
tar x "--$type" -O
|
||||||
|
;;
|
||||||
|
'') cat ;;
|
||||||
*) error_and_exit "Not supported img_type_warp: $img_type_warp" ;;
|
*) error_and_exit "Not supported img_type_warp: $img_type_warp" ;;
|
||||||
esac
|
esac
|
||||||
|
|
||||||
if [ -n "$img_type_warp" ]; then
|
|
||||||
"$img_type_warp" -dc
|
|
||||||
else
|
|
||||||
cat
|
|
||||||
fi
|
|
||||||
}
|
}
|
||||||
|
|
||||||
dd_gzip_xz_raw() {
|
dd_raw_with_extract() {
|
||||||
info "dd gzip xz raw"
|
info "dd raw"
|
||||||
|
|
||||||
# 用官方 wget,一来带进度条,二来自带重试功能
|
# 用官方 wget,一来带进度条,二来自带重试功能
|
||||||
apk add wget
|
apk add wget
|
||||||
@ -4967,7 +4976,7 @@ trans() {
|
|||||||
;;
|
;;
|
||||||
raw)
|
raw)
|
||||||
# 暂时没用到 raw 格式的云镜像
|
# 暂时没用到 raw 格式的云镜像
|
||||||
dd_gzip_xz_raw
|
dd_raw_with_extract
|
||||||
resize_after_install_cloud_image
|
resize_after_install_cloud_image
|
||||||
modify_os_on_disk linux
|
modify_os_on_disk linux
|
||||||
;;
|
;;
|
||||||
@ -4975,7 +4984,7 @@ trans() {
|
|||||||
elif [ "$distro" = "dd" ]; then
|
elif [ "$distro" = "dd" ]; then
|
||||||
case "$img_type" in
|
case "$img_type" in
|
||||||
raw)
|
raw)
|
||||||
dd_gzip_xz_raw
|
dd_raw_with_extract
|
||||||
if false; then
|
if false; then
|
||||||
# linux 扩容后无法轻易缩小,例如 xfs
|
# linux 扩容后无法轻易缩小,例如 xfs
|
||||||
# windows 扩容在 windows 下完成
|
# windows 扩容在 windows 下完成
|
||||||
|
Loading…
Reference in New Issue
Block a user