-
Notifications
You must be signed in to change notification settings - Fork 56
/
Copy pathbuild_rootfs.sh
executable file
·134 lines (111 loc) · 4.31 KB
/
build_rootfs.sh
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
#!/bin/bash
#build the debian rootfs
. ./common.sh
print_help() {
echo "Usage: ./build_rootfs.sh rootfs_path release_name"
echo "Valid named arguments (specify with 'key=value'):"
echo " custom_packages - The packages that will be installed in place of task-xfce-desktop."
echo " hostname - The hostname for the new rootfs."
echo " enable_root - Enable the root user."
echo " root_passwd - The root password. This only has an effect if enable_root is set."
echo " username - The unprivileged user name for the new rootfs."
echo " user_passwd - The password for the unprivileged user."
echo " disable_base - Disable the base packages such as zram, cloud-utils, and command-not-found."
echo " arch - The CPU architecture to build the rootfs for."
echo " distro - The Linux distro to use. This should be either 'debian' or 'alpine'."
echo "If you do not specify the hostname and credentials, you will be prompted for them later."
}
assert_root
assert_deps "realpath debootstrap findmnt wget pcregrep tar"
assert_args "$2"
parse_args "$@"
rootfs_dir=$(realpath -m "${1}")
release_name="${2}"
packages="${args['custom_packages']-task-xfce-desktop}"
arch="${args['arch']-amd64}"
distro="${args['distro']-debian}"
chroot_mounts="proc sys dev run"
mkdir -p $rootfs_dir
unmount_all() {
for mountpoint in $chroot_mounts; do
umount -l "$rootfs_dir/$mountpoint"
done
}
need_remount() {
local target="$1"
local mnt_options="$(findmnt -T "$target" | tail -n1 | rev | cut -f1 -d' '| rev)"
echo "$mnt_options" | grep -e "noexec" -e "nodev"
}
do_remount() {
local target="$1"
local mountpoint="$(findmnt -T "$target" | tail -n1 | cut -f1 -d' ')"
mount -o remount,dev,exec "$mountpoint"
}
if [ "$(need_remount "$rootfs_dir")" ]; then
do_remount "$rootfs_dir"
fi
if [ "$distro" = "debian" ]; then
print_info "bootstraping debian chroot"
debootstrap --arch $arch --components=main,contrib,non-free,non-free-firmware "$release_name" "$rootfs_dir" http://deb.debian.org/debian/
chroot_script="/opt/setup_rootfs.sh"
elif [ "$distro" = "ubuntu" ]; then
print_info "bootstraping ubuntu chroot"
repo_url="http://archive.ubuntu.com/ubuntu"
if [ "$arch" = "amd64" ]; then
repo_url="http://archive.ubuntu.com/ubuntu"
else
repo_url="http://ports.ubuntu.com"
fi
debootstrap --arch $arch "$release_name" "$rootfs_dir" "$repo_url"
chroot_script="/opt/setup_rootfs.sh"
elif [ "$distro" = "alpine" ]; then
print_info "downloading alpine package list"
pkg_list_url="https://dl-cdn.alpinelinux.org/alpine/latest-stable/main/x86_64/"
pkg_data="$(wget -qO- --show-progress "$pkg_list_url" | grep "apk-tools-static")"
pkg_url="$pkg_list_url$(echo "$pkg_data" | pcregrep -o1 '"(.+?.apk)"')"
print_info "downloading and extracting apk-tools-static"
pkg_extract_dir="/tmp/apk-tools-static"
pkg_dl_path="$pkg_extract_dir/pkg.apk"
apk_static="$pkg_extract_dir/sbin/apk.static"
mkdir -p "$pkg_extract_dir"
wget -q --show-progress "$pkg_url" -O "$pkg_dl_path"
tar --warning=no-unknown-keyword -xzf "$pkg_dl_path" -C "$pkg_extract_dir"
print_info "bootstraping alpine chroot"
real_arch="x86_64"
if [ "$arch" = "arm64" ]; then
real_arch="aarch64"
fi
$apk_static \
--arch $real_arch \
-X http://dl-cdn.alpinelinux.org/alpine/$release_name/main/ \
-U --allow-untrusted \
--root "$rootfs_dir" \
--initdb add alpine-base
chroot_script="/opt/setup_rootfs_alpine.sh"
else
print_error "'$distro' is an invalid distro choice."
exit 1
fi
print_info "copying rootfs setup scripts"
cp -arv rootfs/* "$rootfs_dir"
cp /etc/resolv.conf "$rootfs_dir/etc/resolv.conf"
print_info "creating bind mounts for chroot"
trap unmount_all EXIT
for mountpoint in $chroot_mounts; do
mount --make-rslave --rbind "/${mountpoint}" "${rootfs_dir}/$mountpoint"
done
hostname="${args['hostname']}"
root_passwd="${args['root_passwd']}"
enable_root="${args['enable_root']}"
username="${args['username']}"
user_passwd="${args['user_passwd']}"
disable_base="${args['disable_base']}"
chroot_command="$chroot_script \
'$DEBUG' '$release_name' '$packages' \
'$hostname' '$root_passwd' '$username' \
'$user_passwd' '$enable_root' '$disable_base' \
'$arch'"
LC_ALL=C chroot $rootfs_dir /bin/sh -c "${chroot_command}"
trap - EXIT
unmount_all
print_info "rootfs has been created"