2018-07-25 19:23:05 +00:00
|
|
|
#!/usr/bin/env bash
|
2018-08-18 18:42:42 +00:00
|
|
|
# Copyright (C) 2018 Luke Shumaker
|
2024-01-04 22:16:23 +00:00
|
|
|
# Copyright (C) 2023-2024 Umorpha Systems
|
2018-08-18 18:42:42 +00:00
|
|
|
# SPDX-License-Identifier: AGPL-3.0-or-later
|
2018-08-06 01:56:44 +00:00
|
|
|
declare -r NAME=osi-mk
|
2023-10-23 22:19:22 +00:00
|
|
|
declare -r VERSION=20231023
|
2018-07-25 19:23:05 +00:00
|
|
|
|
|
|
|
# Why is this different than mkosi[1]?
|
|
|
|
#
|
|
|
|
# - mkosi claims to be "legacy-free"--but they call everything that's
|
2018-08-06 00:09:42 +00:00
|
|
|
# not systemd "legacy"; that clearly won't do for creating OpenRC
|
|
|
|
# images.
|
2018-07-25 19:23:05 +00:00
|
|
|
#
|
|
|
|
# - mkosi claims to be "legacy-free", only supporting GPT
|
|
|
|
# disk-labels--but btrfs can be booted directly, without a separate
|
|
|
|
# disk-label, or a separate ESP partition. To a btrfs disk, GPT/ESP
|
|
|
|
# is legacy.
|
|
|
|
#
|
|
|
|
# - Using a raw btrfs disk means that it can easily be mounted without
|
2023-10-25 11:04:49 +00:00
|
|
|
# first dissecting the disk-label.
|
2018-07-25 19:23:05 +00:00
|
|
|
#
|
|
|
|
# [1]: https://github.com/systemd/mkosi
|
|
|
|
|
2018-08-12 21:53:08 +00:00
|
|
|
set -euE -o pipefail
|
2023-11-10 06:18:15 +00:00
|
|
|
shopt -s expand_aliases
|
|
|
|
|
2024-01-04 22:16:23 +00:00
|
|
|
install_prefix="$(realpath --logical --canonicalize-missing -- "${BASH_SOURCE[0]}/../..")"
|
|
|
|
readonly install_prefix
|
|
|
|
|
|
|
|
source "${install_prefix}/lib/osi.sh"
|
2024-01-30 05:38:29 +00:00
|
|
|
source "${install_prefix}/lib/argparse.sh"
|
2018-08-06 01:49:45 +00:00
|
|
|
|
|
|
|
loaded_modules=()
|
2018-08-06 04:02:37 +00:00
|
|
|
load_module() {
|
2018-08-06 01:49:45 +00:00
|
|
|
local module
|
2018-08-06 04:02:37 +00:00
|
|
|
if ! [[ -f $1 ]]; then
|
2024-01-30 05:38:29 +00:00
|
|
|
argparse:error 'Module does not exist: %s' "$1"
|
|
|
|
return
|
2018-08-06 04:02:37 +00:00
|
|
|
fi
|
2018-08-06 01:49:45 +00:00
|
|
|
module="$(realpath -- "$1")"
|
|
|
|
if in_array "$module" "${loaded_modules[@]}"; then
|
2018-08-06 04:02:37 +00:00
|
|
|
return 0
|
2018-08-06 01:49:45 +00:00
|
|
|
fi
|
2018-08-06 04:02:37 +00:00
|
|
|
loaded_modules+=("$module")
|
2023-10-24 21:42:25 +00:00
|
|
|
# shellcheck source=/dev/null
|
2018-08-06 04:02:37 +00:00
|
|
|
source "$1"
|
2018-08-06 01:49:45 +00:00
|
|
|
}
|
|
|
|
|
2023-11-10 06:18:15 +00:00
|
|
|
beg_indent() {
|
|
|
|
exec 8> >(
|
2024-01-29 20:33:10 +00:00
|
|
|
gprintf -v prefix -- "$@"
|
|
|
|
# shellcheck disable=SC2154 # false-positive, doesn't understand gprintf
|
2024-01-04 22:16:23 +00:00
|
|
|
"${install_prefix}/lib/indent" "$NAME [$prefix] "
|
2023-11-10 06:18:15 +00:00
|
|
|
)
|
|
|
|
_indent_pid=$!
|
|
|
|
}
|
|
|
|
|
|
|
|
alias end_indent=' <&- >&8; exec 8<&-; wait $_indent_pid'
|
|
|
|
|
2018-08-12 21:53:08 +00:00
|
|
|
osi-mk:genfstab() {
|
|
|
|
local arg_mountpoint=$1
|
2023-10-30 21:17:34 +00:00
|
|
|
{
|
|
|
|
# This header mimics the stock fstat from 'filesystem'.
|
|
|
|
echo '# Static information about the filesystems.'
|
|
|
|
echo '# See fstab(5) for details.'
|
|
|
|
echo
|
|
|
|
{
|
|
|
|
# This doesn't quite mimic the header from 'filesystem'
|
|
|
|
#
|
|
|
|
# - I swapped "file system" for "device", for clarity,
|
|
|
|
# and so that it's one word for `column -t`.
|
|
|
|
# - I swapped "pass" for "fsck", for clarity.
|
|
|
|
# - I removed the space after the "#", so it works with
|
|
|
|
# `column -t`.
|
|
|
|
echo '#<device> <mountpoint> <type> <options> <dump> <fsck>'
|
2023-10-30 21:14:27 +00:00
|
|
|
case "${arg_conf[format]}" in
|
|
|
|
erofs)
|
|
|
|
echo "UUID=${arg_conf[fsuuid]}" / erofs defaults 0 1
|
|
|
|
;;
|
|
|
|
*)
|
|
|
|
genfstab -t uuid "$arg_mountpoint" | grep '^[#]' | awk '$3 != "swap"'
|
|
|
|
;;
|
|
|
|
esac
|
2023-10-30 21:17:34 +00:00
|
|
|
} | column -t
|
|
|
|
} >"${arg_mountpoint}/etc/fstab"
|
2018-08-12 21:53:08 +00:00
|
|
|
}
|
|
|
|
|
2018-08-06 01:56:44 +00:00
|
|
|
osi-mk:directories() {
|
2018-08-06 01:49:45 +00:00
|
|
|
local arg_mountpoint=$1
|
|
|
|
local spec outside inside
|
2018-08-06 04:02:37 +00:00
|
|
|
for spec in "${arg_directories[@]}"; do
|
2018-08-06 01:49:45 +00:00
|
|
|
outside="${spec%:*}"
|
|
|
|
inside="${spec#"${outside}:"}"
|
2018-08-13 01:03:56 +00:00
|
|
|
# TODO: maybe rsync would be better?
|
2018-08-06 04:02:37 +00:00
|
|
|
mkdir -p -- "$(dirname -- "${arg_mountpoint}/${inside}")"
|
2018-08-13 05:16:36 +00:00
|
|
|
print 'Copying %q to %q:%q' "$outside" "$arg_file" "$inside"
|
2023-10-29 07:07:28 +00:00
|
|
|
cp -aT --no-preserve=ownership -- "$outside" "${arg_mountpoint}/${inside}"
|
2018-08-06 01:49:45 +00:00
|
|
|
done
|
|
|
|
}
|
|
|
|
|
2023-10-28 22:32:47 +00:00
|
|
|
osi-mk:hostname() {
|
|
|
|
local arg_mountpoint=$1
|
|
|
|
|
|
|
|
echo "${arg_conf[hostname]}" >"$arg_mountpoint/etc/hostname"
|
|
|
|
}
|
|
|
|
|
2023-10-25 17:54:35 +00:00
|
|
|
osi-mk:mkinitcpio() {
|
|
|
|
local arg_mountpoint=$1
|
|
|
|
print 'Configuring mkinitcpio to include all drivers'
|
|
|
|
mkdir -p -- "$arg_mountpoint/etc/mkinitcpio.conf.d"
|
2023-10-31 20:06:27 +00:00
|
|
|
cat >"$arg_mountpoint/etc/mkinitcpio.conf.d/osi-mk.conf" <<'EOF'
|
2023-10-25 17:54:35 +00:00
|
|
|
#!/hint/bash
|
|
|
|
# Copyright (C) 2023 Umorpha Systems
|
|
|
|
# SPDX-License-Identifier: AGPL-3.0-or-later
|
|
|
|
|
|
|
|
# Remove 'autodetect' from HOOKS; include all drivers.
|
|
|
|
for ((i=0; i<${#HOOKS[@]}; i++)); do
|
|
|
|
if [[ ${HOOKS[i]} == autodetect ]]; then
|
2023-10-31 20:06:27 +00:00
|
|
|
HOOKS=("${HOOKS[@]:0:i}" "${HOOKS[@]:i+1}")
|
2023-10-25 17:54:35 +00:00
|
|
|
fi
|
|
|
|
done
|
|
|
|
EOF
|
|
|
|
}
|
|
|
|
|
2018-08-12 21:53:08 +00:00
|
|
|
osi-mk:grub-install() {
|
|
|
|
local arg_mountpoint=$1
|
2018-08-16 05:32:59 +00:00
|
|
|
# shellcheck disable=SC2016
|
2018-08-12 21:53:08 +00:00
|
|
|
arch-chroot -- "$arg_mountpoint" sh -c \
|
2023-10-25 17:54:35 +00:00
|
|
|
'grub-install --target=i386-pc "$(awk '\''$2 == "/" { print $1 }'\'' </proc/mounts)"'
|
2018-08-14 20:53:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
osi-mk:grub-mkconfig() {
|
|
|
|
local arg_mountpoint=$1
|
2018-08-12 21:53:08 +00:00
|
|
|
arch-chroot -- "$arg_mountpoint" grub-mkconfig -o /boot/grub/grub.cfg
|
|
|
|
}
|
|
|
|
|
2023-11-05 07:25:59 +00:00
|
|
|
osi-mk:source-date-epoch() {
|
|
|
|
local arg_mountpoint=$1
|
|
|
|
|
|
|
|
# Backdate package install dates to 1 second before SOURCE_DATE_EPOCH.
|
|
|
|
local file
|
|
|
|
for file in "$arg_mountpoint"/var/lib/pacman/local/*/desc; do
|
|
|
|
awk -v maxdate="$((SOURCE_DATE_EPOCH-1))" '
|
|
|
|
BEGIN{
|
|
|
|
x=0
|
|
|
|
}
|
|
|
|
$0 == "%INSTALLDATE%" {
|
|
|
|
x=2
|
|
|
|
}
|
|
|
|
{
|
|
|
|
if (x == 1 && $0 > maxdate) {
|
|
|
|
print maxdate
|
|
|
|
} else {
|
|
|
|
print
|
|
|
|
}
|
|
|
|
x--
|
|
|
|
}
|
|
|
|
' <"$file" >"$file.tmp"
|
|
|
|
mv -T -- "$file.tmp" "$file"
|
|
|
|
touch --date="@$((SOURCE_DATE_EPOCH-1))" -- "$file"
|
|
|
|
done
|
|
|
|
|
|
|
|
# Backdate file timestamps to SOURCE_DATE_EPOCH.
|
|
|
|
case "${arg_conf[format]}" in
|
|
|
|
raw_btrfs)
|
|
|
|
# Ugg, it's lame that `find` can't find files newer than a
|
|
|
|
# timestamp without using a temporary file.
|
|
|
|
touch --date="@$SOURCE_DATE_EPOCH" -- "$arg_mountpoint/.source_date_epoch"
|
|
|
|
find "$arg_mountpoint" -xdev -depth \
|
|
|
|
-newer "$arg_mountpoint/.source_date_epoch" \
|
|
|
|
-exec touch --no-dereference --date="@$SOURCE_DATE_EPOCH" -- {} +
|
|
|
|
rm -f -- "$arg_mountpoint/.source_date_epoch"
|
|
|
|
touch --date="@$SOURCE_DATE_EPOCH" -- "$arg_mountpoint/"
|
|
|
|
;;
|
|
|
|
erofs)
|
|
|
|
# Nothing to do; mkfs.erofs supports SOURCE_DATE_EPOCH.
|
|
|
|
:
|
|
|
|
;;
|
|
|
|
esac
|
|
|
|
}
|
|
|
|
|
2024-01-30 05:38:29 +00:00
|
|
|
|
2018-07-25 19:23:05 +00:00
|
|
|
main() {
|
2024-01-30 05:38:29 +00:00
|
|
|
# argparse specifications ##############################################
|
|
|
|
#
|
|
|
|
# There got to be enough args with enough validation logic
|
|
|
|
# that the declare/parse/set/validate locations were all far
|
|
|
|
# enough apart that it was hard to maintain.
|
|
|
|
#
|
|
|
|
# So I factored out an argparse.sh library, so that I could
|
|
|
|
# keep the logic for a given argument close to itself.
|
|
|
|
#
|
|
|
|
# Please don't hate me for failing to KISS or remember that
|
|
|
|
# YAGNI; please trust that I did actually "NI".
|
|
|
|
|
|
|
|
opt_specs+=(': --inside : MOUNTPOINT : internal-use only')
|
2018-07-25 19:23:05 +00:00
|
|
|
local arg_mountpoint=
|
2024-01-30 05:38:29 +00:00
|
|
|
opt_visit:inside() { argparse:setmode inside; arg_mountpoint=$1; }
|
|
|
|
opt_final:inside() { :; }
|
|
|
|
|
|
|
|
opt_specs+=('-e : --edit : [BASE.img] : edit an existing image')
|
2018-08-12 02:35:52 +00:00
|
|
|
local arg_edit=false
|
2024-01-30 05:38:29 +00:00
|
|
|
local arg_edit_base=''
|
|
|
|
opt_visit:edit() { arg_edit=true; arg_edit_base=$1; }
|
|
|
|
opt_final:edit() {
|
|
|
|
if $arg_edit; then
|
|
|
|
# We set arg_file below in opt_positional().
|
|
|
|
if ! [[ -f ${arg_edit_base:-$arg_file} ]]; then
|
|
|
|
argparse:error 'Image must already exist to --edit: %s' "${arg_edit_base:-$arg_file}"
|
|
|
|
fi
|
|
|
|
fi
|
|
|
|
}
|
|
|
|
|
|
|
|
opt_specs+=(': --disorderfs : : use disorderfs')
|
2024-01-26 06:35:45 +00:00
|
|
|
local arg_disorderfs=false
|
2024-01-30 05:38:29 +00:00
|
|
|
opt_visit:disorderfs() { arg_disorderfs=true; }
|
|
|
|
opt_final:disorderfs() { :; }
|
|
|
|
|
|
|
|
opt_specs+=('')
|
2018-08-06 01:49:45 +00:00
|
|
|
|
2024-01-30 05:38:29 +00:00
|
|
|
opt_specs+=('-d : --directory : OUTSIDE\:INSIDE : include the given directory')
|
2023-11-08 05:07:03 +00:00
|
|
|
local arg_directories=()
|
2024-01-30 05:38:29 +00:00
|
|
|
opt_visit:directory() {
|
|
|
|
if ! [[ -d "${1%:*}" ]]; then
|
|
|
|
argparse:error 'Directory does not exist: %s' "${1%:*}"
|
|
|
|
return
|
|
|
|
fi
|
|
|
|
arg_directories+=("$1")
|
|
|
|
}
|
|
|
|
opt_final:directory() { :; }
|
|
|
|
|
|
|
|
opt_specs+=('-m : --module : MOD.sh : include the given module')
|
2018-08-06 00:09:42 +00:00
|
|
|
local arg_modules=()
|
2024-01-30 05:38:29 +00:00
|
|
|
opt_visit:module() {
|
|
|
|
if ! [[ -f $1 ]]; then
|
|
|
|
argparse:error 'Module does not exist: %s' "$1"
|
|
|
|
return
|
|
|
|
fi
|
|
|
|
arg_modules+=("$1")
|
|
|
|
}
|
|
|
|
opt_final:module() { :; }
|
|
|
|
|
|
|
|
opt_specs+=('-p : --package : PKGNAME : include the given package (or group)')
|
2018-08-06 01:49:45 +00:00
|
|
|
local arg_packages=()
|
2024-01-30 05:38:29 +00:00
|
|
|
opt_visit:package() { arg_packages+=("$1"); }
|
|
|
|
opt_final:package() { :; }
|
|
|
|
|
|
|
|
opt_specs+=('-P : --package-file : PKG.pkg.tar.xz : include the given package file')
|
2018-08-18 21:50:57 +00:00
|
|
|
local arg_package_files=()
|
2024-01-30 05:38:29 +00:00
|
|
|
opt_visit:package-file() {
|
|
|
|
if ! [[ -f "$1" ]]; then
|
|
|
|
argparse:error 'Package file does not exist: %s' "$1"
|
|
|
|
return
|
|
|
|
fi
|
|
|
|
arg_package_files+=("$1")
|
|
|
|
}
|
|
|
|
opt_final:package-file() { :; }
|
|
|
|
|
|
|
|
opt_specs+=('-C : --conf : KEY=VAL : set set a config option')
|
2023-10-30 21:14:27 +00:00
|
|
|
declare -A arg_conf=(
|
2024-01-26 07:20:35 +00:00
|
|
|
# not package names
|
2023-10-30 21:14:27 +00:00
|
|
|
[format]=raw_btrfs
|
|
|
|
[bootloader]=grub
|
2023-10-31 07:51:56 +00:00
|
|
|
[genfstab]=true
|
2024-01-26 07:20:35 +00:00
|
|
|
|
|
|
|
# package names
|
|
|
|
[initramfs]=mkinitcpio
|
2024-01-19 05:41:46 +00:00
|
|
|
[dbus-units]=dbus-broker-units
|
2023-10-30 21:14:27 +00:00
|
|
|
)
|
2023-11-05 07:25:59 +00:00
|
|
|
if [[ "${SOURCE_DATE_EPOCH:-}" =~ ^([0-9]+|auto)$ ]]; then
|
|
|
|
arg_conf[SOURCE_DATE_EPOCH]="$SOURCE_DATE_EPOCH"
|
|
|
|
arg_orig=(--conf=SOURCE_DATE_EPOCH="${arg_conf[SOURCE_DATE_EPOCH]}" "${arg_orig[@]}")
|
|
|
|
fi
|
2024-01-30 05:38:29 +00:00
|
|
|
opt_visit:conf() {
|
|
|
|
local key val
|
|
|
|
key="${1%%=*}"
|
|
|
|
val="${1#*=}"
|
|
|
|
case "$key" in
|
|
|
|
format)
|
|
|
|
case "$val" in
|
|
|
|
raw_btrfs) :;;
|
|
|
|
erofs) :;;
|
|
|
|
*) argparse:error 'Unrecognized --conf=format= value: %q' "$val"; return;;
|
|
|
|
esac
|
|
|
|
;;
|
|
|
|
bootloader)
|
|
|
|
case "$val" in
|
|
|
|
grub) :;;
|
|
|
|
none) :;;
|
|
|
|
*) argparse:error 'Unrecognized --conf=bootloader= value: %q' "$val"; return;;
|
|
|
|
esac
|
|
|
|
;;
|
|
|
|
genfstab)
|
|
|
|
case "${val,,}" in
|
|
|
|
1|t|true|y|yes) val=true;;
|
|
|
|
0|f|false|n|no) val=false;;
|
|
|
|
*) argparse:error 'Unrecognized --conf=genfstab= value: %q' "$val"; return;;
|
|
|
|
esac
|
|
|
|
;;
|
|
|
|
SOURCE_DATE_EPOCH)
|
|
|
|
if ! [[ "$val" =~ ^([0-9]+|auto|)$ ]]; then
|
|
|
|
argparse:error 'Invalid --conf=SOURCE_DATE_EPOCH= value: %q' "$val"
|
|
|
|
return
|
2018-08-06 00:09:42 +00:00
|
|
|
fi
|
2024-01-30 05:38:29 +00:00
|
|
|
;;
|
|
|
|
esac
|
|
|
|
arg_conf["$key"]="$val";
|
|
|
|
}
|
|
|
|
opt_final:conf() {
|
|
|
|
case "${arg_conf[format]}" in
|
|
|
|
raw_btrfs)
|
|
|
|
if [[ -z ${arg_conf[size]:-} ]]; then
|
|
|
|
argparse:error 'Must specify --conf=size= when creating a new image'
|
|
|
|
fi
|
|
|
|
;;
|
|
|
|
erofs)
|
|
|
|
if $arg_edit; then
|
|
|
|
argparse:error '--conf=format=erofs does not support --edit'
|
|
|
|
fi
|
|
|
|
if [[ ${arg_conf[bootloader]} != none ]]; then
|
|
|
|
argparse:error '--conf=format=erofs requires --conf=bootloader=none'
|
2018-08-16 05:17:54 +00:00
|
|
|
fi
|
2018-08-06 00:09:42 +00:00
|
|
|
;;
|
|
|
|
esac
|
2024-01-30 05:38:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
local arg_file
|
|
|
|
opt_positional() {
|
|
|
|
if (( $# != 1 )); then
|
|
|
|
if (( $# == 0 )); then
|
|
|
|
argparse:error "Expected 1 positional argument, got none"
|
|
|
|
else
|
|
|
|
argparse:error "Expected 1 positional argument, got %d: %s" "$#" "${*@Q}"
|
|
|
|
fi
|
|
|
|
return
|
|
|
|
fi
|
|
|
|
arg_file=$1
|
|
|
|
if [[ "$opt_mode" != inside ]]; then
|
|
|
|
if [[ ( $arg_edit = false || -n $arg_edit_base ) && -e $arg_file ]]; then
|
|
|
|
argparse:error 'Image file already exists, refusing to overwrite: %s' "$arg_file"
|
|
|
|
fi
|
|
|
|
fi
|
|
|
|
}
|
|
|
|
|
|
|
|
# actually parse the args ##############################################
|
|
|
|
|
|
|
|
local arg_orig=("$@")
|
|
|
|
argparse outside "$@"
|
|
|
|
|
|
|
|
# main ################################################################
|
|
|
|
|
|
|
|
case "$opt_mode" in
|
2023-10-25 10:59:35 +00:00
|
|
|
error)
|
|
|
|
print "Try '%q --help' for more information" "${0##*/}" >&2
|
|
|
|
return $EXIT_INVALIDARGUMENT
|
|
|
|
;;
|
2018-07-25 19:23:05 +00:00
|
|
|
version)
|
2023-10-23 22:28:09 +00:00
|
|
|
print "%s (osi-tools) %s" "$NAME" "$VERSION"
|
2023-10-25 10:59:35 +00:00
|
|
|
return $EXIT_SUCCESS
|
2018-07-25 19:23:05 +00:00
|
|
|
;;
|
2024-01-30 05:38:29 +00:00
|
|
|
help)
|
2018-08-12 20:35:56 +00:00
|
|
|
print 'Usage: %s [OPTIONS] FILENAME.img' "${0##*/}"
|
|
|
|
print 'Operating System Image: Make'
|
2018-07-25 19:23:05 +00:00
|
|
|
echo
|
|
|
|
print 'Create a mountable, bootable OS image.'
|
|
|
|
echo
|
|
|
|
print 'OPTIONS:'
|
2024-01-30 05:38:29 +00:00
|
|
|
echo "${opt_flaghelp}" | grep -v -e --inside
|
2023-10-25 10:59:35 +00:00
|
|
|
return $EXIT_SUCCESS
|
2018-07-25 19:23:05 +00:00
|
|
|
;;
|
|
|
|
|
|
|
|
# main code starts here
|
2023-10-25 11:04:49 +00:00
|
|
|
outside) # the part that runs as a normal user without the image mounted
|
2023-10-30 21:14:27 +00:00
|
|
|
mount_dev="$arg_file"
|
|
|
|
mount_opt=''
|
2023-11-10 06:18:15 +00:00
|
|
|
|
2023-11-10 07:43:55 +00:00
|
|
|
beg_indent 'format:before'
|
2018-08-16 06:00:07 +00:00
|
|
|
{
|
2023-10-30 21:14:27 +00:00
|
|
|
case "${arg_conf[format]}" in
|
|
|
|
raw_btrfs)
|
2023-10-30 21:14:27 +00:00
|
|
|
if $arg_edit; then
|
|
|
|
if [[ -n $arg_edit_base ]]; then
|
|
|
|
cp -T --reflink -- "$arg_edit_base" "$arg_file"
|
2023-10-30 21:14:27 +00:00
|
|
|
fi
|
|
|
|
if [[ -n ${arg_conf[fsuuid]:-} ]]; then
|
|
|
|
# Explicit new FSUUID.
|
|
|
|
btrfstune -f -U "${arg_conf[fsuuid]:-}" "$arg_file"
|
|
|
|
elif [[ -n $arg_edit_base ]]; then
|
|
|
|
# Random new FSUUID.
|
2023-10-30 21:14:27 +00:00
|
|
|
btrfstune -fu "$arg_file"
|
|
|
|
fi
|
2023-10-30 21:14:27 +00:00
|
|
|
if [[ -n ${arg_conf[size]:-} ]]; then
|
2023-10-30 21:14:27 +00:00
|
|
|
# Calculate sizes in exact bytes now, to avoid any discrepancies
|
|
|
|
# between truncate(1) and btrfs-filesystem(8).
|
|
|
|
# Use truncate(1) to do this. It's a little gross,
|
|
|
|
# but because TMPDIR is likely on tmpfs, it's time-cheap,
|
|
|
|
# and because of sparse files, it's space-cheap.
|
|
|
|
tmpfile=$(mktemp -t -- "${0##*/}.XXXXXXXXXX")
|
|
|
|
trap "rm -- ${tmpfile@Q}" EXIT
|
2023-10-30 21:14:27 +00:00
|
|
|
case "${arg_conf[size]}" in
|
2023-10-30 21:14:27 +00:00
|
|
|
'+'*|'-'*|'<'*|'>'*|'/'*|'%'*)
|
2023-10-30 21:14:27 +00:00
|
|
|
truncate --reference="$arg_file" --size="${arg_conf[size]}" -- "$tmpfile";;
|
2023-10-30 21:14:27 +00:00
|
|
|
*)
|
2023-10-30 21:14:27 +00:00
|
|
|
truncate --size="${arg_conf[size]}" -- "$tmpfile";;
|
2023-10-30 21:14:27 +00:00
|
|
|
esac
|
|
|
|
old_size=$(stat --format='%s' -- "$arg_file")
|
|
|
|
new_size=$(stat --format='%s' -- "$tmpfile")
|
|
|
|
rm -- "$tmpfile"
|
|
|
|
trap - EXIT
|
2018-08-16 05:19:05 +00:00
|
|
|
|
2023-10-30 21:14:27 +00:00
|
|
|
# Do the resize
|
|
|
|
arg_mountpoint=$(mktemp -dt -- "${0##*/}.XXXXXXXXXX")
|
|
|
|
trap "rmdir -- ${arg_mountpoint@Q}" EXIT
|
|
|
|
if (( new_size > old_size )); then
|
|
|
|
truncate --size="$new_size" -- "$arg_file"
|
2024-01-26 01:43:54 +00:00
|
|
|
_sudo -- "${install_prefix}/bin/osi-mount" --root -- "$arg_file" "$arg_mountpoint" btrfs filesystem resize max "$arg_mountpoint"
|
2023-10-30 21:14:27 +00:00
|
|
|
elif (( new_size < old_size )); then
|
2024-01-26 01:43:54 +00:00
|
|
|
_sudo -- "${install_prefix}/bin/osi-mount" --root -- "$arg_file" "$arg_mountpoint" btrfs filesystem resize "$new_size" "$arg_mountpoint"
|
2023-10-30 21:14:27 +00:00
|
|
|
truncate --size="$new_size" -- "$arg_file"
|
|
|
|
fi
|
|
|
|
rmdir -- "$arg_mountpoint"
|
|
|
|
trap - EXIT
|
|
|
|
fi
|
|
|
|
else
|
2023-10-30 21:14:27 +00:00
|
|
|
truncate --size="${arg_conf[size]}" -- "$arg_file"
|
|
|
|
mkfs.btrfs ${arg_conf[fsuuid]:+"--uuid=${arg_conf[fsuuid]}"} -- "$arg_file"
|
2018-08-16 06:00:07 +00:00
|
|
|
fi
|
2023-10-30 21:14:27 +00:00
|
|
|
;;
|
2023-10-30 21:14:27 +00:00
|
|
|
erofs)
|
|
|
|
if $arg_edit; then
|
|
|
|
error 2 '--conf=format=erofs does not support --edit'
|
|
|
|
fi
|
|
|
|
if [[ -z "${arg_conf[fsuuid]:-}" ]]; then
|
|
|
|
arg_conf[fsuuid]=$(uuidgen)
|
|
|
|
arg_orig=(--conf=fsuuid="${arg_conf[fsuuid]}" "${arg_orig[@]}")
|
|
|
|
fi
|
|
|
|
|
|
|
|
mount_dev=$(mktemp -dt -- "${0##*/}.XXXXXXXXXX")
|
|
|
|
# Make our own tmpfs because $TMPDIR might be mounted
|
|
|
|
# with 'noexec' or something. Also, so we can clean it
|
|
|
|
# up without a `sudo rm -rf`.
|
2024-01-26 01:43:54 +00:00
|
|
|
_sudo mount -t tmpfs tmpfs:osi-mk "$mount_dev"
|
|
|
|
trap "_sudo umount ${mount_dev@Q}; rmdir -- ${mount_dev@Q}" EXIT
|
2023-10-30 21:14:27 +00:00
|
|
|
mount_opt=bind
|
|
|
|
;;
|
2023-10-30 21:14:27 +00:00
|
|
|
esac
|
2023-11-10 06:18:15 +00:00
|
|
|
} end_indent
|
2023-10-30 21:14:27 +00:00
|
|
|
|
2018-07-25 19:23:05 +00:00
|
|
|
arg_mountpoint=$(mktemp -dt -- "${0##*/}.XXXXXXXXXX")
|
2023-10-30 21:14:27 +00:00
|
|
|
r=0
|
2024-01-26 01:43:54 +00:00
|
|
|
_sudo -- "${install_prefix}/bin/osi-mount" --root \
|
2024-01-26 06:35:45 +00:00
|
|
|
--rwdir="${TMPDIR:-/tmp}" \
|
2024-01-26 01:43:54 +00:00
|
|
|
--rwdir=/var/cache/pacman/pkg \
|
|
|
|
--rwdir=/etc/pacman.d/gnupg \
|
|
|
|
--options="$mount_opt" \
|
|
|
|
-- \
|
|
|
|
"$mount_dev" "$arg_mountpoint" \
|
|
|
|
"${BASH_SOURCE[0]}" --inside="$arg_mountpoint" \
|
|
|
|
"${arg_orig[@]}" || r=$?
|
2023-10-30 21:14:27 +00:00
|
|
|
rmdir -- "$arg_mountpoint"
|
|
|
|
if (( r != 0 )); then
|
|
|
|
return $r
|
|
|
|
fi
|
|
|
|
|
2023-11-10 07:43:55 +00:00
|
|
|
beg_indent 'format:after'
|
2023-11-04 06:55:45 +00:00
|
|
|
{
|
|
|
|
case "${arg_conf[format]}" in
|
|
|
|
erofs)
|
2023-11-05 07:25:59 +00:00
|
|
|
if [[ "${arg_conf[SOURCE_DATE_EPOCH]:-}" == auto ]]; then
|
|
|
|
local file
|
|
|
|
for file in "$mount_dev"/var/lib/pacman/local/*/desc; do
|
|
|
|
arg_conf[SOURCE_DATE_EPOCH]="$(sed -n '/^%INSTALLDATE%$/{n;p;}' -- "$file")"
|
|
|
|
break
|
|
|
|
done
|
|
|
|
fi
|
|
|
|
if [[ -n "${arg_conf[SOURCE_DATE_EPOCH]:-}" ]]; then
|
|
|
|
export SOURCE_DATE_EPOCH="${arg_conf[SOURCE_DATE_EPOCH]}"
|
|
|
|
fi
|
2023-11-04 06:55:45 +00:00
|
|
|
tmpdir=$(mktemp -dt -- "${0##*/}.XXXXXXXXXX")
|
|
|
|
trap "sudo umount ${mount_dev@Q}; rmdir -- ${mount_dev@Q}; rm -f -- ${tmpdir@Q}/erofs.img; rmdir -- ${tmpdir@Q}" EXIT
|
2024-01-26 01:43:54 +00:00
|
|
|
_sudo mkfs.erofs \
|
|
|
|
-U"${arg_conf[fsuuid]}" \
|
|
|
|
${arg_conf[erofs_compression]:+"-z${arg_conf[erofs_compression]}"} \
|
|
|
|
"$tmpdir/erofs.img" "$mount_dev"
|
2023-11-04 06:55:45 +00:00
|
|
|
cp -T "$tmpdir/erofs.img" "$arg_file"
|
|
|
|
;;
|
|
|
|
esac
|
2023-11-10 06:18:15 +00:00
|
|
|
} end_indent
|
2023-11-04 06:55:45 +00:00
|
|
|
|
|
|
|
print '%s Done' "$NAME"
|
2018-07-25 19:23:05 +00:00
|
|
|
;;
|
2023-10-25 11:04:49 +00:00
|
|
|
inside) # the part that runs as root with the image mounted
|
2018-08-11 18:57:24 +00:00
|
|
|
needs_sudo
|
2018-07-26 20:35:32 +00:00
|
|
|
|
2018-08-06 01:49:45 +00:00
|
|
|
### Load modules ###
|
2023-10-27 18:27:27 +00:00
|
|
|
packages=("${arg_packages[@]}")
|
2023-10-25 18:09:28 +00:00
|
|
|
pre_install=()
|
2018-08-12 21:53:08 +00:00
|
|
|
post_install=(
|
|
|
|
50:osi-mk:directories
|
|
|
|
)
|
2023-10-30 21:14:27 +00:00
|
|
|
|
2024-01-30 05:38:29 +00:00
|
|
|
if "${arg_conf[genfstab]}"; then
|
|
|
|
if ! $arg_edit || [[ -n $arg_edit_base ]] || [[ -n "${arg_conf[fsuuid]:-}" ]]; then
|
|
|
|
post_install+=(10:osi-mk:genfstab)
|
|
|
|
fi
|
|
|
|
fi
|
2023-10-30 21:14:27 +00:00
|
|
|
case "${arg_conf[format]}" in
|
|
|
|
raw_btrfs)
|
|
|
|
packages+=(btrfs-progs)
|
|
|
|
;;
|
|
|
|
erofs)
|
|
|
|
packages+=(erofs-utils)
|
|
|
|
;;
|
|
|
|
esac
|
|
|
|
case "${arg_conf[bootloader]}" in
|
|
|
|
grub)
|
|
|
|
packages+=(grub)
|
2023-11-05 07:25:59 +00:00
|
|
|
post_install+=(88:osi-mk:grub-mkconfig) # before 89:osi-mk:source-date-epoch
|
2023-10-30 21:14:27 +00:00
|
|
|
if ! $arg_edit; then
|
2023-11-05 07:25:59 +00:00
|
|
|
post_install+=(87:osi-mk:grub-install) # before '88:osi-mk:grub-mkconfig'
|
2023-10-30 21:14:27 +00:00
|
|
|
fi
|
|
|
|
;;
|
|
|
|
esac
|
|
|
|
case "${arg_conf[initramfs]:-}" in
|
|
|
|
mkinitcpio)
|
2023-11-04 07:06:55 +00:00
|
|
|
pre_install+=(10:osi-mk:mkinitcpio)
|
2023-10-30 21:14:27 +00:00
|
|
|
;;
|
|
|
|
esac
|
2023-10-28 22:32:47 +00:00
|
|
|
if [[ -n "${arg_conf[hostname]:-}" ]]; then
|
2023-11-04 07:06:55 +00:00
|
|
|
post_install+=(10:osi-mk:hostname)
|
2023-10-28 22:32:47 +00:00
|
|
|
fi
|
2023-11-05 07:25:59 +00:00
|
|
|
if [[ -n "${arg_conf[SOURCE_DATE_EPOCH]:-}" ]]; then
|
|
|
|
post_install+=(89:osi-mk:source-date-epoch)
|
|
|
|
fi
|
2023-10-30 21:14:27 +00:00
|
|
|
|
2023-12-13 02:58:08 +00:00
|
|
|
local pkgfile pkgdeps
|
|
|
|
for pkgfile in "${arg_package_files[@]}"; do
|
|
|
|
mapfile -t pkgdeps < <(bsdtar xfO "$pkgfile" .PKGINFO | sed -n 's/^depend = //p')
|
|
|
|
cache_packages+=("${pkgdeps[@]}")
|
|
|
|
done
|
|
|
|
|
2018-08-06 00:09:42 +00:00
|
|
|
for module in "${arg_modules[@]}"; do
|
2018-08-06 04:02:37 +00:00
|
|
|
load_module "$module"
|
2018-08-06 00:09:42 +00:00
|
|
|
done
|
2018-08-06 04:38:14 +00:00
|
|
|
cache_packages+=("${packages[@]}")
|
2018-08-06 01:49:45 +00:00
|
|
|
|
2024-01-26 06:35:45 +00:00
|
|
|
### Disorderfs ###
|
|
|
|
if $arg_disorderfs; then
|
|
|
|
orig_arg_mountpoint=$arg_mountpoint
|
|
|
|
arg_mountpoint=$(mktemp -dt -- "${0##*/}.XXXXXXXXXX")
|
|
|
|
disorderfs --multi-user=yes --shuffle-dirents=yes "$orig_arg_mountpoint" "$arg_mountpoint"
|
|
|
|
trap "fusermount -u ${arg_mountpoint@Q}; rmdir ${arg_mountpoint@Q}" EXIT
|
|
|
|
fi
|
|
|
|
|
2024-01-29 20:16:30 +00:00
|
|
|
pacman_conf=/usr/share/pacman/defaults/pacman.conf.x86_64
|
|
|
|
|
2023-10-23 22:52:38 +00:00
|
|
|
### Download ###
|
2023-11-10 07:43:55 +00:00
|
|
|
beg_indent 'download:repos'
|
2018-08-12 21:53:08 +00:00
|
|
|
{
|
2023-10-25 11:17:25 +00:00
|
|
|
# Download syncdbs to the image
|
|
|
|
mkdir -p -- "$arg_mountpoint"/var/{lib/pacman,log}
|
2024-01-29 20:16:30 +00:00
|
|
|
pacman -r "$arg_mountpoint" --config="$pacman_conf" \
|
2023-10-25 11:17:25 +00:00
|
|
|
-Sy --noconfirm
|
2023-11-10 06:18:15 +00:00
|
|
|
} end_indent
|
2023-11-10 07:43:55 +00:00
|
|
|
beg_indent 'download:check-config'
|
2023-10-25 11:33:05 +00:00
|
|
|
{
|
|
|
|
# Validate that the user specified an answer to
|
|
|
|
# every question that pacman asks.
|
|
|
|
opt_fail=false
|
|
|
|
while read -r -a options; do
|
|
|
|
name="${options[0]%:}"
|
|
|
|
options=("${options[@]:1}")
|
|
|
|
|
|
|
|
base=${name%%[<>=]*}
|
|
|
|
if in_array "${arg_conf["$base"]:-}" "${options[@]}" "${options[@]#*/}"; then
|
2023-10-27 18:27:27 +00:00
|
|
|
print 'inserting package %s=%s' "$name" "${arg_conf["$base"]}"
|
|
|
|
packages+=("${arg_conf["$base"]}")
|
|
|
|
cache_packages+=("${arg_conf["$base"]}")
|
2023-10-25 11:33:05 +00:00
|
|
|
else
|
|
|
|
error 0 "must set option '%q' to one of [%s]" "$base" "${options[*]}"
|
|
|
|
opt_fail=true
|
|
|
|
fi
|
2024-01-04 22:16:23 +00:00
|
|
|
done < <("${install_prefix}/lib/pacman-choices" \
|
2024-01-29 20:16:30 +00:00
|
|
|
-r "$arg_mountpoint" --config="$pacman_conf" \
|
2023-10-28 20:19:45 +00:00
|
|
|
--oneline -- "${cache_packages[@]}")
|
2023-10-25 11:33:05 +00:00
|
|
|
if [[ $opt_fail == true ]]; then
|
|
|
|
exit $EXIT_NOTCONFIGURED
|
|
|
|
fi
|
2023-11-10 06:18:15 +00:00
|
|
|
} end_indent
|
2023-11-10 07:43:55 +00:00
|
|
|
beg_indent 'download:packages'
|
2023-10-25 11:17:25 +00:00
|
|
|
{
|
|
|
|
if (( ${#cache_packages[@]} > 0 )); then # this check is important for --edit
|
|
|
|
# Download needed packages to the host cache
|
2024-01-29 20:16:30 +00:00
|
|
|
pacman -r "$arg_mountpoint" --config="$pacman_conf" \
|
2018-08-12 22:20:59 +00:00
|
|
|
-Syw --noconfirm -- "${cache_packages[@]}"
|
2023-10-25 11:17:25 +00:00
|
|
|
|
2018-08-12 22:25:10 +00:00
|
|
|
# Copy needed packages from host cache to image cache
|
2023-10-25 11:17:25 +00:00
|
|
|
mkdir -p -- "$arg_mountpoint"/var/cache/pacman/pkg
|
2024-01-29 20:16:30 +00:00
|
|
|
pacman -r "$arg_mountpoint" --config="$pacman_conf" \
|
2018-08-12 22:20:59 +00:00
|
|
|
-Sp --print-format='%l' -- "${cache_packages[@]}" \
|
2023-10-25 11:17:25 +00:00
|
|
|
| sed -En 's,^file://(.*),\1\n\1.sig,p' \
|
|
|
|
| xargs -d $'\n' -r cp -t "$arg_mountpoint/var/cache/pacman/pkg" --
|
2018-08-12 22:20:59 +00:00
|
|
|
fi
|
2023-11-10 06:18:15 +00:00
|
|
|
} end_indent
|
2023-10-23 22:52:38 +00:00
|
|
|
|
2023-11-05 07:25:59 +00:00
|
|
|
if [[ "${arg_conf[SOURCE_DATE_EPOCH]:-}" =~ ^[0-9]+$ ]]; then
|
|
|
|
print '%s SOURCE_DATE_EPOCH=%s' "$NAME" "${arg_conf[SOURCE_DATE_EPOCH]}"
|
|
|
|
export SOURCE_DATE_EPOCH="${arg_conf[SOURCE_DATE_EPOCH]}"
|
|
|
|
fi
|
|
|
|
|
2023-10-25 18:09:28 +00:00
|
|
|
### pre_install ###
|
|
|
|
while IFS=: read -r n fn; do
|
2023-11-10 07:43:55 +00:00
|
|
|
beg_indent 'pre_install:%s:%s' "$n" "$fn"
|
2023-11-10 06:18:15 +00:00
|
|
|
(
|
2023-10-25 18:09:28 +00:00
|
|
|
print Begin
|
|
|
|
"$fn" "$arg_mountpoint"
|
|
|
|
print End
|
2023-11-10 06:18:15 +00:00
|
|
|
) end_indent
|
2024-01-19 05:36:20 +00:00
|
|
|
done < <([[ "${#pre_install[@]}" == 0 ]] || printf '%s\n' "${pre_install[@]}" | sort)
|
2023-10-25 18:09:28 +00:00
|
|
|
|
2023-10-23 22:52:38 +00:00
|
|
|
### Install ###
|
2023-11-10 07:43:55 +00:00
|
|
|
beg_indent 'install'
|
2023-10-23 22:52:38 +00:00
|
|
|
{
|
2023-10-25 11:17:25 +00:00
|
|
|
if (( ${#packages[@]} > 0 )); then # this check is important for --edit
|
2024-01-29 20:16:30 +00:00
|
|
|
local pacstrap_args=(
|
|
|
|
-M # don't copy the host's mirrorlist
|
|
|
|
-C "$pacman_conf"
|
|
|
|
--
|
|
|
|
"$arg_mountpoint"
|
|
|
|
--hookdir="$arg_mountpoint/etc/pacman.d/hooks" # hack around https://bugs.archlinux.org/task/49347
|
|
|
|
--needed # for --edit
|
2024-01-26 07:20:35 +00:00
|
|
|
"${packages[@]}"
|
2024-01-29 20:16:30 +00:00
|
|
|
)
|
|
|
|
TMPDIR='' pacstrap "${pacstrap_args[@]}"
|
2018-08-12 22:20:59 +00:00
|
|
|
fi
|
2018-08-18 21:50:57 +00:00
|
|
|
if (( ${#arg_package_files[@]} > 0 )); then
|
2023-11-21 17:43:29 +00:00
|
|
|
dir="$(mktemp -d -- "$arg_mountpoint/var/tmp/package-files.XXXXXXXXXX")"
|
2018-08-18 21:50:57 +00:00
|
|
|
trap "rm -rf -- ${dir@Q}" EXIT
|
|
|
|
cp -t "$dir" -- "${arg_package_files[@]}"
|
2023-11-21 17:43:29 +00:00
|
|
|
local arg filelist=()
|
|
|
|
for arg in "${arg_package_files[@]}"; do
|
|
|
|
filelist+=("/var/tmp/${dir##*/var/tmp/}/${arg##*/}")
|
|
|
|
done
|
|
|
|
arch-chroot -- "$arg_mountpoint" pacman -U --noconfirm -- "${filelist[@]}"
|
2018-08-18 21:50:57 +00:00
|
|
|
rm -rf -- "$dir"
|
|
|
|
trap - EXIT
|
|
|
|
fi
|
2023-11-10 06:18:15 +00:00
|
|
|
} end_indent
|
2018-07-26 20:35:32 +00:00
|
|
|
|
2023-11-05 07:25:59 +00:00
|
|
|
if [[ "${arg_conf[SOURCE_DATE_EPOCH]:-}" == auto ]]; then
|
|
|
|
# set it to 1 second after the last timestamp from any
|
|
|
|
# package (that extra second is so that post-install
|
|
|
|
# things can be timestamped after the package files).
|
|
|
|
arg_conf[SOURCE_DATE_EPOCH]=$(($({
|
|
|
|
zcat "$arg_mountpoint"/var/lib/pacman/local/*/mtree | grep -o 'time=[0-9]*' | cut -d= -f2
|
|
|
|
cat /var/lib/pacman/local/*/desc | sed -n '/^%BUILDDATE%$/{n;p;}'
|
|
|
|
} | sort -n | tail -n1) + 1))
|
|
|
|
fi
|
|
|
|
if [[ -n "${arg_conf[SOURCE_DATE_EPOCH]:-}" ]]; then
|
|
|
|
print '%s SOURCE_DATE_EPOCH=%s' "$NAME" "${arg_conf[SOURCE_DATE_EPOCH]}"
|
|
|
|
export SOURCE_DATE_EPOCH="${arg_conf[SOURCE_DATE_EPOCH]}"
|
|
|
|
fi
|
|
|
|
|
2018-08-06 01:49:45 +00:00
|
|
|
### post_install ###
|
|
|
|
while IFS=: read -r n fn; do
|
2023-11-10 07:43:55 +00:00
|
|
|
beg_indent 'post_install:%s:%s' "$n" "$fn"
|
2023-11-10 06:18:15 +00:00
|
|
|
(
|
2018-08-12 22:05:29 +00:00
|
|
|
print Begin
|
2018-08-12 21:53:08 +00:00
|
|
|
"$fn" "$arg_mountpoint"
|
2018-08-12 22:05:29 +00:00
|
|
|
print End
|
2023-11-10 06:18:15 +00:00
|
|
|
) end_indent
|
2024-01-19 05:36:20 +00:00
|
|
|
done < <([[ "${#post_install[@]}" == 0 ]] || printf '%s\n' "${post_install[@]}" | sort)
|
2018-07-25 19:23:05 +00:00
|
|
|
;;
|
|
|
|
|
2023-10-25 10:59:35 +00:00
|
|
|
*) error $EXIT_FAILURE 'Internal error. The programmer writing this tool screwed up.';;
|
2018-07-25 19:23:05 +00:00
|
|
|
esac
|
|
|
|
}
|
|
|
|
|
|
|
|
main "$@"
|