feather/contrib/guix/libexec/build.sh

483 lines
18 KiB
Bash
Raw Normal View History

2022-06-02 12:50:33 +00:00
#!/usr/bin/env bash
# Copyright (c) 2019-2021 The Bitcoin Core developers
# Copyright (c) 2022-2022 The Monero Project
# Distributed under the MIT software license, see the accompanying
# file ../LICENSE.txt or http://www.opensource.org/licenses/mit-license.php.
export LC_ALL=C
set -e -o pipefail
export TZ=UTC
2023-03-18 11:25:48 +00:00
export DEBUG_GENID=1
export HOSTNAME=host
2022-06-02 12:50:33 +00:00
2023-01-05 18:35:05 +00:00
# shellcheck source=contrib/shell/git-utils.bash
source contrib/shell/git-utils.bash
2022-06-02 12:50:33 +00:00
# Although Guix _does_ set umask when building its own packages (in our case,
# this is all packages in manifest.scm), it does not set it for `guix
# environment`. It does make sense for at least `guix environment --container`
# to set umask, so if that change gets merged upstream and we bump the
# time-machine to a commit which includes the aforementioned change, we can
# remove this line.
#
# This line should be placed before any commands which creates files.
umask 0022
if [ -n "$V" ]; then
# Print both unexpanded (-v) and expanded (-x) forms of commands as they are
# read from this file.
set -vx
# Set VERBOSE for CMake-based builds
export VERBOSE="$V"
fi
# Check that required environment variables are set
cat << EOF
Required environment variables as seen inside the container:
DIST_ARCHIVE_BASE: ${DIST_ARCHIVE_BASE:?not set}
DISTNAME: ${DISTNAME:?not set}
HOST: ${HOST:?not set}
COMMIT_TIMESTAMP: ${COMMIT_TIMESTAMP:?not set}
2022-06-02 12:50:33 +00:00
JOBS: ${JOBS:?not set}
DISTSRC: ${DISTSRC:?not set}
OUTDIR: ${OUTDIR:?not set}
2024-03-13 10:44:29 +00:00
LOGDIR: ${LOGDIR:?not set}
2022-12-21 15:15:22 +00:00
OPTIONS: ${OPTIONS}
2022-06-02 12:50:33 +00:00
EOF
ACTUAL_OUTDIR="${OUTDIR}"
OUTDIR="${DISTSRC}/output"
# Use a fixed timestamp for depends builds so hashes match across commits that don't make changes to the build system
export SOURCE_DATE_EPOCH=1397818193
2022-06-02 12:50:33 +00:00
#####################
# Environment Setup #
#####################
2024-03-13 10:44:29 +00:00
# Collect some information about the build environment to help debug potential reproducibility issues
mkdir -p "${LOGDIR}"
ls -1 /gnu/store | sort > ${LOGDIR}/guix-hashes.txt
printenv | sort | grep -v '^\(BASE_CACHE=\|DISTNAME=\|DISTSRC=\|OUTDIR=\|LOGDIR=\|SOURCES_PATH=\|JOBS=\|OPTIONS=\|DEPENDS_ONLY=\)' > ${LOGDIR}/guix-env.txt
2022-06-02 12:50:33 +00:00
# The depends folder also serves as a base-prefix for depends packages for
# $HOSTs after successfully building.
BASEPREFIX="${PWD}/contrib/depends"
# Given a package name and an output name, return the path of that output in our
# current guix environment
store_path() {
grep --extended-regexp "/[^-]{32}-${1}-[^-]+${2:+-${2}}" "${GUIX_ENVIRONMENT}/manifest" \
| head --lines=1 \
| sed --expression='s|\x29*$||' \
--expression='s|^[[:space:]]*"||' \
2022-06-02 12:50:33 +00:00
--expression='s|"[[:space:]]*$||'
}
# Set environment variables to point the NATIVE toolchain to the right
# includes/libs
NATIVE_GCC="$(store_path gcc-toolchain)"
NATIVE_GCC_STATIC="$(store_path gcc-toolchain static)"
QT_LIBS="$(store_path xcb-util):$(store_path xcb-util-renderutil):$(store_path xcb-util-keysyms):$(store_path xcb-util-cursor):$(store_path xcb-util-image):$(store_path xcb-util-wm):"
QT_LIBS_LIBS="$(echo "${QT_LIBS}" | sed 's/:/\/lib:/g')"
2022-06-02 12:50:33 +00:00
unset LIBRARY_PATH
unset CPATH
unset C_INCLUDE_PATH
unset CPLUS_INCLUDE_PATH
unset OBJC_INCLUDE_PATH
unset OBJCPLUS_INCLUDE_PATH
export LIBRARY_PATH="${NATIVE_GCC}/lib:${NATIVE_GCC}/lib64:${NATIVE_GCC_STATIC}/lib:${NATIVE_GCC_STATIC}/lib64"
export C_INCLUDE_PATH="${NATIVE_GCC}/include"
export CPLUS_INCLUDE_PATH="${NATIVE_GCC}/include/c++:${NATIVE_GCC}/include"
export OBJC_INCLUDE_PATH="${NATIVE_GCC}/include"
export OBJCPLUS_INCLUDE_PATH="${NATIVE_GCC}/include/c++:${NATIVE_GCC}/include"
2023-03-29 13:17:12 +00:00
export QT_LIBS
export QT_LIBS_LIBS
2022-06-02 12:50:33 +00:00
prepend_to_search_env_var() {
export "${1}=${2}${!1:+:}${!1}"
}
# Set environment variables to point the CROSS toolchain to the right
# includes/libs for $HOST
case "$HOST" in
*mingw*)
# Determine output paths to use in CROSS_* environment variables
CROSS_GLIBC="$(store_path "mingw-w64-x86_64-winpthreads")"
CROSS_GCC="$(store_path "gcc-cross-${HOST}")"
CROSS_GCC_LIB_STORE="$(store_path "gcc-cross-${HOST}" lib)"
CROSS_GCC_LIBS=( "${CROSS_GCC_LIB_STORE}/lib/gcc/${HOST}"/* ) # This expands to an array of directories...
CROSS_GCC_LIB="${CROSS_GCC_LIBS[0]}" # ...we just want the first one (there should only be one)
# The search path ordering is generally:
# 1. gcc-related search paths
# 2. libc-related search paths
# 2. kernel-header-related search paths (not applicable to mingw-w64 hosts)
export CROSS_C_INCLUDE_PATH="${CROSS_GCC_LIB}/include:${CROSS_GCC_LIB}/include-fixed:${CROSS_GLIBC}/include"
export CROSS_CPLUS_INCLUDE_PATH="${CROSS_GCC}/include/c++:${CROSS_GCC}/include/c++/${HOST}:${CROSS_GCC}/include/c++/backward:${CROSS_C_INCLUDE_PATH}"
export CROSS_LIBRARY_PATH="${CROSS_GCC_LIB_STORE}/lib:${CROSS_GCC_LIB}:${CROSS_GLIBC}/lib"
2023-05-28 09:36:16 +00:00
WMF_LIBS="$(store_path "mingw-w64-x86_64-winpthreads")"
export WMF_LIBS
2022-06-02 12:50:33 +00:00
;;
*darwin*)
# The CROSS toolchain for darwin uses the SDK and ignores environment variables.
# See depends/hosts/darwin.mk for more details.
;;
*linux*)
CROSS_GLIBC="$(store_path "glibc-cross-${HOST}")"
CROSS_GLIBC_STATIC="$(store_path "glibc-cross-${HOST}" static)"
CROSS_KERNEL="$(store_path "linux-libre-headers-cross-${HOST}")"
CROSS_GCC="$(store_path "gcc-cross-${HOST}")"
CROSS_GCC_LIB_STORE="$(store_path "gcc-cross-${HOST}" lib)"
CROSS_GCC_LIBS=( "${CROSS_GCC_LIB_STORE}/lib/gcc/${HOST}"/* ) # This expands to an array of directories...
CROSS_GCC_LIB="${CROSS_GCC_LIBS[0]}" # ...we just want the first one (there should only be one)
export CROSS_C_INCLUDE_PATH="${CROSS_GCC_LIB}/include:${CROSS_GCC_LIB}/include-fixed:${CROSS_GLIBC}/include:${CROSS_KERNEL}/include"
export CROSS_CPLUS_INCLUDE_PATH="${CROSS_GCC}/include/c++:${CROSS_GCC}/include/c++/${HOST}:${CROSS_GCC}/include/c++/backward:${CROSS_C_INCLUDE_PATH}"
export CROSS_LIBRARY_PATH="${CROSS_GCC_LIB_STORE}/lib:${CROSS_GCC_LIB}:${CROSS_GLIBC}/lib:${CROSS_GLIBC_STATIC}/lib"
;;
*)
exit 1 ;;
esac
# Sanity check CROSS_*_PATH directories
IFS=':' read -ra PATHS <<< "${CROSS_C_INCLUDE_PATH}:${CROSS_CPLUS_INCLUDE_PATH}:${CROSS_LIBRARY_PATH}"
for p in "${PATHS[@]}"; do
if [ -n "$p" ] && [ ! -d "$p" ]; then
echo "'$p' doesn't exist or isn't a directory... Aborting..."
exit 1
fi
done
# Disable Guix ld auto-rpath behavior
2024-11-02 16:33:59 +00:00
export GUIX_LD_WRAPPER_DISABLE_RPATH=yes
2022-06-02 12:50:33 +00:00
# Make /usr/bin if it doesn't exist
[ -e /usr/bin ] || mkdir -p /usr/bin
# Symlink file and env to a conventional path
[ -e /usr/bin/file ] || ln -s --no-dereference "$(command -v file)" /usr/bin/file
[ -e /usr/bin/env ] || ln -s --no-dereference "$(command -v env)" /usr/bin/env
# Determine the correct value for -Wl,--dynamic-linker for the current $HOST
case "$HOST" in
*linux*)
glibc_dynamic_linker=$(
case "$HOST" in
x86_64-linux-gnu) echo /lib64/ld-linux-x86-64.so.2 ;;
arm-linux-gnueabihf) echo /lib/ld-linux-armhf.so.3 ;;
aarch64-linux-gnu) echo /lib/ld-linux-aarch64.so.1 ;;
riscv64-linux-gnu) echo /lib/ld-linux-riscv64-lp64d.so.1 ;;
powerpc64-linux-gnu) echo /lib64/ld64.so.1;;
powerpc64le-linux-gnu) echo /lib64/ld64.so.2;;
*) exit 1 ;;
esac
)
;;
esac
2022-12-23 15:43:34 +00:00
export GLIBC_DYNAMIC_LINKER=${glibc_dynamic_linker}
2022-06-02 12:50:33 +00:00
# Environment variables for determinism
export TAR_OPTIONS="--owner=0 --group=0 --numeric-owner --mtime='@${SOURCE_DATE_EPOCH}' --sort=name"
export TZ="UTC"
####################
# Depends Building #
####################
2022-12-21 15:15:22 +00:00
# LDFLAGS
case "$HOST" in
*linux*) HOST_LDFLAGS="-Wl,--as-needed -Wl,--dynamic-linker=$glibc_dynamic_linker -static-libstdc++ -Wl,-O2" ;;
*mingw*) HOST_LDFLAGS="-Wl,--no-insert-timestamp" ;;
esac
2023-03-18 11:25:48 +00:00
mkdir -p "$OUTDIR"
2024-03-13 10:44:29 +00:00
# Log the depends build ids
make -C contrib/depends --no-print-directory HOST="$HOST" print-final_build_id_long | tr ':' '\n' > ${LOGDIR}/depends-hashes.txt
export CMAKE_BUILD_PARALLEL_LEVEL=$JOBS
2022-06-02 12:50:33 +00:00
# Build the depends tree, overriding variables that assume multilib gcc
make -C contrib/depends --jobs="$JOBS" HOST="$HOST" \
${V:+V=1} \
${SOURCES_PATH+SOURCES_PATH="$SOURCES_PATH"} \
${BASE_CACHE+BASE_CACHE="$BASE_CACHE"} \
${SDK_PATH+SDK_PATH="$SDK_PATH"} \
2023-03-18 11:25:48 +00:00
OUTDIR="$OUTDIR" \
2024-03-13 10:44:29 +00:00
LOGDIR="$LOGDIR" \
2022-06-02 12:50:33 +00:00
x86_64_linux_CC=x86_64-linux-gnu-gcc \
x86_64_linux_CXX=x86_64-linux-gnu-g++ \
x86_64_linux_AR=x86_64-linux-gnu-gcc-ar \
x86_64_linux_RANLIB=x86_64-linux-gnu-gcc-ranlib \
x86_64_linux_NM=x86_64-linux-gnu-gcc-nm \
2022-06-02 12:50:33 +00:00
x86_64_linux_STRIP=x86_64-linux-gnu-strip \
2022-12-21 15:15:22 +00:00
guix_ldflags="$HOST_LDFLAGS"
2022-06-02 12:50:33 +00:00
2024-03-13 10:44:29 +00:00
# Log the depends package hashes
DEPENDS_PACKAGES="$(make -C contrib/depends --no-print-directory HOST="$HOST" print-all_packages)"
DEPENDS_CACHE="$(make -C contrib/depends --no-print-directory ${BASE_CACHE+BASE_CACHE="$BASE_CACHE"} print-BASE_CACHE)"
{
for package in ${DEPENDS_PACKAGES}; do
cat "${DEPENDS_CACHE}/${HOST}/${package}"/*.hash
done
} | sort -k2 > "${LOGDIR}/depends-packages.txt"
2022-06-02 12:50:33 +00:00
###########################
# Source Tarball Building #
###########################
# Use COMMIT_TIMESTAMP for the source and release binary archives
export SOURCE_DATE_EPOCH=${COMMIT_TIMESTAMP}
export TAR_OPTIONS="--owner=0 --group=0 --numeric-owner --mtime='@${SOURCE_DATE_EPOCH}' --sort=name"
2022-06-02 12:50:33 +00:00
GIT_ARCHIVE="${DIST_ARCHIVE_BASE}/${DISTNAME}.tar.gz"
# Create the source tarball if not already there
if [ ! -e "$GIT_ARCHIVE" ]; then
mkdir -p "$(dirname "$GIT_ARCHIVE")"
2023-01-06 21:14:48 +00:00
git rev-parse --short=12 HEAD > githash.txt
( git ls-files --recurse-submodules ; echo "githash.txt" ) \
| cat \
| sort \
| tar --create --transform "s,^,${DISTNAME}/," --mode='u+rw,go+r-w,a+X' --files-from=- \
| gzip -9n > ${GIT_ARCHIVE}
2022-12-22 12:22:26 +00:00
sha256sum "$GIT_ARCHIVE"
2022-06-02 12:50:33 +00:00
fi
###########################
# Binary Tarball Building #
###########################
# CFLAGS
2024-03-13 10:44:29 +00:00
HOST_CFLAGS="-O2"
2022-12-21 15:15:22 +00:00
HOST_CFLAGS+=$(find /gnu/store -maxdepth 1 -mindepth 1 -type d -exec echo -n " -ffile-prefix-map={}=/usr" \;)
2022-06-02 12:50:33 +00:00
case "$HOST" in
*linux*) HOST_CFLAGS+=" -ffile-prefix-map=${PWD}=." ;;
*mingw*) HOST_CFLAGS+=" -fno-ident" ;;
*darwin*) unset HOST_CFLAGS ;;
esac
# CXXFLAGS
HOST_CXXFLAGS="$HOST_CFLAGS"
case "$HOST" in
arm-linux-gnueabihf) HOST_CXXFLAGS="${HOST_CXXFLAGS} -Wno-psabi" ;;
esac
2024-09-29 07:36:18 +00:00
export USE_DEVICE_TREZOR_MANDATORY=1
2022-06-02 12:50:33 +00:00
# Make $HOST-specific native binaries from depends available in $PATH
2022-12-21 15:15:22 +00:00
export PATH="${BASEPREFIX}/${HOST}/native/bin:${PATH}"
2022-06-02 12:50:33 +00:00
mkdir -p "$DISTSRC"
(
cd "$DISTSRC"
# Extract the source tarball
tar --strip-components=1 -xf "${GIT_ARCHIVE}"
2022-12-21 15:15:22 +00:00
# Setup the directory where our Bitcoin Core build for HOST will be
2022-06-02 12:50:33 +00:00
# installed. This directory will also later serve as the input for our
# binary tarballs.
2022-12-21 15:15:22 +00:00
INSTALLPATH="${DISTSRC}/installed"
2022-06-02 12:50:33 +00:00
mkdir -p "${INSTALLPATH}"
2022-12-21 15:15:22 +00:00
# Set appropriate CMake options for build type
2024-02-27 18:29:47 +00:00
CMAKEVARS="-DWITH_SCANNER=On -DCHECK_UPDATES=On -DSELF_CONTAINED=On -DDONATE_BEG=On -DFEATHER_TARGET_TRIPLET=${HOST} -DWITH_PLUGIN_REDDIT=Off"
2023-03-01 15:02:58 +00:00
ANONDIST=""
2022-12-21 15:15:22 +00:00
case "$HOST" in
*mingw32)
case "$OPTIONS" in
installer)
CMAKEVARS+=" -DPLATFORM_INSTALLER=On -DTOR_DIR=Off -DTOR_VERSION=Off"
2022-12-21 15:15:22 +00:00
;;
esac
;;
*linux*)
2023-01-26 12:12:33 +00:00
CMAKEVARS+=" -DSTACK_TRACE=ON"
2022-12-21 15:15:22 +00:00
case "$OPTIONS" in
2023-02-13 11:04:03 +00:00
no-tor-bundle)
2022-12-21 15:15:22 +00:00
CMAKEVARS+=" -DTOR_DIR=Off -DTOR_VERSION=Off"
2023-03-01 15:02:58 +00:00
ANONDIST+="-a"
2022-12-21 15:15:22 +00:00
;;
2023-06-28 15:30:29 +00:00
pack)
2023-05-06 11:48:11 +00:00
CMAKEVARS+=" -DCHECK_UPDATES=Off -DSELF_CONTAINED=Off"
;;
2022-12-21 15:15:22 +00:00
esac
;;
2022-12-23 15:43:34 +00:00
*gnueabihf)
CMAKEVARS+=" -DNO_AES=On" # Raspberry Pi
;;
2022-12-21 15:15:22 +00:00
esac
# Configure this DISTSRC for $HOST
# shellcheck disable=SC2086
env CFLAGS="${HOST_CFLAGS}" CXXFLAGS="${HOST_CXXFLAGS}" \
cmake --toolchain "${BASEPREFIX}/${HOST}/share/toolchain.cmake" -S . -B build \
-DCMAKE_INSTALL_PREFIX="${INSTALLPATH}" \
-DCCACHE=OFF \
${CONFIGFLAGS} \
-DCMAKE_EXE_LINKER_FLAGS="${HOST_LDFLAGS}" \
-DCMAKE_SHARED_LINKER_FLAGS="${HOST_LDFLAGS}" \
${CMAKEVARS}
make -C build --jobs="$JOBS"
2023-01-06 22:03:13 +00:00
LINUX_ARCH=""
case "$HOST" in
aarch64-linux*)
LINUX_ARCH="-arm64"
;;
arm-linux*)
LINUX_ARCH="-arm"
;;
2023-03-22 16:34:24 +00:00
riscv64-linux*)
LINUX_ARCH="-riscv64"
;;
2023-01-06 22:03:13 +00:00
esac
2023-03-23 19:10:01 +00:00
DARWIN_ARCH=""
case "$HOST" in
arm64-apple-darwin)
DARWIN_ARCH="-arm64"
;;
esac
2022-12-21 15:15:22 +00:00
case "$HOST" in
*linux*)
2023-06-28 15:30:29 +00:00
if [ "$OPTIONS" != "pack" ]; then
bash contrib/AppImage/build-appimage.sh
2023-05-06 11:48:11 +00:00
APPIMAGENAME=${DISTNAME}${ANONDIST}${LINUX_ARCH}.AppImage
mv feather.AppImage "${APPIMAGENAME}"
2023-05-06 11:48:11 +00:00
cp "${APPIMAGENAME}" "${INSTALLPATH}/"
cp "${APPIMAGENAME}" "${OUTDIR}/"
fi
2022-12-21 15:15:22 +00:00
;;
esac
mkdir -p "$OUTDIR"
# Make the os-specific installers
case "$HOST" in
*mingw*)
case "$OPTIONS" in
installer)
makensis -DCUR_PATH=$PWD -V2 contrib/installers/windows/setup.nsi
cp contrib/installers/windows/FeatherWalletSetup-*.exe "${INSTALLPATH}/"
mv contrib/installers/windows/FeatherWalletSetup-*.exe "${OUTDIR}/"
;;
esac
;;
esac
# Install built Feather to $INSTALLPATH
case "$HOST" in
*darwin*)
make -C build install/strip ${V:+V=1}
;;
*.installer)
2022-12-21 15:15:22 +00:00
;;
*)
case "$OPTIONS" in
installer)
# do nothing, we don't want feather.exe in the final .zip
;;
*)
make -C build install ${V:+V=1}
;;
esac
2022-12-21 15:15:22 +00:00
esac
(
cd installed
case "$HOST" in
2023-03-15 13:31:45 +00:00
*darwin*)
mv "feather.app" "Feather.app"
;;
esac
2022-12-21 15:15:22 +00:00
# Finally, deterministically produce {non-,}debug binary tarballs ready
# for release
case "$HOST" in
*mingw*)
case "$OPTIONS" in
installer)
find . -print0 \
| xargs -0r touch --no-dereference --date="@${SOURCE_DATE_EPOCH}"
find . \
| sort \
| zip -X@ "${OUTDIR}/${DISTNAME}-win-installer.zip" \
|| ( rm -f "${OUTDIR}/${DISTNAME}-win-installer.zip" && exit 1 )
;;
"")
mv feather.exe ${DISTNAME}.exe && \
find . -print0 \
| xargs -0r touch --no-dereference --date="@${SOURCE_DATE_EPOCH}"
find . \
| sort \
| zip -X@ "${OUTDIR}/${DISTNAME}-win.zip" \
|| ( rm -f "${OUTDIR}/${DISTNAME}-win.zip" && exit 1 )
;;
esac
2022-12-21 15:15:22 +00:00
;;
*linux*)
2023-06-28 15:30:29 +00:00
if [ "$OPTIONS" != "pack" ]; then
2023-05-06 11:48:11 +00:00
mv feather "${DISTNAME}"
case "$OPTIONS" in
"")
find . -not -name "*.AppImage" -print0 \
| xargs -0r touch --no-dereference --date="@${SOURCE_DATE_EPOCH}"
find . -not -name "*.AppImage" \
| sort \
| zip -X@ "${OUTDIR}/${DISTNAME}-linux${LINUX_ARCH}${ANONDIST}.zip" \
|| ( rm -f "${OUTDIR}/${DISTNAME}-linux${LINUX_ARCH}${ANONDIST}.zip" && exit 1 )
;;
esac
find . -name "*.AppImage" -print0 \
| xargs -0r touch --no-dereference --date="@${SOURCE_DATE_EPOCH}"
find . -name "*.AppImage" \
| sort \
| zip -X@ "${OUTDIR}/${DISTNAME}-linux${LINUX_ARCH}-appimage${ANONDIST}.zip" \
|| ( rm -f "${OUTDIR}/${DISTNAME}-linux${LINUX_ARCH}-appimage${ANONDIST}.zip" && exit 1 )
2023-05-06 11:48:11 +00:00
else
find . -print0 \
| xargs -0r touch --no-dereference --date="@${SOURCE_DATE_EPOCH}"
2023-06-28 18:23:37 +00:00
cp feather "${OUTDIR}"
2023-05-06 11:48:11 +00:00
fi
2022-12-21 15:15:22 +00:00
;;
2023-03-15 13:31:45 +00:00
*darwin*)
find . -print0 \
| xargs -0r touch --no-dereference --date="@${SOURCE_DATE_EPOCH}"
find . \
| sort \
2023-03-23 19:10:01 +00:00
| zip -X@ "${OUTDIR}/${DISTNAME}-mac${DARWIN_ARCH}.zip" \
|| ( rm -f "${OUTDIR}/${DISTNAME}-mac${DARWIN_ARCH}.zip" && exit 1 )
2023-03-15 13:31:45 +00:00
;;
2022-12-21 15:15:22 +00:00
esac
)
2022-06-02 12:50:33 +00:00
) # $DISTSRC
2022-12-21 15:15:22 +00:00
rm -rf "$ACTUAL_OUTDIR"
mv --no-target-directory "$OUTDIR" "$ACTUAL_OUTDIR" \
|| ( rm -rf "$ACTUAL_OUTDIR" && exit 1 )
(
cd /outdir-base
{
echo "$GIT_ARCHIVE"
2023-10-06 13:38:54 +00:00
find "$ACTUAL_OUTDIR" -type f -not -name "*.txt"
2022-12-21 15:15:22 +00:00
} | xargs realpath --relative-base="$PWD" \
| xargs sha256sum \
| sort -k2 \
2024-03-13 10:44:29 +00:00
| sponge "$LOGDIR"/SHA256SUMS.part
2022-12-21 15:15:22 +00:00
)