2021-10-13 21:11:27 -07:00
|
|
|
#!/bin/bash
|
|
|
|
|
2021-10-16 21:17:55 -07:00
|
|
|
# Fail on any error.
|
|
|
|
set -e
|
|
|
|
|
2022-11-23 22:24:00 -08:00
|
|
|
# Debug mode for diagnosing issues.
|
|
|
|
# Setup first before other operations.
|
|
|
|
debug="${2}"
|
2023-03-23 20:20:24 -07:00
|
|
|
test "${debug}" = "true" && set -x
|
2022-11-23 22:24:00 -08:00
|
|
|
|
2022-03-26 12:42:40 -07:00
|
|
|
# Include library.
|
|
|
|
script_dir="$(dirname -- "$(realpath -- "${0}")")"
|
2022-03-26 12:48:16 -07:00
|
|
|
source "${script_dir}/lib.sh"
|
2022-03-26 12:42:40 -07:00
|
|
|
|
2021-10-13 21:11:27 -07:00
|
|
|
# Directory that holds the cached packages.
|
2022-07-19 20:42:48 -07:00
|
|
|
cache_dir="${1}"
|
2021-10-21 20:57:52 -07:00
|
|
|
|
2021-10-13 21:11:27 -07:00
|
|
|
# List of the packages to use.
|
2022-11-23 22:24:00 -08:00
|
|
|
input_packages="${@:3}"
|
2022-03-16 13:36:54 -07:00
|
|
|
|
2023-03-08 09:04:22 -08:00
|
|
|
if ! apt-fast --version > /dev/null 2>&1; then
|
|
|
|
log "Installing apt-fast for optimized installs..."
|
|
|
|
# Install apt-fast for optimized installs.
|
|
|
|
/bin/bash -c "$(curl -sL https://git.io/vokNn)"
|
|
|
|
log "done"
|
|
|
|
|
|
|
|
log_empty_line
|
|
|
|
fi
|
|
|
|
|
2022-07-19 20:42:48 -07:00
|
|
|
log "Updating APT package list..."
|
2023-02-11 21:36:46 -08:00
|
|
|
if [[ -z "$(find -H /var/lib/apt/lists -maxdepth 0 -mmin -5)" ]]; then
|
2024-10-20 16:55:58 -07:00
|
|
|
apt-fast update > /dev/null
|
2022-11-23 22:24:00 -08:00
|
|
|
log "done"
|
|
|
|
else
|
2023-02-11 21:36:46 -08:00
|
|
|
log "skipped (fresh within at least 5 minutes)"
|
2022-11-23 22:24:00 -08:00
|
|
|
fi
|
2022-07-23 17:06:17 -07:00
|
|
|
|
|
|
|
log_empty_line
|
2021-10-21 12:41:47 -07:00
|
|
|
|
2023-10-11 08:07:11 -07:00
|
|
|
packages="$(get_normalized_package_list "${input_packages}")"
|
|
|
|
package_count=$(wc -w <<< "${packages}")
|
|
|
|
log "Clean installing and caching ${package_count} package(s)."
|
|
|
|
|
|
|
|
log_empty_line
|
|
|
|
|
|
|
|
manifest_main=""
|
|
|
|
log "Package list:"
|
|
|
|
for package in ${packages}; do
|
|
|
|
manifest_main="${manifest_main}${package},"
|
|
|
|
log "- ${package}"
|
|
|
|
done
|
|
|
|
write_manifest "main" "${manifest_main}" "${cache_dir}/manifest_main.log"
|
|
|
|
|
|
|
|
log_empty_line
|
|
|
|
|
2022-07-19 20:42:48 -07:00
|
|
|
# Strictly contains the requested packages.
|
|
|
|
manifest_main=""
|
|
|
|
# Contains all packages including dependencies.
|
|
|
|
manifest_all=""
|
2022-03-16 13:36:54 -07:00
|
|
|
|
2022-08-02 21:14:51 -07:00
|
|
|
install_log_filepath="${cache_dir}/install.log"
|
2022-07-19 20:42:48 -07:00
|
|
|
|
2022-08-02 21:14:51 -07:00
|
|
|
log "Clean installing ${package_count} packages..."
|
|
|
|
# Zero interaction while installing or upgrading the system via apt.
|
2024-10-20 16:55:58 -07:00
|
|
|
DEBIAN_FRONTEND=noninteractive apt-fast --yes install ${packages} > "${install_log_filepath}"
|
2022-07-23 17:06:17 -07:00
|
|
|
log "done"
|
2022-08-02 21:14:51 -07:00
|
|
|
log "Installation log written to ${install_log_filepath}"
|
2022-07-23 17:06:17 -07:00
|
|
|
|
|
|
|
log_empty_line
|
|
|
|
|
2022-08-02 21:14:51 -07:00
|
|
|
installed_packages=$(get_installed_packages "${install_log_filepath}")
|
|
|
|
log "Installed package list:"
|
|
|
|
for installed_package in ${installed_packages}; do
|
2023-03-23 22:19:43 -07:00
|
|
|
# Reformat for human friendly reading.
|
|
|
|
log "- $(echo ${installed_package} | awk -F\= '{print $1" ("$2")"}')"
|
2022-08-02 21:14:51 -07:00
|
|
|
done
|
2022-07-23 17:06:17 -07:00
|
|
|
|
|
|
|
log_empty_line
|
|
|
|
|
2023-03-23 20:20:24 -07:00
|
|
|
installed_packages_count=$(wc -w <<< "${installed_packages}")
|
|
|
|
log "Caching ${installed_packages_count} installed packages..."
|
2022-08-02 21:14:51 -07:00
|
|
|
for installed_package in ${installed_packages}; do
|
2022-11-23 22:24:00 -08:00
|
|
|
cache_filepath="${cache_dir}/${installed_package}.tar"
|
2022-07-23 17:06:17 -07:00
|
|
|
|
2022-08-02 21:14:51 -07:00
|
|
|
# Sanity test in case APT enumerates duplicates.
|
2022-07-23 17:06:17 -07:00
|
|
|
if test ! -f "${cache_filepath}"; then
|
2023-03-23 20:20:24 -07:00
|
|
|
read package_name package_ver < <(get_package_name_ver "${installed_package}")
|
|
|
|
log " * Caching ${package_name} to ${cache_filepath}..."
|
2022-11-23 22:24:00 -08:00
|
|
|
|
|
|
|
# Pipe all package files (no folders) and installation control data to Tar.
|
2023-03-23 20:20:24 -07:00
|
|
|
{ dpkg -L "${package_name}" \
|
|
|
|
& get_install_script_filepath "" "${package_name}" "preinst" \
|
|
|
|
& get_install_script_filepath "" "${package_name}" "postinst"; } |
|
2022-11-23 22:24:00 -08:00
|
|
|
while IFS= read -r f; do test -f "${f}" -o -L "${f}" && get_tar_relpath "${f}"; done |
|
2023-10-30 11:12:50 -07:00
|
|
|
# Single quotes ensure literals like backslash get captured. Use \0 to avoid field separation.
|
|
|
|
awk -F"\0" '{print "\x27"$1"\x27"}' |
|
2024-10-20 16:55:58 -07:00
|
|
|
xargs tar -cf "${cache_filepath}" -C /
|
2022-11-23 22:24:00 -08:00
|
|
|
|
2022-07-23 17:06:17 -07:00
|
|
|
log " done (compressed size $(du -h "${cache_filepath}" | cut -f1))."
|
|
|
|
fi
|
|
|
|
|
|
|
|
# Comma delimited name:ver pairs in the all packages manifest.
|
2023-03-23 22:19:43 -07:00
|
|
|
manifest_all="${manifest_all}${package_name}=${package_ver},"
|
2022-07-23 17:06:17 -07:00
|
|
|
done
|
|
|
|
log "done (total cache size $(du -h ${cache_dir} | tail -1 | awk '{print $1}'))"
|
|
|
|
|
|
|
|
log_empty_line
|
2021-10-13 21:11:27 -07:00
|
|
|
|
2022-07-19 20:42:48 -07:00
|
|
|
write_manifest "all" "${manifest_all}" "${cache_dir}/manifest_all.log"
|