commit 8b888909f4c2df23ee93d35997cafbd2e06a9241 Author: CentOS Sources Date: Tue May 17 05:01:21 2022 -0400 import rear-2.6-11.el9 diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..5d90595 --- /dev/null +++ b/.gitignore @@ -0,0 +1 @@ +SOURCES/rear-2.6.tar.gz diff --git a/.rear.metadata b/.rear.metadata new file mode 100644 index 0000000..a5f7fb3 --- /dev/null +++ b/.rear.metadata @@ -0,0 +1 @@ +13c23ad59254438ffcd0cde6400fd991cbfe194e SOURCES/rear-2.6.tar.gz diff --git a/SOURCES/0001-skip-kernel-buildin-modules.patch b/SOURCES/0001-skip-kernel-buildin-modules.patch new file mode 100644 index 0000000..c7d07f5 --- /dev/null +++ b/SOURCES/0001-skip-kernel-buildin-modules.patch @@ -0,0 +1,47 @@ +From df5e18b8d7c8359b48bc133bfa29734934d18160 Mon Sep 17 00:00:00 2001 +From: Johannes Meixner +Date: Mon, 10 Aug 2020 16:20:38 +0200 +Subject: [PATCH] Merge pull request #2469 from + rear/skip-kernel-builtin-modules-issue2414 + +In 400_copy_modules.sh skip copying kernel modules that are builtin modules. +The new behaviour is that when modules are listed in modules.builtin +and are also shown by modinfo then those modules are now skipped. +Before for such modules the modules file(s) would have been included +in the recovery system. +See https://github.com/rear/rear/issues/2414 +--- + usr/share/rear/build/GNU/Linux/400_copy_modules.sh | 12 ++++++++++-- + 1 file changed, 10 insertions(+), 2 deletions(-) + +diff --git a/usr/share/rear/build/GNU/Linux/400_copy_modules.sh b/usr/share/rear/build/GNU/Linux/400_copy_modules.sh +index d8d733d2..641b7f83 100644 +--- a/usr/share/rear/build/GNU/Linux/400_copy_modules.sh ++++ b/usr/share/rear/build/GNU/Linux/400_copy_modules.sh +@@ -133,8 +133,13 @@ for dummy in "once" ; do + module=${module#.o} + # Strip trailing ".ko" if there: + module=${module#.ko} +- # Continue with the next module if the current one does not exist: ++ # Continue with the next module if the current one does not exist as a module file: + modinfo $module 1>/dev/null || continue ++ # Continue with the next module if the current one is a kernel builtin module ++ # cf. https://github.com/rear/rear/issues/2414#issuecomment-668632798 ++ # Quoting the grep search value is mandatory here ($module might be empty or blank), ++ # cf. "Beware of the emptiness" in https://github.com/rear/rear/wiki/Coding-Style ++ grep -q "$( echo $module | tr '_-' '..' )" /lib/modules/$KERNEL_VERSION/modules.builtin && continue + # Resolve module dependencies: + # Get the module file plus the module files of other needed modules. + # This is currently only a "best effort" attempt because +@@ -166,7 +171,10 @@ done + + # Remove those modules that are specified in the EXCLUDE_MODULES array: + for exclude_module in "${EXCLUDE_MODULES[@]}" ; do +- # Continue with the next module if the current one does not exist: ++ # Continue with the next module only if the current one does not exist as a module file ++ # but do not continue with the next module if the current one is a kernel builtin module ++ # so when a module file exists that gets removed regardless if it is also a builtin module ++ # cf. https://github.com/rear/rear/issues/2414#issuecomment-669115481 + modinfo $exclude_module 1>/dev/null || continue + # In this case it is ignored when a module exists but 'modinfo -F filename' cannot show its filename + # because then it is assumed that also no module file had been copied above: diff --git a/SOURCES/rear-bz1492177-warning.patch b/SOURCES/rear-bz1492177-warning.patch new file mode 100644 index 0000000..1f5556f --- /dev/null +++ b/SOURCES/rear-bz1492177-warning.patch @@ -0,0 +1,15 @@ +diff --git a/usr/share/rear/output/ISO/Linux-i386/249_check_rhel_grub2_efi_package.sh b/usr/share/rear/output/ISO/Linux-i386/249_check_rhel_grub2_efi_package.sh +new file mode 100644 +index 00000000..4c4ded08 +--- /dev/null ++++ b/usr/share/rear/output/ISO/Linux-i386/249_check_rhel_grub2_efi_package.sh +@@ -0,0 +1,9 @@ ++# 249_check_rhel_grub2_efi_package.sh ++ ++is_true $USING_UEFI_BOOTLOADER || return # empty or 0 means NO UEFI ++ ++( ++ VERBOSE=1 ++ test -r /usr/lib/grub/x86_64-efi/moddep.lst ++ PrintIfError "WARNING: /usr/lib/grub/x86_64-efi/moddep.lst not found, grub2-mkimage will likely fail. Please install the grub2-efi-x64-modules package to fix this." ++) diff --git a/SOURCES/rear-bz1747468.patch b/SOURCES/rear-bz1747468.patch new file mode 100644 index 0000000..a2f7bb3 --- /dev/null +++ b/SOURCES/rear-bz1747468.patch @@ -0,0 +1,112 @@ +diff --git a/usr/share/rear/layout/prepare/GNU/Linux/110_include_lvm_code.sh b/usr/share/rear/layout/prepare/GNU/Linux/110_include_lvm_code.sh +index 7cfdfcf2..1be17ba8 100644 +--- a/usr/share/rear/layout/prepare/GNU/Linux/110_include_lvm_code.sh ++++ b/usr/share/rear/layout/prepare/GNU/Linux/110_include_lvm_code.sh +@@ -68,9 +68,9 @@ create_lvmgrp() { + local vg=${vgrp#/dev/} + + cat >> "$LAYOUT_CODE" <> "$LAYOUT_CODE" <&2 ; then + + LogPrint "Sleeping 3 seconds to let udev or systemd-udevd create their devices..." + sleep 3 >&2 +- create_volume_group=0 +- create_logical_volumes=0 ++ create_volume_group=( \$( RmInArray "$vg" "\${create_volume_group[@]}" ) ) ++ create_logical_volumes=( \$( RmInArray "$vg" "\${create_logical_volumes[@]}" ) ) + ++EOF ++ if is_true "${FORCE_VGCFGRESTORE-no}"; then ++ cat >> "$LAYOUT_CODE" <&2 ; then + sleep 3 >&2 + + # All logical volumes have been created, except Thin volumes and pools +- create_volume_group=0 +- create_thin_volumes_only=1 ++ create_volume_group=( \$( RmInArray "$vg" "\${create_volume_group[@]}" ) ) ++ create_thin_volumes_only+=( "$vg" ) + ++EOF ++ fi ++ cat >> "$LAYOUT_CODE" <> "$LAYOUT_CODE" < in the first argument) ++# doesn't contain any LVs that use kernel metadata. ++# If the function returns true, we can safely use vgcfgrestore to restore the VG. ++function lvmgrp_supports_vgcfgrestore() { ++ if is_true "${FORCE_VGCFGRESTORE-no}"; then ++ # If we are willing to use vgcfgrestore --force and then remove broken volumes, ++ # then everything can be considered supported. Don't do it by default though. ++ return 0 ++ fi ++ ++ local lvmvol vgrp lvname size layout kval ++ ++ local supported_layouts=("linear" "striped") ++ ++ while read lvmvol vgrp lvname size layout kval; do ++ [ "$vgrp" == "$1" ] || BugError "vgrp '$vgrp' != '$1'" ++ if ! IsInArray $layout "${supported_layouts[@]}"; then ++ LogPrint "Layout '$layout' of LV '$lvname' in VG '$vgrp' not supported by vgcfgrestore" ++ return 1 ++ fi ++ done < <(grep "^lvmvol $1 " "$LAYOUT_FILE") ++} ++ + # vim: set et ts=4 sw=4: diff --git a/SOURCES/rear-bz1832394.patch b/SOURCES/rear-bz1832394.patch new file mode 100644 index 0000000..3405422 --- /dev/null +++ b/SOURCES/rear-bz1832394.patch @@ -0,0 +1,351 @@ +diff --git a/doc/user-guide/06-layout-configuration.adoc b/doc/user-guide/06-layout-configuration.adoc +index f59384db..88ba0420 100644 +--- a/doc/user-guide/06-layout-configuration.adoc ++++ b/doc/user-guide/06-layout-configuration.adoc +@@ -630,7 +630,7 @@ lvmvol [key:value ...] + + === LUKS Devices === + ---------------------------------- +-crypt /dev/mapper/ [cipher=] [key_size=] [hash=] [uuid=] [keyfile=] [password=] ++crypt /dev/mapper/ [type=] [cipher=] [key_size=] [hash=] [uuid=] [keyfile=] [password=] + ---------------------------------- + + === DRBD === +diff --git a/usr/share/rear/layout/prepare/GNU/Linux/160_include_luks_code.sh b/usr/share/rear/layout/prepare/GNU/Linux/160_include_luks_code.sh +index 05279bc8..0c662f67 100644 +--- a/usr/share/rear/layout/prepare/GNU/Linux/160_include_luks_code.sh ++++ b/usr/share/rear/layout/prepare/GNU/Linux/160_include_luks_code.sh +@@ -1,35 +1,75 @@ ++ + # Code to recreate and/or open LUKS volumes. + + create_crypt() { ++ # See the create_device() function in lib/layout-functions.sh what "device type" means: ++ local device_type="$1" ++ if ! grep -q "^crypt $device_type " "$LAYOUT_FILE" ; then ++ LogPrintError "Skip recreating LUKS volume $device_type (no 'crypt $device_type' entry in $LAYOUT_FILE)" ++ # FIXME: The return code is ignored in the create_device() function in lib/layout-functions.sh: ++ return 1 ++ fi ++ + local crypt target_device source_device options +- read crypt target_device source_device options < <(grep "^crypt $1 " "$LAYOUT_FILE") ++ local mapping_name option key value ++ local cryptsetup_options="" keyfile="" password="" + +- local target_name=${target_device#/dev/mapper/} ++ read crypt target_device source_device options < <( grep "^crypt $device_type " "$LAYOUT_FILE" ) ++ ++ # Careful! One cannot 'test -b $source_device' here at the time when this code is run ++ # because the source device is usually a disk partition block device like /dev/sda2 ++ # but disk partition block devices usually do not yet exist (in particular not on a new clean disk) ++ # because partitions are actually created later when the diskrestore.sh script is run ++ # but not here when this code is run which only generates the diskrestore.sh script: ++ if ! test $source_device ; then ++ LogPrintError "Skip recreating LUKS volume $device_type: No source device (see the 'crypt $device_type' entry in $LAYOUT_FILE)" ++ # FIXME: The return code is ignored in the create_device() function in lib/layout-functions.sh: ++ return 1 ++ fi ++ ++ mapping_name=${target_device#/dev/mapper/} ++ if ! test $mapping_name ; then ++ LogPrintError "Skip recreating LUKS volume $device_type on $source_device: No /dev/mapper/... mapping name (see the 'crypt $device_type' entry in $LAYOUT_FILE)" ++ # FIXME: The return code is ignored in the create_device() function in lib/layout-functions.sh: ++ return 1 ++ fi + +- local cryptsetup_options="" keyfile="" password="" +- local option key value + for option in $options ; do +- key=${option%=*} ++ # $option is of the form keyword=value and ++ # we assume keyword has no '=' character but value could be anything that may have a '=' character ++ # so we split keyword=value at the leftmost '=' character so that ++ # e.g. keyword=foo=bar gets split into key="keyword" and value="foo=bar": ++ key=${option%%=*} + value=${option#*=} +- ++ # The "cryptseup luksFormat" command does not require any of the type, cipher, key-size, hash, uuid option values ++ # because if omitted a cryptseup default value is used so we treat those values as optional. ++ # Using plain test to ensure the value is a single non empty and non blank word ++ # without quoting because test " " would return zero exit code ++ # cf. "Beware of the emptiness" in https://github.com/rear/rear/wiki/Coding-Style + case "$key" in +- cipher) +- cryptsetup_options+=" --cipher $value" ++ (type) ++ test $value && cryptsetup_options+=" --type $value" ++ ;; ++ (cipher) ++ test $value && cryptsetup_options+=" --cipher $value" ++ ;; ++ (key_size) ++ test $value && cryptsetup_options+=" --key-size $value" + ;; +- key_size) +- cryptsetup_options+=" --key-size $value" ++ (hash) ++ test $value && cryptsetup_options+=" --hash $value" + ;; +- hash) +- cryptsetup_options+=" --hash $value" ++ (uuid) ++ test $value && cryptsetup_options+=" --uuid $value" + ;; +- uuid) +- cryptsetup_options+=" --uuid $value" ++ (keyfile) ++ test $value && keyfile=$value + ;; +- keyfile) +- keyfile=$value ++ (password) ++ test $value && password=$value + ;; +- password) +- password=$value ++ (*) ++ LogPrintError "Skipping unsupported LUKS cryptsetup option '$key' in 'crypt $target_device $source_device' entry in $LAYOUT_FILE" + ;; + esac + done +@@ -37,26 +77,25 @@ create_crypt() { + cryptsetup_options+=" $LUKS_CRYPTSETUP_OPTIONS" + + ( +- echo "Log \"Creating LUKS device $target_name on $source_device\"" ++ echo "LogPrint \"Creating LUKS volume $mapping_name on $source_device\"" + if [ -n "$keyfile" ] ; then + # Assign a temporary keyfile at this stage so that original keyfiles do not leak onto the rescue medium. + # The original keyfile will be restored from the backup and then re-assigned to the LUKS device in the + # 'finalize' stage. + # The scheme for generating a temporary keyfile path must be the same here and in the 'finalize' stage. +- keyfile="${TMPDIR:-/tmp}/LUKS-keyfile-$target_name" ++ keyfile="$TMP_DIR/LUKS-keyfile-$mapping_name" + dd bs=512 count=4 if=/dev/urandom of="$keyfile" + chmod u=rw,go=- "$keyfile" +- + echo "cryptsetup luksFormat --batch-mode $cryptsetup_options $source_device $keyfile" +- echo "cryptsetup luksOpen --key-file $keyfile $source_device $target_name" ++ echo "cryptsetup luksOpen --key-file $keyfile $source_device $mapping_name" + elif [ -n "$password" ] ; then + echo "echo \"$password\" | cryptsetup luksFormat --batch-mode $cryptsetup_options $source_device" +- echo "echo \"$password\" | cryptsetup luksOpen $source_device $target_name" ++ echo "echo \"$password\" | cryptsetup luksOpen $source_device $mapping_name" + else +- echo "LogPrint \"Please enter the password for LUKS device $target_name ($source_device):\"" ++ echo "LogUserOutput \"Set the password for LUKS volume $mapping_name (for 'cryptsetup luksFormat' on $source_device):\"" + echo "cryptsetup luksFormat --batch-mode $cryptsetup_options $source_device" +- echo "LogPrint \"Please re-enter the password for LUKS device $target_name ($source_device):\"" +- echo "cryptsetup luksOpen $source_device $target_name" ++ echo "LogUserOutput \"Enter the password for LUKS volume $mapping_name (for 'cryptsetup luksOpen' on $source_device):\"" ++ echo "cryptsetup luksOpen $source_device $mapping_name" + fi + echo "" + ) >> "$LAYOUT_CODE" +@@ -64,38 +103,61 @@ create_crypt() { + + # Function open_crypt() is meant to be used by the 'mountonly' workflow + open_crypt() { ++ # See the do_mount_device() function in lib/layout-functions.sh what "device type" means: ++ local device_type="$1" ++ if ! grep -q "^crypt $device_type " "$LAYOUT_FILE" ; then ++ LogPrintError "Skip opening LUKS volume $device_type (no 'crypt $device_type' entry in $LAYOUT_FILE)" ++ # FIXME: The return code is ignored in the do_mount_device() function in lib/layout-functions.sh: ++ return 1 ++ fi ++ + local crypt target_device source_device options +- read crypt target_device source_device options < <(grep "^crypt $1 " "$LAYOUT_FILE") ++ local mapping_name option key value ++ local cryptsetup_options="" keyfile="" password="" + +- local target_name=${target_device#/dev/mapper/} ++ read crypt target_device source_device options < <( grep "^crypt $device_type " "$LAYOUT_FILE" ) ++ ++ if ! test -b "$source_device" ; then ++ LogPrintError "Skip opening LUKS volume $device_type on device '$source_device' that is no block device (see the 'crypt $device_type' entry in $LAYOUT_FILE)" ++ # FIXME: The return code is ignored in the do_mount_device() function in lib/layout-functions.sh: ++ return 1 ++ fi ++ ++ mapping_name=${target_device#/dev/mapper/} ++ if ! test $mapping_name ; then ++ LogPrintError "Skip opening LUKS volume $device_type on $source_device: No /dev/mapper/... mapping name (see the 'crypt $device_type' entry in $LAYOUT_FILE)" ++ # FIXME: The return code is ignored in the do_mount_device() function in lib/layout-functions.sh: ++ return 1 ++ fi + +- local cryptsetup_options="" keyfile="" password="" +- local option key value + for option in $options ; do +- key=${option%=*} ++ # $option is of the form keyword=value and ++ # we assume keyword has no '=' character but value could be anything that may have a '=' character ++ # so we split keyword=value at the leftmost '=' character so that ++ # e.g. keyword=foo=bar gets split into key="keyword" and value="foo=bar": ++ key=${option%%=*} + value=${option#*=} +- + case "$key" in +- keyfile) +- keyfile=$value ++ (keyfile) ++ test $value && keyfile=$value + ;; +- password) +- password=$value ++ (password) ++ test $value && password=$value + ;; + esac + done + + ( +- echo "Log \"Opening LUKS device $target_name on $source_device\"" ++ echo "LogPrint \"Opening LUKS volume $mapping_name on $source_device\"" + if [ -n "$keyfile" ] ; then + # During a 'mountonly' workflow, the original keyfile is supposed to be + # available at this point. +- echo "cryptsetup luksOpen --key-file $keyfile $source_device $target_name" ++ echo "cryptsetup luksOpen --key-file $keyfile $source_device $mapping_name" + elif [ -n "$password" ] ; then +- echo "echo \"$password\" | cryptsetup luksOpen $source_device $target_name" ++ echo "echo \"$password\" | cryptsetup luksOpen $source_device $mapping_name" + else +- echo "LogPrint \"Please enter the password for LUKS device $target_name ($source_device):\"" +- echo "cryptsetup luksOpen $source_device $target_name" ++ echo "LogUserOutput \"Enter the password for LUKS volume $mapping_name (for 'cryptsetup luksOpen' on $source_device):\"" ++ echo "cryptsetup luksOpen $source_device $mapping_name" + fi + echo "" + ) >> "$LAYOUT_CODE" +diff --git a/usr/share/rear/layout/save/GNU/Linux/260_crypt_layout.sh b/usr/share/rear/layout/save/GNU/Linux/260_crypt_layout.sh +index c1e1cfd5..afeabf6a 100644 +--- a/usr/share/rear/layout/save/GNU/Linux/260_crypt_layout.sh ++++ b/usr/share/rear/layout/save/GNU/Linux/260_crypt_layout.sh +@@ -9,6 +9,8 @@ Log "Saving Encrypted volumes." + REQUIRED_PROGS+=( cryptsetup dmsetup ) + COPY_AS_IS+=( /usr/share/cracklib/\* /etc/security/pwquality.conf ) + ++local invalid_cryptsetup_option_value="no" ++ + while read target_name junk ; do + # find the target device we're mapping + if ! [ -e /dev/mapper/$target_name ] ; then +@@ -30,29 +32,96 @@ while read target_name junk ; do + source_device="$(get_device_name ${slave##*/})" + done + +- if ! cryptsetup isLuks $source_device >/dev/null 2>&1; then ++ if ! blkid -p -o export $source_device >$TMP_DIR/blkid.output ; then ++ LogPrintError "Error: Cannot get attributes for $target_name ('blkid -p -o export $source_device' failed)" + continue + fi + +- # gather crypt information +- cipher=$(cryptsetup luksDump $source_device | grep "Cipher name" | sed -r 's/^.+:\s*(.+)$/\1/') +- mode=$(cryptsetup luksDump $source_device | grep "Cipher mode" | cut -d: -f2- | awk '{printf("%s",$1)};') +- key_size=$(cryptsetup luksDump $source_device | grep "MK bits" | sed -r 's/^.+:\s*(.+)$/\1/') +- hash=$(cryptsetup luksDump $source_device | grep "Hash spec" | sed -r 's/^.+:\s*(.+)$/\1/') +- uuid=$(cryptsetup luksDump $source_device | grep "UUID" | sed -r 's/^.+:\s*(.+)$/\1/') +- keyfile_option=$([ -f /etc/crypttab ] && awk '$1 == "'"$target_name"'" && $3 != "none" && $3 != "-" && $3 != "" { print "keyfile=" $3; }' /etc/crypttab) ++ if ! grep -q "TYPE=crypto_LUKS" $TMP_DIR/blkid.output ; then ++ Log "Skipping $target_name (no 'TYPE=crypto_LUKS' in 'blkid -p -o export $source_device' output)" ++ continue ++ fi + +- # LUKS version 2 is not yet suppported, see https://github.com/rear/rear/issues/2204 +- # When LUKS version 2 is used the above code fails at least to determine the hash value +- # so we use an empty hash value as a simple test if gathering crypt information was successful: +- test "$hash" || Error "No hash value for LUKS device '$target_name' at '$source_device' (only LUKS version 1 is supported)" ++ # Detect LUKS version: ++ # Remove all non-digits in particular to avoid leading or trailing spaces in the version string ++ # cf. "Beware of the emptiness" in https://github.com/rear/rear/wiki/Coding-Style ++ # that could happen if the blkid output contains "VERSION = 2" so that 'cut -d= -f2' results " 2". ++ version=$( grep "VERSION" $TMP_DIR/blkid.output | cut -d= -f2 | tr -c -d '[:digit:]' ) ++ if ! test "$version" = "1" -o "$version" = "2" ; then ++ LogPrintError "Error: Unsupported LUKS version for $target_name ('blkid -p -o export $source_device' shows 'VERSION=$version')" ++ continue ++ fi ++ luks_type=luks$version + +- echo "crypt /dev/mapper/$target_name $source_device cipher=$cipher-$mode key_size=$key_size hash=$hash uuid=$uuid $keyfile_option" >> $DISKLAYOUT_FILE +-done < <( dmsetup ls --target crypt ) ++ # Gather crypt information: ++ if ! cryptsetup luksDump $source_device >$TMP_DIR/cryptsetup.luksDump ; then ++ LogPrintError "Error: Cannot get LUKS$version values for $target_name ('cryptsetup luksDump $source_device' failed)" ++ continue ++ fi ++ uuid=$( grep "UUID" $TMP_DIR/cryptsetup.luksDump | sed -r 's/^.+:\s*(.+)$/\1/' ) ++ keyfile_option=$( [ -f /etc/crypttab ] && awk '$1 == "'"$target_name"'" && $3 != "none" && $3 != "-" && $3 != "" { print "keyfile=" $3; }' /etc/crypttab ) ++ if test $luks_type = "luks1" ; then ++ cipher_name=$( grep "Cipher name" $TMP_DIR/cryptsetup.luksDump | sed -r 's/^.+:\s*(.+)$/\1/' ) ++ cipher_mode=$( grep "Cipher mode" $TMP_DIR/cryptsetup.luksDump | cut -d: -f2- | awk '{printf("%s",$1)};' ) ++ cipher=$cipher_name-$cipher_mode ++ key_size=$( grep "MK bits" $TMP_DIR/cryptsetup.luksDump | sed -r 's/^.+:\s*(.+)$/\1/' ) ++ hash=$( grep "Hash spec" $TMP_DIR/cryptsetup.luksDump | sed -r 's/^.+:\s*(.+)$/\1/' ) ++ elif test $luks_type = "luks2" ; then ++ cipher=$( grep "cipher:" $TMP_DIR/cryptsetup.luksDump | sed -r 's/^.+:\s*(.+)$/\1/' ) ++ # More than one keyslot may be defined - use key_size from the first slot. ++ # Depending on the version the "cryptsetup luksDump" command outputs the key_size value ++ # as a line like ++ # Key: 512 bits ++ # and/or as a line like ++ # Cipher key: 512 bits ++ # cf. https://github.com/rear/rear/pull/2504#issuecomment-718729198 and subsequent comments ++ # so we grep for both lines but use only the first match from the first slot: ++ key_size=$( egrep -m 1 "Key:|Cipher key:" $TMP_DIR/cryptsetup.luksDump | sed -r 's/^.+:\s*(.+) bits$/\1/' ) ++ hash=$( grep "Hash" $TMP_DIR/cryptsetup.luksDump | sed -r 's/^.+:\s*(.+)$/\1/' ) ++ fi + +-# cryptsetup is required in the recovery system if disklayout.conf contains at least one 'crypt' entry +-# see the create_crypt function in layout/prepare/GNU/Linux/160_include_luks_code.sh +-# what program calls are written to diskrestore.sh +-# cf. https://github.com/rear/rear/issues/1963 +-grep -q '^crypt ' $DISKLAYOUT_FILE && REQUIRED_PROGS+=( cryptsetup ) || true ++ # Basic checks that the cipher key_size hash uuid values exist ++ # cf. https://github.com/rear/rear/pull/2504#issuecomment-718729198 ++ # because some values are needed during "rear recover" ++ # to set cryptsetup options in layout/prepare/GNU/Linux/160_include_luks_code.sh ++ # and it seems cryptsetup fails when options with empty values are specified ++ # cf. https://github.com/rear/rear/pull/2504#issuecomment-719479724 ++ # For example a LUKS1 crypt entry in disklayout.conf looks like ++ # crypt /dev/mapper/luks1test /dev/sda7 type=luks1 cipher=aes-xts-plain64 key_size=256 hash=sha256 uuid=1b4198c9-d9b0-4c57-b9a3-3433e391e706 ++ # and a LUKS1 crypt entry in disklayout.conf looks like ++ # crypt /dev/mapper/luks2test /dev/sda8 type=luks2 cipher=aes-xts-plain64 key_size=256 hash=sha256 uuid=3e874a28-7415-4f8c-9757-b3f28a96c4d2 ++ # Only the keyfile_option value is optional and the luks_type value is already tested above. ++ # Using plain test to ensure a value is a single non empty and non blank word ++ # without quoting because test " " would return zero exit code ++ # cf. "Beware of the emptiness" in https://github.com/rear/rear/wiki/Coding-Style ++ # Do not error out instantly here but only report errors here so the user can see all messages ++ # and actually error out at the end of this script if there was one actually invalid value: ++ if ! test $cipher ; then ++ LogPrint "No 'cipher' value for LUKS$version volume $target_name in $source_device" ++ fi ++ if test $key_size ; then ++ if ! is_positive_integer $key_size ; then ++ LogPrintError "Error: 'key_size=$key_size' is no positive integer for LUKS$version volume $target_name in $source_device" ++ invalid_cryptsetup_option_value="yes" ++ fi ++ else ++ LogPrint "No 'key_size' value for LUKS$version volume $target_name in $source_device" ++ fi ++ if ! test $hash ; then ++ LogPrint "No 'hash' value for LUKS$version volume $target_name in $source_device" ++ fi ++ if ! test $uuid ; then ++ # Report a missig uuid value as an error to have the user informed ++ # but do not error out here because things can be fixed manually during "rear recover" ++ # cf. https://github.com/rear/rear/pull/2506#issuecomment-721757810 ++ # and https://github.com/rear/rear/pull/2506#issuecomment-722315498 ++ # and https://github.com/rear/rear/issues/2509 ++ LogPrintError "Error: No 'uuid' value for LUKS$version volume $target_name in $source_device (mounting it or booting the recreated system may fail)" ++ fi ++ ++ echo "crypt /dev/mapper/$target_name $source_device type=$luks_type cipher=$cipher key_size=$key_size hash=$hash uuid=$uuid $keyfile_option" >> $DISKLAYOUT_FILE ++ ++done < <( dmsetup ls --target crypt ) + ++# Let this script return successfully when invalid_cryptsetup_option_value is not true: ++is_true $invalid_cryptsetup_option_value && Error "Invalid or empty LUKS cryptsetup option value(s) in $DISKLAYOUT_FILE" || true diff --git a/SOURCES/rear-bz1930662.patch b/SOURCES/rear-bz1930662.patch new file mode 100644 index 0000000..aaeae6f --- /dev/null +++ b/SOURCES/rear-bz1930662.patch @@ -0,0 +1,693 @@ +diff --git a/usr/share/rear/backup/NETFS/default/500_make_backup.sh b/usr/share/rear/backup/NETFS/default/500_make_backup.sh +index 02c204c5..60c80b5f 100644 +--- a/usr/share/rear/backup/NETFS/default/500_make_backup.sh ++++ b/usr/share/rear/backup/NETFS/default/500_make_backup.sh +@@ -16,6 +16,8 @@ function set_tar_features () { + FEATURE_TAR_IS_SET=1 + } + ++local backup_prog_rc ++ + local scheme=$( url_scheme $BACKUP_URL ) + local path=$( url_path $BACKUP_URL ) + local opath=$( backup_path $scheme $path ) +diff --git a/usr/share/rear/backup/RSYNC/GNU/Linux/610_start_selinux.sh b/usr/share/rear/backup/RSYNC/GNU/Linux/610_start_selinux.sh +index c560ec94..1692ba4c 100644 +--- a/usr/share/rear/backup/RSYNC/GNU/Linux/610_start_selinux.sh ++++ b/usr/share/rear/backup/RSYNC/GNU/Linux/610_start_selinux.sh +@@ -1,5 +1,7 @@ + # Start SELinux if it was stopped - check presence of $TMP_DIR/selinux.mode + ++local backup_prog_rc ++ + [ -f $TMP_DIR/selinux.mode ] && { + touch "${TMP_DIR}/selinux.autorelabel" + cat $TMP_DIR/selinux.mode > $SELINUX_ENFORCE +@@ -13,19 +15,19 @@ + ssh $RSYNC_USER@$RSYNC_HOST "chmod $v 755 ${RSYNC_PATH}/${RSYNC_PREFIX}/backup" 2>/dev/null + $BACKUP_PROG -a "${TMP_DIR}/selinux.autorelabel" \ + "$RSYNC_USER@$RSYNC_HOST:${RSYNC_PATH}/${RSYNC_PREFIX}/backup/.autorelabel" 2>/dev/null +- _rc=$? +- if [ $_rc -ne 0 ]; then +- LogPrint "Failed to create .autorelabel on ${RSYNC_PATH}/${RSYNC_PREFIX}/backup [${rsync_err_msg[$_rc]}]" ++ backup_prog_rc=$? ++ if [ $backup_prog_rc -ne 0 ]; then ++ LogPrint "Failed to create .autorelabel on ${RSYNC_PATH}/${RSYNC_PREFIX}/backup [${rsync_err_msg[$backup_prog_rc]}]" + #StopIfError "Failed to create .autorelabel on ${RSYNC_PATH}/${RSYNC_PREFIX}/backup" + fi + ;; + + (rsync) +- $BACKUP_PROG -a "${TMP_DIR}/selinux.autorelabel" ${BACKUP_RSYNC_OPTIONS[@]} \ ++ $BACKUP_PROG -a "${TMP_DIR}/selinux.autorelabel" "${BACKUP_RSYNC_OPTIONS[@]}" \ + "${RSYNC_PROTO}://${RSYNC_USER}@${RSYNC_HOST}:${RSYNC_PORT}/${RSYNC_PATH}/${RSYNC_PREFIX}/backup/.autorelabel" +- _rc=$? +- if [ $_rc -ne 0 ]; then +- LogPrint "Failed to create .autorelabel on ${RSYNC_PATH}/${RSYNC_PREFIX}/backup [${rsync_err_msg[$_rc]}]" ++ backup_prog_rc=$? ++ if [ $backup_prog_rc -ne 0 ]; then ++ LogPrint "Failed to create .autorelabel on ${RSYNC_PATH}/${RSYNC_PREFIX}/backup [${rsync_err_msg[$backup_prog_rc]}]" + #StopIfError "Failed to create .autorelabel on ${RSYNC_PATH}/${RSYNC_PREFIX}/backup" + fi + ;; +diff --git a/usr/share/rear/backup/RSYNC/GNU/Linux/620_force_autorelabel.sh b/usr/share/rear/backup/RSYNC/GNU/Linux/620_force_autorelabel.sh +index cae12e38..9a17d6bb 100644 +--- a/usr/share/rear/backup/RSYNC/GNU/Linux/620_force_autorelabel.sh ++++ b/usr/share/rear/backup/RSYNC/GNU/Linux/620_force_autorelabel.sh +@@ -1,3 +1,5 @@ ++local backup_prog_rc ++ + [ -f $TMP_DIR/force.autorelabel ] && { + + > "${TMP_DIR}/selinux.autorelabel" +@@ -11,19 +13,19 @@ + ssh $RSYNC_USER@$RSYNC_HOST "chmod $v 755 ${RSYNC_PATH}/${RSYNC_PREFIX}/backup" 2>/dev/null + $BACKUP_PROG -a "${TMP_DIR}/selinux.autorelabel" \ + "$RSYNC_USER@$RSYNC_HOST:${RSYNC_PATH}/${RSYNC_PREFIX}/backup/.autorelabel" 2>/dev/null +- _rc=$? +- if [ $_rc -ne 0 ]; then +- LogPrint "Failed to create .autorelabel on ${RSYNC_PATH}/${RSYNC_PREFIX}/backup [${rsync_err_msg[$_rc]}]" ++ backup_prog_rc=$? ++ if [ $backup_prog_rc -ne 0 ]; then ++ LogPrint "Failed to create .autorelabel on ${RSYNC_PATH}/${RSYNC_PREFIX}/backup [${rsync_err_msg[$backup_prog_rc]}]" + #StopIfError "Failed to create .autorelabel on ${RSYNC_PATH}/${RSYNC_PREFIX}/backup" + fi + ;; + + (rsync) +- $BACKUP_PROG -a "${TMP_DIR}/selinux.autorelabel" ${BACKUP_RSYNC_OPTIONS[@]} \ ++ $BACKUP_PROG -a "${TMP_DIR}/selinux.autorelabel" "${BACKUP_RSYNC_OPTIONS[@]}" \ + "${RSYNC_PROTO}://${RSYNC_USER}@${RSYNC_HOST}:${RSYNC_PORT}/${RSYNC_PATH}/${RSYNC_PREFIX}/backup/.autorelabel" +- _rc=$? +- if [ $_rc -ne 0 ]; then +- LogPrint "Failed to create .autorelabel on ${RSYNC_PATH}/${RSYNC_PREFIX}/backup [${rsync_err_msg[$_rc]}]" ++ backup_prog_rc=$? ++ if [ $backup_prog_rc -ne 0 ]; then ++ LogPrint "Failed to create .autorelabel on ${RSYNC_PATH}/${RSYNC_PREFIX}/backup [${rsync_err_msg[$backup_prog_rc]}]" + #StopIfError "Failed to create .autorelabel on ${RSYNC_PATH}/${RSYNC_PREFIX}/backup" + fi + ;; +@@ -35,8 +37,7 @@ + # probably using the BACKUP=NETFS workflow instead + if [ -d "${opath}" ]; then + if [ ! -f "${opath}/selinux.autorelabel" ]; then +- > "${opath}/selinux.autorelabel" +- StopIfError "Failed to create selinux.autorelabel on ${opath}" ++ > "${opath}/selinux.autorelabel" || Error "Failed to create selinux.autorelabel on ${opath}" + fi + fi + ;; +diff --git a/usr/share/rear/backup/RSYNC/default/200_check_rsync_relative_option.sh b/usr/share/rear/backup/RSYNC/default/200_check_rsync_relative_option.sh +index 60330007..cedee9ce 100644 +--- a/usr/share/rear/backup/RSYNC/default/200_check_rsync_relative_option.sh ++++ b/usr/share/rear/backup/RSYNC/default/200_check_rsync_relative_option.sh +@@ -4,7 +4,7 @@ + # check for the --relative option in BACKUP_RSYNC_OPTIONS array + # for the default values see the standard definition in conf/default.conf file + +-if ! grep -q relative <<< $(echo ${BACKUP_RSYNC_OPTIONS[@]}); then ++if ! grep -q relative <<< "${BACKUP_RSYNC_OPTIONS[*]}" ; then + BACKUP_RSYNC_OPTIONS+=( --relative ) + Log "Added option '--relative' to the BACKUP_RSYNC_OPTIONS array during $WORKFLOW workflow" + fi +diff --git a/usr/share/rear/backup/RSYNC/default/500_make_rsync_backup.sh b/usr/share/rear/backup/RSYNC/default/500_make_rsync_backup.sh +index 0d67d362..750a04ca 100644 +--- a/usr/share/rear/backup/RSYNC/default/500_make_rsync_backup.sh ++++ b/usr/share/rear/backup/RSYNC/default/500_make_rsync_backup.sh +@@ -2,6 +2,9 @@ + # This file is part of Relax-and-Recover, licensed under the GNU General + # Public License. Refer to the included COPYING for full text of license. + ++local backup_prog_rc ++local backup_log_message ++ + Log "Include list:" + while read -r ; do + Log " $REPLY" +@@ -11,9 +14,9 @@ while read -r ; do + Log " $REPLY" + done < $TMP_DIR/backup-exclude.txt + +-LogPrint "Creating $BACKUP_PROG archive on '${RSYNC_HOST}:${RSYNC_PATH}'" ++LogPrint "Creating $BACKUP_PROG backup on '${RSYNC_HOST}:${RSYNC_PATH}'" + +-ProgressStart "Running archive operation" ++ProgressStart "Running backup operation" + ( + case "$(basename $BACKUP_PROG)" in + +@@ -37,7 +40,7 @@ ProgressStart "Running archive operation" + ;; + + (*) +- # no other backup programs foreseen then rsync so far ++ # no other backup programs foreseen than rsync so far + : + ;; + +@@ -96,7 +99,7 @@ case "$(basename $BACKUP_PROG)" in + ;; + esac + +- ProgressInfo "Archived $((size/1024/1024)) MiB [avg $((size/1024/(SECONDS-starttime))) KiB/sec]" ++ ProgressInfo "Backed up $((size/1024/1024)) MiB [avg $((size/1024/(SECONDS-starttime))) KiB/sec]" + done + ;; + +@@ -113,24 +116,23 @@ ProgressStop + wait $BackupPID + + transfertime="$((SECONDS-starttime))" +-_rc="$(cat $TMP_DIR/retval)" ++backup_prog_rc="$(cat $TMP_DIR/retval)" + + sleep 1 + # everyone should see this warning, even if not verbose +-test "$_rc" -gt 0 && VERBOSE=1 LogPrint "WARNING ! +-There was an error (${rsync_err_msg[$_rc]}) during archive creation. +-Please check the archive and see '$RUNTIME_LOGFILE' for more information. ++test "$backup_prog_rc" -gt 0 && Error " ++There was an error (${rsync_err_msg[$backup_prog_rc]}) during backup creation. ++Please check the destination and see '$RUNTIME_LOGFILE' for more information. + +-Since errors are often related to files that cannot be saved by +-$BACKUP_PROG, we will continue the $WORKFLOW process. However, you MUST +-verify the backup yourself before trusting it ! ++If the error is related to files that cannot and should not be saved by ++$BACKUP_PROG, they should be excluded from the backup. + + " + +-_message="$(tail -14 ${TMP_DIR}/${BACKUP_PROG_ARCHIVE}.log)" +-if [ $_rc -eq 0 -a "$_message" ] ; then +- LogPrint "$_message in $transfertime seconds." ++backup_log_message="$(tail -14 ${TMP_DIR}/${BACKUP_PROG_ARCHIVE}.log)" ++if [ $backup_prog_rc -eq 0 -a "$backup_log_message" ] ; then ++ LogPrint "$backup_log_message in $transfertime seconds." + elif [ "$size" ]; then +- LogPrint "Archived $((size/1024/1024)) MiB in $((transfertime)) seconds [avg $((size/1024/transfertime)) KiB/sec]" ++ LogPrint "Backed up $((size/1024/1024)) MiB in $((transfertime)) seconds [avg $((size/1024/transfertime)) KiB/sec]" + fi + +diff --git a/usr/share/rear/backup/RSYNC/default/700_copy_backup_log.sh b/usr/share/rear/backup/RSYNC/default/700_copy_backup_log.sh +index 01801a4e..b90d459b 100644 +--- a/usr/share/rear/backup/RSYNC/default/700_copy_backup_log.sh ++++ b/usr/share/rear/backup/RSYNC/default/700_copy_backup_log.sh +@@ -1,6 +1,8 @@ + + # copy the backup.log & rear.log file to remote destination with timestamp added +-Timestamp=$( date +%Y%m%d.%H%M ) ++local timestamp ++ ++timestamp=$( date +%Y%m%d.%H%M ) + + # compress the log file first + gzip "$TMP_DIR/$BACKUP_PROG_ARCHIVE.log" || Error "Failed to 'gzip $TMP_DIR/$BACKUP_PROG_ARCHIVE.log'" +@@ -10,15 +12,15 @@ case $RSYNC_PROTO in + # FIXME: Add an explanatory comment why "2>/dev/null" is useful here + # or remove it according to https://github.com/rear/rear/issues/1395 + $BACKUP_PROG -a "${TMP_DIR}/${BACKUP_PROG_ARCHIVE}.log.gz" \ +- "${RSYNC_USER}@${RSYNC_HOST}:${RSYNC_PATH}/${RSYNC_PREFIX}/${BACKUP_PROG_ARCHIVE}-${Timestamp}.log.gz" 2>/dev/null ++ "${RSYNC_USER}@${RSYNC_HOST}:${RSYNC_PATH}/${RSYNC_PREFIX}/${BACKUP_PROG_ARCHIVE}-${timestamp}.log.gz" 2>/dev/null + +- $BACKUP_PROG -a "$RUNTIME_LOGFILE" "${RSYNC_USER}@${RSYNC_HOST}:${RSYNC_PATH}/${RSYNC_PREFIX}/rear-${Timestamp}.log" 2>/dev/null ++ $BACKUP_PROG -a "$RUNTIME_LOGFILE" "${RSYNC_USER}@${RSYNC_HOST}:${RSYNC_PATH}/${RSYNC_PREFIX}/rear-${timestamp}.log" 2>/dev/null + ;; + (rsync) +- $BACKUP_PROG -a "${TMP_DIR}/${BACKUP_PROG_ARCHIVE}.log.gz" ${BACKUP_RSYNC_OPTIONS[@]} \ +- "${RSYNC_PROTO}://${RSYNC_USER}@${RSYNC_HOST}:${RSYNC_PORT}/${RSYNC_PATH}/${RSYNC_PREFIX}/${BACKUP_PROG_ARCHIVE}-${Timestamp}.log.gz" ++ $BACKUP_PROG -a "${TMP_DIR}/${BACKUP_PROG_ARCHIVE}.log.gz" "${BACKUP_RSYNC_OPTIONS[@]}" \ ++ "${RSYNC_PROTO}://${RSYNC_USER}@${RSYNC_HOST}:${RSYNC_PORT}/${RSYNC_PATH}/${RSYNC_PREFIX}/${BACKUP_PROG_ARCHIVE}-${timestamp}.log.gz" + +- $BACKUP_PROG -a "$RUNTIME_LOGFILE" ${BACKUP_RSYNC_OPTIONS[@]} "${RSYNC_PROTO}://${RSYNC_USER}@${RSYNC_HOST}:${RSYNC_PORT}/${RSYNC_PATH}/${RSYNC_PREFIX}//rear-${Timestamp}.log" ++ $BACKUP_PROG -a "$RUNTIME_LOGFILE" "${BACKUP_RSYNC_OPTIONS[@]}" "${RSYNC_PROTO}://${RSYNC_USER}@${RSYNC_HOST}:${RSYNC_PORT}/${RSYNC_PATH}/${RSYNC_PREFIX}//rear-${timestamp}.log" + ;; + esac + +diff --git a/usr/share/rear/conf/default.conf b/usr/share/rear/conf/default.conf +index 455aa3ce..0c230f38 100644 +--- a/usr/share/rear/conf/default.conf ++++ b/usr/share/rear/conf/default.conf +@@ -1106,7 +1106,8 @@ BACKUP_ONLY_EXCLUDE="no" + MANUAL_INCLUDE=NO + # Disable SELinux policy during backup with NETFS or RSYNC (default yes) + BACKUP_SELINUX_DISABLE=1 +-# Enable integrity check of the backup archive (only with BACKUP=NETFS and BACKUP_PROG=tar) ++# Enable integrity check of the backup archive (full check only with BACKUP=NETFS and BACKUP_PROG=tar, ++# with BACKUP=rsync or BACKUP_PROG=rsync it only checks whether rsync completed the restore successfully) + BACKUP_INTEGRITY_CHECK= + # Define BACKUP_TYPE. + # By default BACKUP_TYPE is empty which means "rear mkbackup" will create a full backup. +diff --git a/usr/share/rear/output/RSYNC/default/200_make_prefix_dir.sh b/usr/share/rear/output/RSYNC/default/200_make_prefix_dir.sh +index 32ac391d..519febf5 100644 +--- a/usr/share/rear/output/RSYNC/default/200_make_prefix_dir.sh ++++ b/usr/share/rear/output/RSYNC/default/200_make_prefix_dir.sh +@@ -2,21 +2,19 @@ + # RSYNC_PREFIX=$HOSTNAME as set in default.conf + + # create temporary local work-spaces to collect files (we already make the remote backup dir with the correct mode!!) +-mkdir -p $v -m0750 "${TMP_DIR}/rsync/${RSYNC_PREFIX}" >&2 +-StopIfError "Could not mkdir '${TMP_DIR}/rsync/${RSYNC_PREFIX}'" +-mkdir -p $v -m0755 "${TMP_DIR}/rsync/${RSYNC_PREFIX}/backup" >&2 +-StopIfError "Could not mkdir '${TMP_DIR}/rsync/${RSYNC_PREFIX}/backup'" ++mkdir -p $v -m0750 "${TMP_DIR}/rsync/${RSYNC_PREFIX}" >&2 || Error "Could not mkdir '${TMP_DIR}/rsync/${RSYNC_PREFIX}'" ++mkdir -p $v -m0755 "${TMP_DIR}/rsync/${RSYNC_PREFIX}/backup" >&2 || Error "Could not mkdir '${TMP_DIR}/rsync/${RSYNC_PREFIX}/backup'" + + case $RSYNC_PROTO in + + (ssh) +- $BACKUP_PROG -a $v -r "${TMP_DIR}/rsync/${RSYNC_PREFIX}" "${RSYNC_USER}@${RSYNC_HOST}:${RSYNC_PATH}" >/dev/null 2>&1 +- StopIfError "Could not create '${RSYNC_PATH}/${RSYNC_PREFIX}' on remote ${RSYNC_HOST}" ++ $BACKUP_PROG -a $v -r "${TMP_DIR}/rsync/${RSYNC_PREFIX}" "${RSYNC_USER}@${RSYNC_HOST}:${RSYNC_PATH}" >/dev/null 2>&1 \ ++ || Error "Could not create '${RSYNC_PATH}/${RSYNC_PREFIX}' on remote ${RSYNC_HOST}" + ;; + + (rsync) +- $BACKUP_PROG -a $v -r "${TMP_DIR}/rsync/${RSYNC_PREFIX}" ${BACKUP_RSYNC_OPTIONS[@]} "${RSYNC_PROTO}://${RSYNC_USER}@${RSYNC_HOST}:${RSYNC_PORT}/${RSYNC_PATH}/" >/dev/null +- StopIfError "Could not create '${RSYNC_PATH}/${RSYNC_PREFIX}' on remote ${RSYNC_HOST}" ++ $BACKUP_PROG -a $v -r "${TMP_DIR}/rsync/${RSYNC_PREFIX}" "${BACKUP_RSYNC_OPTIONS[@]}" "${RSYNC_PROTO}://${RSYNC_USER}@${RSYNC_HOST}:${RSYNC_PORT}/${RSYNC_PATH}/" >/dev/null \ ++ || Error "Could not create '${RSYNC_PATH}/${RSYNC_PREFIX}' on remote ${RSYNC_HOST}" + ;; + + esac +diff --git a/usr/share/rear/output/RSYNC/default/900_copy_result_files.sh b/usr/share/rear/output/RSYNC/default/900_copy_result_files.sh +index c7b430d8..96b62da1 100644 +--- a/usr/share/rear/output/RSYNC/default/900_copy_result_files.sh ++++ b/usr/share/rear/output/RSYNC/default/900_copy_result_files.sh +@@ -5,19 +5,19 @@ LogPrint "Copying resulting files to $OUTPUT_URL location" + + # if called as mkbackuponly then we just don't have any result files. + if test "$RESULT_FILES" ; then +- Log "Copying files '${RESULT_FILES[@]}' to $OUTPUT_URL location" +- cp $v "${RESULT_FILES[@]}" "${TMP_DIR}/rsync/${RSYNC_PREFIX}/" +- StopIfError "Could not copy files to local rsync location" ++ Log "Copying files '${RESULT_FILES[*]}' to $OUTPUT_URL location" ++ cp $v "${RESULT_FILES[@]}" "${TMP_DIR}/rsync/${RSYNC_PREFIX}/" \ ++ || Error "Could not copy files to local rsync location" + fi + +-echo "$VERSION_INFO" >"${TMP_DIR}/rsync/${RSYNC_PREFIX}/VERSION" +-StopIfError "Could not create VERSION file on local rsync location" ++echo "$VERSION_INFO" >"${TMP_DIR}/rsync/${RSYNC_PREFIX}/VERSION" \ ++ || Error "Could not create VERSION file on local rsync location" + +-cp $v $(get_template "RESULT_usage_$OUTPUT.txt") "${TMP_DIR}/rsync/${RSYNC_PREFIX}/README" +-StopIfError "Could not copy usage file to local rsync location" ++cp $v $(get_template "RESULT_usage_$OUTPUT.txt") "${TMP_DIR}/rsync/${RSYNC_PREFIX}/README" \ ++ || Error "Could not copy usage file to local rsync location" + +-cat "$RUNTIME_LOGFILE" >"${TMP_DIR}/rsync/${RSYNC_PREFIX}/rear.log" +-StopIfError "Could not copy $RUNTIME_LOGFILE to local rsync location" ++cat "$RUNTIME_LOGFILE" >"${TMP_DIR}/rsync/${RSYNC_PREFIX}/rear.log" \ ++ || Error "Could not copy $RUNTIME_LOGFILE to local rsync location" + + case $RSYNC_PROTO in + +@@ -25,20 +25,20 @@ case $RSYNC_PROTO in + Log "$BACKUP_PROG -a ${TMP_DIR}/rsync/${RSYNC_PREFIX}/ ${RSYNC_USER}@${RSYNC_HOST}:${RSYNC_PATH}/${RSYNC_PREFIX}/" + # FIXME: Add an explanatory comment why "2>/dev/null" is useful here + # or remove it according to https://github.com/rear/rear/issues/1395 +- $BACKUP_PROG -a "${TMP_DIR}/rsync/${RSYNC_PREFIX}/" "${RSYNC_USER}@${RSYNC_HOST}:${RSYNC_PATH}/${RSYNC_PREFIX}/" 2>/dev/null +- StopIfError "Could not copy '${RESULT_FILES[@]}' to $OUTPUT_URL location" ++ $BACKUP_PROG -a "${TMP_DIR}/rsync/${RSYNC_PREFIX}/" "${RSYNC_USER}@${RSYNC_HOST}:${RSYNC_PATH}/${RSYNC_PREFIX}/" 2>/dev/null \ ++ || Error "Could not copy '${RESULT_FILES[*]}' to $OUTPUT_URL location" + ;; + + (rsync) +- Log "$BACKUP_PROG -a ${TMP_DIR}/rsync/${RSYNC_PREFIX}/ ${BACKUP_RSYNC_OPTIONS[@]} ${RSYNC_PROTO}://${RSYNC_USER}@${RSYNC_HOST}:${RSYNC_PORT}/${RSYNC_PATH}/${RSYNC_PREFIX}/" ++ Log "$BACKUP_PROG -a ${TMP_DIR}/rsync/${RSYNC_PREFIX}/ ${BACKUP_RSYNC_OPTIONS[*]} ${RSYNC_PROTO}://${RSYNC_USER}@${RSYNC_HOST}:${RSYNC_PORT}/${RSYNC_PATH}/${RSYNC_PREFIX}/" + # FIXME: Add an explanatory comment why "2>/dev/null" is useful here + # or remove it according to https://github.com/rear/rear/issues/1395 +- $BACKUP_PROG -a "${TMP_DIR}/rsync/${RSYNC_PREFIX}/" ${BACKUP_RSYNC_OPTIONS[@]} "${RSYNC_PROTO}://${RSYNC_USER}@${RSYNC_HOST}:${RSYNC_PORT}/${RSYNC_PATH}/${RSYNC_PREFIX}/" 2>/dev/null +- StopIfError "Could not copy '${RESULT_FILES[@]}' to $OUTPUT_URL location" ++ $BACKUP_PROG -a "${TMP_DIR}/rsync/${RSYNC_PREFIX}/" "${BACKUP_RSYNC_OPTIONS[@]}" "${RSYNC_PROTO}://${RSYNC_USER}@${RSYNC_HOST}:${RSYNC_PORT}/${RSYNC_PATH}/${RSYNC_PREFIX}/" 2>/dev/null \ ++ || Error "Could not copy '${RESULT_FILES[*]}' to $OUTPUT_URL location" + ;; + + esac + + # cleanup the temporary space (need it for the log file during backup) +-rm -rf "${TMP_DIR}/rsync/${RSYNC_PREFIX}/" +-LogIfError "Could not cleanup temoprary rsync space: ${TMP_DIR}/rsync/${RSYNC_PREFIX}/" ++rm -rf "${TMP_DIR}/rsync/${RSYNC_PREFIX}/" \ ++ || Log "Could not cleanup temporary rsync space: ${TMP_DIR}/rsync/${RSYNC_PREFIX}/" +diff --git a/usr/share/rear/prep/NETFS/default/400_automatic_exclude_recreate.sh b/usr/share/rear/prep/NETFS/default/400_automatic_exclude_recreate.sh +index fadf9d72..3c719c44 100644 +--- a/usr/share/rear/prep/NETFS/default/400_automatic_exclude_recreate.sh ++++ b/usr/share/rear/prep/NETFS/default/400_automatic_exclude_recreate.sh +@@ -31,7 +31,7 @@ case $scheme in + backup_directory_mountpoint=$( df -P "$backup_directory" | tail -1 | awk '{print $6}' ) + test "/" = "$backup_directory_mountpoint" && Error "URL '$BACKUP_URL' has the backup directory '$backup_directory' in the '/' filesystem which is forbidden." + # When the mountpoint of the backup directory is not yet excluded add its mountpoint to the EXCLUDE_RECREATE array: +- if ! grep -q "$backup_directory_mountpoint" <<< $( echo ${EXCLUDE_RECREATE[@]} ) ; then ++ if ! grep -q "$backup_directory_mountpoint" <<< "${EXCLUDE_RECREATE[*]}" ; then + EXCLUDE_RECREATE+=( "fs:$backup_directory_mountpoint" ) + fi + ;; +diff --git a/usr/share/rear/prep/RSYNC/GNU/Linux/200_selinux_in_use.sh b/usr/share/rear/prep/RSYNC/GNU/Linux/200_selinux_in_use.sh +index ac26edfa..eb7df29e 100644 +--- a/usr/share/rear/prep/RSYNC/GNU/Linux/200_selinux_in_use.sh ++++ b/usr/share/rear/prep/RSYNC/GNU/Linux/200_selinux_in_use.sh +@@ -33,7 +33,7 @@ case $(basename $BACKUP_PROG) in + touch $TMP_DIR/force.autorelabel # after reboot the restored system do a forced SELinux relabeling + else + # if --xattrs is already set; no need to do it again +- if ! grep -q xattrs <<< $(echo ${BACKUP_RSYNC_OPTIONS[@]}); then ++ if ! grep -q xattrs <<< "${BACKUP_RSYNC_OPTIONS[*]}" ; then + BACKUP_RSYNC_OPTIONS+=( --xattrs ) + fi + RSYNC_SELINUX=1 # variable used in recover mode (means using xattr and not disable SELinux) +diff --git a/usr/share/rear/prep/RSYNC/default/100_check_rsync.sh b/usr/share/rear/prep/RSYNC/default/100_check_rsync.sh +index b8535352..c964a148 100644 +--- a/usr/share/rear/prep/RSYNC/default/100_check_rsync.sh ++++ b/usr/share/rear/prep/RSYNC/default/100_check_rsync.sh +@@ -33,22 +33,20 @@ RSYNC_PORT=873 # default port (of rsync server) + RSYNC_PATH= + + +-echo $BACKUP_URL | egrep -q '(::)' # new style '::' means rsync protocol +-if [[ $? -eq 0 ]]; then ++if egrep -q '(::)' <<< $BACKUP_URL ; then # new style '::' means rsync protocol + RSYNC_PROTO=rsync + else + RSYNC_PROTO=ssh + fi + +-echo $host | grep -q '@' +-if [[ $? -eq 0 ]]; then ++if grep -q '@' <<< $host ; then + RSYNC_USER="${host%%@*}" # grab user name + else + RSYNC_USER=root + fi + + # remove USER@ if present (we don't need it anymore) +-tmp2="${host#*@}" ++local tmp2="${host#*@}" + + case "$RSYNC_PROTO" in + +@@ -56,8 +54,7 @@ case "$RSYNC_PROTO" in + # tmp2=witsbebelnx02::backup or tmp2=witsbebelnx02:: + RSYNC_HOST="${tmp2%%::*}" + # path=/gdhaese1@witsbebelnx02::backup or path=/backup +- echo $path | grep -q '::' +- if [[ $? -eq 0 ]]; then ++ if grep -q '::' <<< $path ; then + RSYNC_PATH="${path##*::}" + else + RSYNC_PATH="${path##*/}" +@@ -79,8 +76,7 @@ esac + + # check if host is reachable + if test "$PING" ; then +- ping -c 2 "$RSYNC_HOST" >/dev/null +- StopIfError "Backup host [$RSYNC_HOST] not reachable." ++ ping -c 2 "$RSYNC_HOST" >/dev/null || Error "Backup host [$RSYNC_HOST] not reachable." + else + Log "Skipping ping test" + fi +@@ -89,15 +85,15 @@ fi + case "$RSYNC_PROTO" in + + (rsync) +- Log "Test: $BACKUP_PROG ${BACKUP_RSYNC_OPTIONS[@]} ${RSYNC_PROTO}://${RSYNC_USER}@${RSYNC_HOST}:${RSYNC_PORT}/" +- $BACKUP_PROG ${BACKUP_RSYNC_OPTIONS[@]} ${RSYNC_PROTO}://${RSYNC_USER}@${RSYNC_HOST}:${RSYNC_PORT}/ >/dev/null +- StopIfError "Rsync daemon not running on $RSYNC_HOST" ++ Log "Test: $BACKUP_PROG ${BACKUP_RSYNC_OPTIONS[*]} ${RSYNC_PROTO}://${RSYNC_USER}@${RSYNC_HOST}:${RSYNC_PORT}/" ++ $BACKUP_PROG "${BACKUP_RSYNC_OPTIONS[@]}" ${RSYNC_PROTO}://${RSYNC_USER}@${RSYNC_HOST}:${RSYNC_PORT}/ >/dev/null \ ++ || Error "Rsync daemon not running on $RSYNC_HOST" + ;; + + (ssh) + Log "Test: ssh ${RSYNC_USER}@${RSYNC_HOST} /bin/true" +- ssh ${RSYNC_USER}@${RSYNC_HOST} /bin/true >/dev/null 2>&1 +- StopIfError "Secure shell connection not setup properly [$RSYNC_USER@$RSYNC_HOST]" ++ ssh ${RSYNC_USER}@${RSYNC_HOST} /bin/true >/dev/null 2>&1 \ ++ || Error "Secure shell connection not setup properly [$RSYNC_USER@$RSYNC_HOST]" + ;; + + esac +diff --git a/usr/share/rear/prep/RSYNC/default/150_check_rsync_protocol_version.sh b/usr/share/rear/prep/RSYNC/default/150_check_rsync_protocol_version.sh +index 446dd736..e9103531 100644 +--- a/usr/share/rear/prep/RSYNC/default/150_check_rsync_protocol_version.sh ++++ b/usr/share/rear/prep/RSYNC/default/150_check_rsync_protocol_version.sh +@@ -2,15 +2,17 @@ + # This file is part of Relax-and-Recover, licensed under the GNU General + # Public License. Refer to the included COPYING for full text of license. + # try to grab the rsync protocol version of rsync on the remote server ++ ++local remote_mountpoint ++ + if [ -z "$RSYNC_PROTOCOL_VERSION" ]; then + + case $RSYNC_PROTO in + + (ssh) +- ssh ${RSYNC_USER}@${RSYNC_HOST} rsync --version >"$TMP_DIR/rsync_protocol" 2>&1 +- StopIfError "Secure shell connection not setup properly [$RSYNC_USER@$RSYNC_HOST]" +- grep -q "protocol version" "$TMP_DIR/rsync_protocol" +- if [ $? -eq 0 ]; then ++ ssh ${RSYNC_USER}@${RSYNC_HOST} rsync --version >"$TMP_DIR/rsync_protocol" 2>&1 \ ++ || Error "Secure shell connection not setup properly [$RSYNC_USER@$RSYNC_HOST]" ++ if grep -q "protocol version" "$TMP_DIR/rsync_protocol" ; then + RSYNC_PROTOCOL_VERSION=$(grep 'protocol version' "$TMP_DIR/rsync_protocol" | awk '{print $6}') + else + RSYNC_PROTOCOL_VERSION=29 # being conservative (old rsync version < 3.0) +@@ -30,25 +32,21 @@ else + + fi + +-if [ "${RSYNC_USER}" != "root" ]; then ++if [ "${RSYNC_USER}" != "root" -a $RSYNC_PROTO = "ssh" ]; then + if [ $RSYNC_PROTOCOL_VERSION -gt 29 ]; then + if grep -q "no xattrs" "$TMP_DIR/rsync_protocol"; then + # no xattrs available in remote rsync, so --fake-super is not possible + Error "rsync --fake-super not possible on system ($RSYNC_HOST) (no xattrs compiled in rsync)" + else + # when using --fake-super we must have user_xattr mount options on the remote mntpt +- _mntpt=$(ssh ${RSYNC_USER}@${RSYNC_HOST} 'cd ${RSYNC_PATH}; df -P .' 2>/dev/null | tail -1 | awk '{print $6}') +- ssh ${RSYNC_USER}@${RSYNC_HOST} "cd ${RSYNC_PATH} && touch .is_xattr_supported && setfattr -n user.comment -v 'File created by ReaR to test if this filesystems supports extended attributes.' .is_xattr_supported && getfattr -n user.comment .is_xattr_supported 1>/dev/null; find .is_xattr_supported -empty -delete" +- StopIfError "Remote file system $_mntpt does not have user_xattr mount option set!" +- #BACKUP_RSYNC_OPTIONS+=( --xattrs --rsync-path="""rsync --fake-super""" ) ++ remote_mountpoint=$(ssh ${RSYNC_USER}@${RSYNC_HOST} 'cd ${RSYNC_PATH}; df -P .' 2>/dev/null | tail -1 | awk '{print $6}') ++ ssh ${RSYNC_USER}@${RSYNC_HOST} "cd ${RSYNC_PATH} && touch .is_xattr_supported && setfattr -n user.comment -v 'File created by ReaR to test if this filesystems supports extended attributes.' .is_xattr_supported && getfattr -n user.comment .is_xattr_supported 1>/dev/null; find .is_xattr_supported -empty -delete" \ ++ || Error "Remote file system $remote_mountpoint does not have user_xattr mount option set!" ++ #BACKUP_RSYNC_OPTIONS+=( --xattrs --rsync-path="rsync --fake-super" ) + # see issue #366 for explanation of removing --xattrs +- BACKUP_RSYNC_OPTIONS+=( --rsync-path="""rsync --fake-super""" ) ++ BACKUP_RSYNC_OPTIONS+=( --rsync-path="rsync --fake-super" ) + fi + else +- if [ ${BACKUP_RSYNC_OPTIONS[@]/--fake-super/} != ${BACKUP_RSUNC_OPTIONS[@]} ]; then +- Error "rsync --fake-super not possible on system ($RSYNC_HOST) (please upgrade rsync to 3.x)" +- else +- Log "Warning: rsync --fake-super not possible on system ($RSYNC_HOST) (please upgrade rsync to 3.x)" +- fi ++ Error "rsync --fake-super not possible on system ($RSYNC_HOST) (please upgrade rsync to 3.x)" + fi + fi +diff --git a/usr/share/rear/restore/DUPLICITY/default/400_restore_duplicity.sh b/usr/share/rear/restore/DUPLICITY/default/400_restore_duplicity.sh +index 0a9c9648..220ccc57 100644 +--- a/usr/share/rear/restore/DUPLICITY/default/400_restore_duplicity.sh ++++ b/usr/share/rear/restore/DUPLICITY/default/400_restore_duplicity.sh +@@ -5,6 +5,8 @@ + # Restore from remote backup via DUPLICIY over rsync + + if [ "$BACKUP_PROG" = "duplicity" ]; then ++ local backup_prog_rc ++ local restore_log_message + + LogPrint "========================================================================" + LogPrint "Restoring backup with $BACKUP_PROG from '$BACKUP_DUPLICITY_URL'" +@@ -49,7 +51,8 @@ if [ "$BACKUP_PROG" = "duplicity" ]; then + LogPrint "with CMD: $DUPLICITY_PROG -v 5 $GPG_KEY --force --tempdir=$DUPLICITY_TEMPDIR $BACKUP_DUPLICITY_URL/$HOSTNAME/ $TARGET_FS_ROOT" + $DUPLICITY_PROG -v 5 $GPG_KEY --force --tempdir="$DUPLICITY_TEMPDIR" $BACKUP_DUPLICITY_URL/$HOSTNAME/ $TARGET_FS_ROOT 0<&6 | tee $TMP_DIR/duplicity-restore.log + fi +- _rc=$? ++ # FIXME: this collects the exit code from "tee", not from $DUPLICITY_PROG ++ backup_prog_rc=$? + + transfertime="$((SECONDS-$starttime))" + sleep 1 +@@ -65,20 +68,20 @@ if [ "$BACKUP_PROG" = "duplicity" ]; then + LogPrint "========================================================================" + + +- if [ "$_rc" -gt 0 ]; then ++ if [ "$backup_prog_rc" -gt 0 ]; then + LogPrint "WARNING ! + There was an error while restoring the archive. + Please check '$RUNTIME_LOGFILE' and $TMP_DIR/duplicity-restore.log for more information. + You should also manually check the restored system to see whether it is complete. + " + +- _message="$(tail -14 ${TMP_DIR}/duplicity-restore.log)" ++ restore_log_message="$(tail -14 ${TMP_DIR}/duplicity-restore.log)" + + LogPrint "Last 14 Lines of ${TMP_DIR}/duplicity-restore.log:" +- LogPrint "$_message" ++ LogPrint "$restore_log_message" + fi + +- if [ $_rc -eq 0 ] ; then ++ if [ $backup_prog_rc -eq 0 ] ; then + LogPrint "Restore completed in $transfertime seconds." + fi + +diff --git a/usr/share/rear/restore/RBME/default/400_restore_backup.sh b/usr/share/rear/restore/RBME/default/400_restore_backup.sh +index 28a3c354..3e97e16b 100644 +--- a/usr/share/rear/restore/RBME/default/400_restore_backup.sh ++++ b/usr/share/rear/restore/RBME/default/400_restore_backup.sh +@@ -2,6 +2,8 @@ if [[ -z "$RBME_BACKUP" ]] ; then + Error "No RBME backup selected (BACKUP_URL?). Aborting." + fi + ++local backup_prog_rc ++ + scheme=$(url_scheme "$BACKUP_URL") + + LogPrint "Restoring from backup $RBME_BACKUP." +@@ -43,11 +45,11 @@ transfertime="$((SECONDS-starttime))" + # harvest return code from background job. The kill -0 $BackupPID loop above should + # have made sure that this wait won't do any real "waiting" :-) + wait $BackupPID +-_rc=$? ++backup_prog_rc=$? + + sleep 1 +-test "$_rc" -gt 0 && LogPrint "WARNING ! +-There was an error (${rsync_err_msg[$_rc]}) while restoring the archive. ++test "$backup_prog_rc" -gt 0 && LogPrint "WARNING ! ++There was an error (${rsync_err_msg[$backup_prog_rc]}) while restoring the archive. + Please check '$RUNTIME_LOGFILE' for more information. You should also + manually check the restored system to see whether it is complete. + " +diff --git a/usr/share/rear/restore/RSYNC/default/200_remove_relative_rsync_option.sh b/usr/share/rear/restore/RSYNC/default/200_remove_relative_rsync_option.sh +index 53915322..a792f195 100644 +--- a/usr/share/rear/restore/RSYNC/default/200_remove_relative_rsync_option.sh ++++ b/usr/share/rear/restore/RSYNC/default/200_remove_relative_rsync_option.sh +@@ -4,11 +4,11 @@ + # without the --relative option ; my feeling says it is better to remove it from array BACKUP_RSYNC_OPTIONS + # If I'm wrong please let us know (use issue mentioned above to comment) + +-if grep -q relative <<< $(echo ${BACKUP_RSYNC_OPTIONS[@]}); then ++if grep -q -- "--relative" <<< "${BACKUP_RSYNC_OPTIONS[*]}" ; then + BACKUP_RSYNC_OPTIONS=( $( RmInArray "--relative" "${BACKUP_RSYNC_OPTIONS[@]}" ) ) + Log "Removed option '--relative' from the BACKUP_RSYNC_OPTIONS array during $WORKFLOW workflow" + fi +-if grep -q "-R" <<< $(echo ${BACKUP_RSYNC_OPTIONS[@]}); then ++if grep -q -- "-R" <<< "${BACKUP_RSYNC_OPTIONS[*]}" ; then + BACKUP_RSYNC_OPTIONS=( $( RmInArray "-R" "${BACKUP_RSYNC_OPTIONS[@]}" ) ) + Log "Removed option '-R' from the BACKUP_RSYNC_OPTIONS array during $WORKFLOW workflow" + fi +diff --git a/usr/share/rear/restore/RSYNC/default/400_restore_rsync_backup.sh b/usr/share/rear/restore/RSYNC/default/400_restore_rsync_backup.sh +index 2a0bf15e..993088be 100644 +--- a/usr/share/rear/restore/RSYNC/default/400_restore_rsync_backup.sh ++++ b/usr/share/rear/restore/RSYNC/default/400_restore_rsync_backup.sh +@@ -4,10 +4,10 @@ get_size() { + echo $( stat --format '%s' "$TARGET_FS_ROOT/$1" ) + } + +-mkdir -p "${TMP_DIR}/rsync/${NETFS_PREFIX}" +-StopIfError "Could not mkdir '$TMP_DIR/rsync/${NETFS_PREFIX}'" ++local backup_prog_rc ++local restore_log_message + +-LogPrint "Restoring $BACKUP_PROG archive from '${RSYNC_HOST}:${RSYNC_PATH}'" ++LogPrint "Restoring $BACKUP_PROG backup from '${RSYNC_HOST}:${RSYNC_PATH}'" + + ProgressStart "Restore operation" + ( +@@ -33,9 +33,10 @@ ProgressStart "Restore operation" + ;; + + (*) +- # no other backup programs foreseen then rsync so far ++ # no other backup programs foreseen than rsync so far + : + ;; ++ + esac + echo $? >$TMP_DIR/retval + ) >"${TMP_DIR}/${BACKUP_PROG_ARCHIVE}-restore.log" & +@@ -65,6 +66,7 @@ case "$(basename $BACKUP_PROG)" in + ProgressStep + done + ;; ++ + esac + ProgressStop + +@@ -72,20 +74,28 @@ transfertime="$((SECONDS-starttime))" + + # harvest return code from background job. The kill -0 $BackupPID loop above should + # have made sure that this wait won't do any real "waiting" :-) +-wait $BackupPID +-_rc=$? ++wait $BackupPID || LogPrintError "Restore job returned a nonzero exit code $?" ++# harvest the actual return code of rsync. Finishing the pipeline with an error code above is actually unlikely, ++# because rsync is not the last command in it. But error returns from rsync are common and must be handled. ++backup_prog_rc="$(cat $TMP_DIR/retval)" + + sleep 1 +-test "$_rc" -gt 0 && LogPrint "WARNING ! +-There was an error (${rsync_err_msg[$_rc]}) while restoring the archive. ++if test "$backup_prog_rc" -gt 0 ; then ++ # TODO: Shouldn't we tell the user to check ${TMP_DIR}/${BACKUP_PROG_ARCHIVE}-restore.log as well? ++ LogPrintError "WARNING ! ++There was an error (${rsync_err_msg[$backup_prog_rc]}) while restoring the backup. + Please check '$RUNTIME_LOGFILE' for more information. You should also + manually check the restored system to see whether it is complete. + " ++ is_true "$BACKUP_INTEGRITY_CHECK" && Error "Integrity check failed, restore aborted because BACKUP_INTEGRITY_CHECK is enabled" ++fi + +-_message="$(tail -14 ${TMP_DIR}/${BACKUP_PROG_ARCHIVE}-restore.log)" ++restore_log_message="$(tail -14 ${TMP_DIR}/${BACKUP_PROG_ARCHIVE}-restore.log)" + +-if [ $_rc -eq 0 -a "$_message" ] ; then +- LogPrint "$_message in $transfertime seconds." ++if [ $backup_prog_rc -eq 0 -a "$restore_log_message" ] ; then ++ LogPrint "$restore_log_message in $transfertime seconds." + elif [ "$size" ]; then + LogPrint "Restored $((size/1024/1024)) MiB in $((transfertime)) seconds [avg $((size/1024/transfertime)) KiB/sec]" + fi ++ ++return $backup_prog_rc +diff --git a/usr/share/rear/verify/RSYNC/GNU/Linux/600_check_rsync_xattr.sh b/usr/share/rear/verify/RSYNC/GNU/Linux/600_check_rsync_xattr.sh +index 3622884a..890161f1 100644 +--- a/usr/share/rear/verify/RSYNC/GNU/Linux/600_check_rsync_xattr.sh ++++ b/usr/share/rear/verify/RSYNC/GNU/Linux/600_check_rsync_xattr.sh +@@ -3,8 +3,8 @@ + [[ $RSYNC_SELINUX ]] && { + + # if --xattrs is already set; no need to do it again +- if ! grep -q xattrs <<< $(echo ${BACKUP_RSYNC_OPTIONS[@]}); then +- RSYNC_OPTIONS=( "${BACKUP_RSYNC_OPTIONS[@]}" --xattrs ) ++ if ! grep -q xattrs <<< "${BACKUP_RSYNC_OPTIONS[*]}" ; then ++ BACKUP_RSYNC_OPTIONS+=( --xattrs ) + fi + + } +diff --git a/usr/share/rear/verify/RSYNC/default/550_check_remote_backup_archive.sh b/usr/share/rear/verify/RSYNC/default/550_check_remote_backup_archive.sh +index 47ed9e02..b2fb72f5 100644 +--- a/usr/share/rear/verify/RSYNC/default/550_check_remote_backup_archive.sh ++++ b/usr/share/rear/verify/RSYNC/default/550_check_remote_backup_archive.sh +@@ -3,12 +3,12 @@ + case $RSYNC_PROTO in + + (ssh) +- ssh ${RSYNC_USER}@${RSYNC_HOST} "ls -ld ${RSYNC_PATH}/${RSYNC_PREFIX}/backup" >/dev/null 2>&1 +- StopIfError "Archive not found on [$RSYNC_USER@$RSYNC_HOST:${RSYNC_PATH}/${RSYNC_PREFIX}]" ++ ssh ${RSYNC_USER}@${RSYNC_HOST} "ls -ld ${RSYNC_PATH}/${RSYNC_PREFIX}/backup" >/dev/null 2>&1 \ ++ || Error "Archive not found on [$RSYNC_USER@$RSYNC_HOST:${RSYNC_PATH}/${RSYNC_PREFIX}]" + ;; + + (rsync) +- $BACKUP_PROG "${RSYNC_PROTO}://${RSYNC_USER}@${RSYNC_HOST}:${RSYNC_PORT}/${RSYNC_PATH}/${RSYNC_PREFIX}/backup" >/dev/null 2>&1 +- StopIfError "Archive not found on [$RSYNC_USER@$RSYNC_HOST:${RSYNC_PATH}/${RSYNC_PREFIX}]" ++ $BACKUP_PROG "${RSYNC_PROTO}://${RSYNC_USER}@${RSYNC_HOST}:${RSYNC_PORT}/${RSYNC_PATH}/${RSYNC_PREFIX}/backup" >/dev/null 2>&1 \ ++ || Error "Archive not found on [$RSYNC_USER@$RSYNC_HOST:${RSYNC_PATH}/${RSYNC_PREFIX}]" + ;; + esac diff --git a/SOURCES/rear-bz1945869.patch b/SOURCES/rear-bz1945869.patch new file mode 100644 index 0000000..a17a7cd --- /dev/null +++ b/SOURCES/rear-bz1945869.patch @@ -0,0 +1,274 @@ +diff --git a/usr/share/rear/finalize/Linux-i386/670_run_efibootmgr.sh b/usr/share/rear/finalize/Linux-i386/670_run_efibootmgr.sh +old mode 100644 +new mode 100755 +index cc646359..33d87767 +--- a/usr/share/rear/finalize/Linux-i386/670_run_efibootmgr.sh ++++ b/usr/share/rear/finalize/Linux-i386/670_run_efibootmgr.sh +@@ -8,6 +8,10 @@ is_true $USING_UEFI_BOOTLOADER || return 0 + # (cf. finalize/Linux-i386/610_EFISTUB_run_efibootmgr.sh): + is_true $EFI_STUB && return + ++LogPrint "Creating EFI Boot Manager entries..." ++ ++local esp_mountpoint esp_mountpoint_inside boot_efi_parts boot_efi_dev ++ + # When UEFI_BOOTLOADER is not a regular file in the restored target system + # (cf. how esp_mountpoint is set below) it means BIOS is used + # (cf. rescue/default/850_save_sysfs_uefi_vars.sh) +@@ -15,64 +19,80 @@ is_true $EFI_STUB && return + # because when UEFI_BOOTLOADER is empty the test below evaluates to + # test -f /mnt/local/ + # which also returns false because /mnt/local/ is a directory +-# (cf. https://github.com/rear/rear/pull/2051/files#r258826856): +-test -f "$TARGET_FS_ROOT/$UEFI_BOOTLOADER" || return 0 ++# (cf. https://github.com/rear/rear/pull/2051/files#r258826856) ++# but using BIOS conflicts with USING_UEFI_BOOTLOADER is true ++# i.e. we should create EFI Boot Manager entries but we cannot: ++if ! test -f "$TARGET_FS_ROOT/$UEFI_BOOTLOADER" ; then ++ LogPrintError "Failed to create EFI Boot Manager entries (UEFI bootloader '$UEFI_BOOTLOADER' not found under target $TARGET_FS_ROOT)" ++ return 1 ++fi + + # Determine where the EFI System Partition (ESP) is mounted in the currently running recovery system: +-esp_mountpoint=$( df -P "$TARGET_FS_ROOT/$UEFI_BOOTLOADER" | tail -1 | awk '{print $6}' ) +-# Use TARGET_FS_ROOT/boot/efi as fallback ESP mountpoint: +-test "$esp_mountpoint" || esp_mountpoint="$TARGET_FS_ROOT/boot/efi" ++esp_mountpoint=$( filesystem_name "$TARGET_FS_ROOT/$UEFI_BOOTLOADER" ) ++# Use TARGET_FS_ROOT/boot/efi as fallback ESP mountpoint (filesystem_name returns "/" ++# if mountpoint not found otherwise): ++if [ "$esp_mountpoint" = "/" ] ; then ++ esp_mountpoint="$TARGET_FS_ROOT/boot/efi" ++ LogPrint "Mountpoint of $TARGET_FS_ROOT/$UEFI_BOOTLOADER not found, trying $esp_mountpoint" ++fi + + # Skip if there is no esp_mountpoint directory (e.g. the fallback ESP mountpoint may not exist). + # Double quotes are mandatory here because 'test -d' without any (possibly empty) argument results true: +-test -d "$esp_mountpoint" || return 0 +- +-BootEfiDev="$( mount | grep "$esp_mountpoint" | awk '{print $1}' )" +-# /dev/sda1 or /dev/mapper/vol34_part2 or /dev/mapper/mpath99p4 +-Dev=$( get_device_name $BootEfiDev ) +-# 1 (must anyway be a low nr <9) +-ParNr=$( get_partition_number $Dev ) +-# /dev/sda or /dev/mapper/vol34_part or /dev/mapper/mpath99p or /dev/mmcblk0p +-Disk=$( echo ${Dev%$ParNr} ) +- +-# Strip trailing partition remainders like '_part' or '-part' or 'p' +-# if we have 'mapper' in disk device name: +-if [[ ${Dev/mapper//} != $Dev ]] ; then +- # we only expect mpath_partX or mpathpX or mpath-partX +- case $Disk in +- (*p) Disk=${Disk%p} ;; +- (*-part) Disk=${Disk%-part} ;; +- (*_part) Disk=${Disk%_part} ;; +- (*) Log "Unsupported kpartx partition delimiter for $Dev" +- esac ++if ! test -d "$esp_mountpoint" ; then ++ LogPrintError "Failed to create EFI Boot Manager entries (no ESP mountpoint directory $esp_mountpoint)" ++ return 1 + fi + +-# For eMMC devices the trailing 'p' in the Disk value +-# (as in /dev/mmcblk0p that is derived from /dev/mmcblk0p1) +-# needs to be stripped (to get /dev/mmcblk0), otherwise the +-# efibootmgr call fails because of a wrong disk device name. +-# See also https://github.com/rear/rear/issues/2103 +-if [[ $Disk = *'/mmcblk'+([0-9])p ]] ; then +- Disk=${Disk%p} +-fi ++# Mount point inside the target system, ++# accounting for possible trailing slashes in TARGET_FS_ROOT ++esp_mountpoint_inside="${esp_mountpoint#${TARGET_FS_ROOT%%*(/)}}" + +-# For NVMe devices the trailing 'p' in the Disk value +-# (as in /dev/nvme0n1p that is derived from /dev/nvme0n1p1) +-# needs to be stripped (to get /dev/nvme0n1), otherwise the +-# efibootmgr call fails because of a wrong disk device name. +-# See also https://github.com/rear/rear/issues/1564 +-if [[ $Disk = *'/nvme'+([0-9])n+([0-9])p ]] ; then +- Disk=${Disk%p} ++boot_efi_parts=$( find_partition "fs:$esp_mountpoint_inside" fs ) ++if ! test "$boot_efi_parts" ; then ++ LogPrint "Unable to find ESP $esp_mountpoint_inside in layout" ++ LogPrint "Trying to determine device currently mounted at $esp_mountpoint as fallback" ++ boot_efi_dev="$( mount | grep "$esp_mountpoint" | awk '{print $1}' )" ++ if ! test "$boot_efi_dev" ; then ++ LogPrintError "Cannot create EFI Boot Manager entry (unable to find ESP $esp_mountpoint among mounted devices)" ++ return 1 ++ fi ++ if test $(get_component_type "$boot_efi_dev") = part ; then ++ boot_efi_parts="$boot_efi_dev" ++ else ++ boot_efi_parts=$( find_partition "$boot_efi_dev" ) ++ fi ++ if ! test "$boot_efi_parts" ; then ++ LogPrintError "Cannot create EFI Boot Manager entry (unable to find partition for $boot_efi_dev)" ++ return 1 ++ fi ++ LogPrint "Using fallback EFI boot partition(s) $boot_efi_parts (unable to find ESP $esp_mountpoint_inside in layout)" + fi + ++local bootloader partition_block_device partition_number disk efipart ++ + # EFI\fedora\shim.efi +-BootLoader=$( echo $UEFI_BOOTLOADER | cut -d"/" -f4- | sed -e 's;/;\\;g' ) +-LogPrint "Creating EFI Boot Manager entry '$OS_VENDOR $OS_VERSION' for '$BootLoader' (UEFI_BOOTLOADER='$UEFI_BOOTLOADER')" +-Log efibootmgr --create --gpt --disk ${Disk} --part ${ParNr} --write-signature --label \"${OS_VENDOR} ${OS_VERSION}\" --loader \"\\${BootLoader}\" +-if efibootmgr --create --gpt --disk ${Disk} --part ${ParNr} --write-signature --label "${OS_VENDOR} ${OS_VERSION}" --loader "\\${BootLoader}" ; then +- # ok, boot loader has been set-up - tell rear we are done using following var. +- NOBOOTLOADER='' +- return +-fi ++bootloader=$( echo $UEFI_BOOTLOADER | cut -d"/" -f4- | sed -e 's;/;\\;g' ) ++ ++for efipart in $boot_efi_parts ; do ++ # /dev/sda1 or /dev/mapper/vol34_part2 or /dev/mapper/mpath99p4 ++ partition_block_device=$( get_device_name $efipart ) ++ # 1 or 2 or 4 for the examples above ++ partition_number=$( get_partition_number $partition_block_device ) ++ if ! disk=$( get_device_from_partition $partition_block_device $partition_number ) ; then ++ LogPrintError "Cannot create EFI Boot Manager entry for ESP $partition_block_device (unable to find the underlying disk)" ++ # do not error out - we may be able to locate other disks if there are more of them ++ continue ++ fi ++ LogPrint "Creating EFI Boot Manager entry '$OS_VENDOR $OS_VERSION' for '$bootloader' (UEFI_BOOTLOADER='$UEFI_BOOTLOADER') " ++ Log efibootmgr --create --gpt --disk $disk --part $partition_number --write-signature --label \"${OS_VENDOR} ${OS_VERSION}\" --loader \"\\${bootloader}\" ++ if efibootmgr --create --gpt --disk $disk --part $partition_number --write-signature --label "${OS_VENDOR} ${OS_VERSION}" --loader "\\${bootloader}" ; then ++ # ok, boot loader has been set-up - continue with other disks (ESP can be on RAID) ++ NOBOOTLOADER='' ++ else ++ LogPrintError "efibootmgr failed to create EFI Boot Manager entry on $disk partition $partition_number (ESP $partition_block_device )" ++ fi ++done + +-LogPrintError "efibootmgr failed to create EFI Boot Manager entry for '$BootLoader' (UEFI_BOOTLOADER='$UEFI_BOOTLOADER')" ++is_true $NOBOOTLOADER || return 0 ++LogPrintError "efibootmgr failed to create EFI Boot Manager entry for '$bootloader' (UEFI_BOOTLOADER='$UEFI_BOOTLOADER')" ++return 1 +diff --git a/usr/share/rear/lib/layout-functions.sh b/usr/share/rear/lib/layout-functions.sh +index 54ddb50f..cdd81a14 100644 +--- a/usr/share/rear/lib/layout-functions.sh ++++ b/usr/share/rear/lib/layout-functions.sh +@@ -302,12 +302,20 @@ get_child_components() { + done + } + +-# Return all ancestors of component $1 [ of type $2 ] ++# Return all ancestors of component $1 [ of type $2 [ skipping types $3 during resolution ] ] + get_parent_components() { +- declare -a ancestors devlist +- declare current child parent ++ declare -a ancestors devlist ignoretypes ++ declare current child parent parenttype + + devlist=( "$1" ) ++ if [[ "$3" ]] ; then ++ # third argument should, if present, be a space-separated list ++ # of types to ignore when walking up the dependency tree. ++ # Convert it to array ++ ignoretypes=( $3 ) ++ else ++ ignoretypes=() ++ fi + while (( ${#devlist[@]} )) ; do + current=${devlist[0]} + +@@ -318,6 +326,13 @@ get_parent_components() { + if IsInArray "$parent" "${ancestors[@]}" ; then + continue + fi ++ ### ...test if parent is of a correct type if requested... ++ if [[ ${#ignoretypes[@]} -gt 0 ]] ; then ++ parenttype=$(get_component_type "$parent") ++ if IsInArray "$parenttype" "${ignoretypes[@]}" ; then ++ continue ++ fi ++ fi + ### ...and add them to the list + devlist+=( "$parent" ) + ancestors+=( "$parent" ) +@@ -345,22 +360,24 @@ get_parent_components() { + } + + # find_devices ++# ${2+"$2"} in the following functions ensures that $2 gets passed down quoted if present ++# and ignored if not present + # Find the disk device(s) component $1 resides on. + find_disk() { +- get_parent_components "$1" "disk" ++ get_parent_components "$1" "disk" ${2+"$2"} + } + + find_multipath() { +- get_parent_components "$1" "multipath" ++ get_parent_components "$1" "multipath" ${2+"$2"} + } + + find_disk_and_multipath() { +- find_disk "$1" +- is_true "$AUTOEXCLUDE_MULTIPATH" || find_multipath "$1" ++ find_disk "$1" ${2+"$2"} ++ is_true "$AUTOEXCLUDE_MULTIPATH" || find_multipath "$1" ${2+"$2"} + } + + find_partition() { +- get_parent_components "$1" "part" ++ get_parent_components "$1" "part" ${2+"$2"} + } + + # The get_partition_number function +@@ -413,6 +430,54 @@ get_partition_number() { + echo $partition_number + } + ++# Extract the underlying device name from the full partition device name. ++# Underlying device may be a disk, a multipath device or other devices that can be partitioned. ++# Should we use the information in $LAYOUT_DEPS, like get_parent_component does, ++# instead of string munging? ++function get_device_from_partition() { ++ local partition_block_device ++ local device ++ local partition_number ++ ++ partition_block_device=$1 ++ test -b "$partition_block_device" || BugError "get_device_from_partition called with '$partition_block_device' that is no block device" ++ partition_number=${2-$(get_partition_number $partition_block_device )} ++ # /dev/sda or /dev/mapper/vol34_part or /dev/mapper/mpath99p or /dev/mmcblk0p ++ device=${partition_block_device%$partition_number} ++ ++ # Strip trailing partition remainders like '_part' or '-part' or 'p' ++ # if we have 'mapper' in disk device name: ++ if [[ ${partition_block_device/mapper//} != $partition_block_device ]] ; then ++ # we only expect mpath_partX or mpathpX or mpath-partX ++ case $device in ++ (*p) device=${device%p} ;; ++ (*-part) device=${device%-part} ;; ++ (*_part) device=${device%_part} ;; ++ (*) Log "Unsupported kpartx partition delimiter for $partition_block_device" ++ esac ++ fi ++ ++ # For eMMC devices the trailing 'p' in the $device value ++ # (as in /dev/mmcblk0p that is derived from /dev/mmcblk0p1) ++ # needs to be stripped (to get /dev/mmcblk0), otherwise the ++ # efibootmgr call fails because of a wrong disk device name. ++ # See also https://github.com/rear/rear/issues/2103 ++ if [[ $device = *'/mmcblk'+([0-9])p ]] ; then ++ device=${device%p} ++ fi ++ ++ # For NVMe devices the trailing 'p' in the $device value ++ # (as in /dev/nvme0n1p that is derived from /dev/nvme0n1p1) ++ # needs to be stripped (to get /dev/nvme0n1), otherwise the ++ # efibootmgr call fails because of a wrong disk device name. ++ # See also https://github.com/rear/rear/issues/1564 ++ if [[ $device = *'/nvme'+([0-9])n+([0-9])p ]] ; then ++ device=${device%p} ++ fi ++ ++ test -b "$device" && echo $device ++} ++ + # Returns partition start block or 'unknown' + # sda/sda1 or + # dm-XX diff --git a/SOURCES/rear-bz1958247.patch b/SOURCES/rear-bz1958247.patch new file mode 100644 index 0000000..c85f6ad --- /dev/null +++ b/SOURCES/rear-bz1958247.patch @@ -0,0 +1,2040 @@ +diff --git a/usr/share/rear/backup/DUPLICITY/default/100_mount_duplicity_path.sh b/usr/share/rear/backup/DUPLICITY/default/100_mount_duplicity_path.sh +index 64b7a792..6ba7d543 100644 +--- a/usr/share/rear/backup/DUPLICITY/default/100_mount_duplicity_path.sh ++++ b/usr/share/rear/backup/DUPLICITY/default/100_mount_duplicity_path.sh +@@ -1,10 +1,4 @@ +-# create mount point + if [ -n "$BACKUP_DUPLICITY_NETFS_URL" -o -n "$BACKUP_DUPLICITY_NETFS_MOUNTCMD" ]; then +- mkdir -p $v "$BUILD_DIR/outputfs" >&2 +- StopIfError "Could not mkdir '$BUILD_DIR/outputfs'" +- +- AddExitTask "rmdir $v $BUILD_DIR/outputfs >&2" +- + if [[ "$BACKUP_DUPLICITY_NETFS_MOUNTCMD" ]] ; then + BACKUP_DUPLICITY_NETFS_URL="var://BACKUP_DUPLICITY_NETFS_MOUNTCMD" + fi +diff --git a/usr/share/rear/backup/DUPLICITY/default/980_unmount_duplicity_path.sh b/usr/share/rear/backup/DUPLICITY/default/980_unmount_duplicity_path.sh +index 185dbd95..8525ab1d 100644 +--- a/usr/share/rear/backup/DUPLICITY/default/980_unmount_duplicity_path.sh ++++ b/usr/share/rear/backup/DUPLICITY/default/980_unmount_duplicity_path.sh +@@ -6,10 +6,4 @@ if [ -n "$BACKUP_DUPLICITY_NETFS_URL" -o -n "$BACKUP_DUPLICITY_NETFS_UMOUNTCMD" + fi + + umount_url $BACKUP_DUPLICITY_NETFS_URL $BUILD_DIR/outputfs +- +- rmdir $v $BUILD_DIR/outputfs >&2 +- if [[ $? -eq 0 ]] ; then +- # the argument to RemoveExitTask has to be identical to the one given to AddExitTask +- RemoveExitTask "rmdir $v $BUILD_DIR/outputfs >&2" +- fi + fi +diff --git a/usr/share/rear/backup/NETFS/default/100_mount_NETFS_path.sh b/usr/share/rear/backup/NETFS/default/100_mount_NETFS_path.sh +index 5c7696db..b6a955db 100644 +--- a/usr/share/rear/backup/NETFS/default/100_mount_NETFS_path.sh ++++ b/usr/share/rear/backup/NETFS/default/100_mount_NETFS_path.sh +@@ -1,9 +1,3 @@ +-# create mount point +-mkdir -p $v "$BUILD_DIR/outputfs" >&2 +-StopIfError "Could not mkdir '$BUILD_DIR/outputfs'" +- +-AddExitTask "rmdir $v $BUILD_DIR/outputfs >&2" +- + if [[ "$BACKUP_MOUNTCMD" ]] ; then + BACKUP_URL="var://BACKUP_MOUNTCMD" + fi +diff --git a/usr/share/rear/backup/NETFS/default/150_save_copy_of_prefix_dir.sh b/usr/share/rear/backup/NETFS/default/150_save_copy_of_prefix_dir.sh +index d79653b4..9bf8f76a 100644 +--- a/usr/share/rear/backup/NETFS/default/150_save_copy_of_prefix_dir.sh ++++ b/usr/share/rear/backup/NETFS/default/150_save_copy_of_prefix_dir.sh +@@ -3,20 +3,17 @@ + [ -z "${NETFS_KEEP_OLD_BACKUP_COPY}" ] && return + + # do not do this for tapes and special attention for file:///path +-url="$( echo $stage | tr '[:lower:]' '[:upper:]')_URL" +-local scheme=$(url_scheme ${!url}) +-local path=$(url_path ${!url}) +-local opath=$(backup_path $scheme $path) ++local scheme=$( url_scheme $BACKUP_URL ) ++local path=$( url_path $BACKUP_URL ) ++local opath=$( backup_path $scheme $path ) + + # if $opath is empty return silently (e.g. scheme tape) + [ -z "$opath" ] && return 0 + + if ! test -f "${opath}/.lockfile" ; then + if test -d "${opath}" ; then +- rm -rf $v "${opath}.old" >&2 +- StopIfError "Could not remove '${opath}.old'" +- mv -f $v "${opath}" "${opath}.old" >&2 +- StopIfError "Could not move '${opath}'" ++ rm -rf $v "${opath}.old" || Error "Could not remove '${opath}.old'" ++ mv -f $v "${opath}" "${opath}.old" || Error "Could not move '${opath}'" + fi + else + # lockfile was already made through the output workflow (hands off) +diff --git a/usr/share/rear/backup/NETFS/default/200_make_prefix_dir.sh b/usr/share/rear/backup/NETFS/default/200_make_prefix_dir.sh +index db15bca2..43f5b651 100644 +--- a/usr/share/rear/backup/NETFS/default/200_make_prefix_dir.sh ++++ b/usr/share/rear/backup/NETFS/default/200_make_prefix_dir.sh +@@ -2,13 +2,14 @@ + # to $HOSTNAME + + # do not do this for tapes and special attention for file:///path +-url="$( echo $stage | tr '[:lower:]' '[:upper:]')_URL" +-local scheme=$(url_scheme ${!url}) +-local path=$(url_path ${!url}) +-local opath=$(backup_path $scheme $path) ++local scheme=$( url_scheme $BACKUP_URL ) ++local path=$( url_path $BACKUP_URL ) ++local opath=$( backup_path $scheme $path ) + + # if $opath is empty return silently (e.g. scheme tape) + [ -z "$opath" ] && return 0 + +-mkdir -p $v -m0750 "${opath}" >&2 +-StopIfError "Could not mkdir '${opath}'" ++mkdir -p $v -m0750 "${opath}" && return ++ ++# A failure to create the $NETFS_PREFIX sub-directory is fatal: ++Error "Failed to create '$opath' directory for BACKUP_URL=$BACKUP_URL" +diff --git a/usr/share/rear/backup/NETFS/default/250_create_lock.sh b/usr/share/rear/backup/NETFS/default/250_create_lock.sh +index 59090a22..36d547ec 100644 +--- a/usr/share/rear/backup/NETFS/default/250_create_lock.sh ++++ b/usr/share/rear/backup/NETFS/default/250_create_lock.sh +@@ -2,15 +2,13 @@ + # made by a previous mkbackup run when the variable NETFS_KEEP_OLD_BACKUP_COPY has been set + + # do not do this for tapes and special attention for file:///path +-url="$( echo $stage | tr '[:lower:]' '[:upper:]')_URL" +-local scheme=$(url_scheme ${!url}) +-local path=$(url_path ${!url}) +-local opath=$(backup_path $scheme $path) ++local scheme=$( url_scheme $BACKUP_URL ) ++local path=$( url_path $BACKUP_URL ) ++local opath=$( backup_path $scheme $path ) + + # if $opath is empty return silently (e.g. scheme tape) + [ -z "$opath" ] && return 0 + + if test -d "${opath}" ; then +- > "${opath}/.lockfile" +- StopIfError "Could not create '${opath}/.lockfile'" ++ > "${opath}/.lockfile" || Error "Could not create '${opath}/.lockfile'" + fi +diff --git a/usr/share/rear/backup/NETFS/default/970_remove_lock.sh b/usr/share/rear/backup/NETFS/default/970_remove_lock.sh +index f69f7bd8..7038f5b9 100644 +--- a/usr/share/rear/backup/NETFS/default/970_remove_lock.sh ++++ b/usr/share/rear/backup/NETFS/default/970_remove_lock.sh +@@ -1,8 +1,7 @@ + # remove the lockfile +-url="$( echo $stage | tr '[:lower:]' '[:upper:]')_URL" +-local scheme=$(url_scheme ${!url}) +-local path=$(url_path ${!url}) +-local opath=$(backup_path $scheme $path) ++local scheme=$( url_scheme $BACKUP_URL ) ++local path=$( url_path $BACKUP_URL ) ++local opath=$( backup_path $scheme $path ) + + # if $opath is empty return silently (e.g. scheme tape) + [ -z "$opath" ] && return 0 +diff --git a/usr/share/rear/backup/NETFS/default/980_umount_NETFS_dir.sh b/usr/share/rear/backup/NETFS/default/980_umount_NETFS_dir.sh +index f28c6cbf..e1954dc5 100644 +--- a/usr/share/rear/backup/NETFS/default/980_umount_NETFS_dir.sh ++++ b/usr/share/rear/backup/NETFS/default/980_umount_NETFS_dir.sh +@@ -5,9 +5,3 @@ if [[ "$BACKUP_UMOUNTCMD" ]] ; then + fi + + umount_url $BACKUP_URL $BUILD_DIR/outputfs +- +-rmdir $v $BUILD_DIR/outputfs >&2 +-if [[ $? -eq 0 ]] ; then +- # the argument to RemoveExitTask has to be identical to the one given to AddExitTask +- RemoveExitTask "rmdir $v $BUILD_DIR/outputfs >&2" +-fi +diff --git a/usr/share/rear/backup/YUM/default/400_create_include_exclude_files.sh b/usr/share/rear/backup/YUM/default/400_create_include_exclude_files.sh +deleted file mode 100644 +index 6111f89b..00000000 +--- a/usr/share/rear/backup/YUM/default/400_create_include_exclude_files.sh ++++ /dev/null +@@ -1,33 +0,0 @@ +- +-# Backup all that is explicitly specified in BACKUP_PROG_INCLUDE: +-for backup_include_item in "${BACKUP_PROG_INCLUDE[@]}" ; do +- test "$backup_include_item" && echo "$backup_include_item" +-done > $TMP_DIR/backup-include.txt +- +-# Implicitly also backup all local filesystems as defined in mountpoint_device +-# except BACKUP_ONLY_INCLUDE or MANUAL_INCLUDE is set: +-if ! is_true "$BACKUP_ONLY_INCLUDE" ; then +- if [ "${MANUAL_INCLUDE:-NO}" != "YES" ] ; then +- # Add the mountpoints that will be recovered to the backup include list +- # unless a mountpoint is excluded: +- while read mountpoint device junk ; do +- if ! IsInArray "$mountpoint" "${EXCLUDE_MOUNTPOINTS[@]}" ; then +- echo "$mountpoint" +- fi +- done <"$VAR_DIR/recovery/mountpoint_device" >> $TMP_DIR/backup-include.txt +- fi +-fi +- +-# Exclude all that is explicitly specified in BACKUP_PROG_EXCLUDE: +-for backup_exclude_item in "${BACKUP_PROG_EXCLUDE[@]}" ; do +- test "$backup_exclude_item" && echo "$backup_exclude_item" +-done > $TMP_DIR/backup-exclude.txt +- +-# Implicitly also add excluded mountpoints to the backup exclude list +-# except BACKUP_ONLY_EXCLUDE is set: +-if ! is_true "$BACKUP_ONLY_EXCLUDE" ; then +- for excluded_mountpoint in "${EXCLUDE_MOUNTPOINTS[@]}" ; do +- test "$excluded_mountpoint" && echo "$excluded_mountpoint/" +- done >> $TMP_DIR/backup-exclude.txt +-fi +- +diff --git a/usr/share/rear/backup/YUM/default/400_create_include_exclude_files.sh b/usr/share/rear/backup/YUM/default/400_create_include_exclude_files.sh +new file mode 120000 +index 00000000..d8d12c0b +--- /dev/null ++++ b/usr/share/rear/backup/YUM/default/400_create_include_exclude_files.sh +@@ -0,0 +1 @@ ++../../NETFS/default/400_create_include_exclude_files.sh +\ No newline at end of file +diff --git a/usr/share/rear/build/YUM/default/600_create_python_symlink.sh b/usr/share/rear/build/YUM/default/600_create_python_symlink.sh +deleted file mode 100644 +index 29d85905..00000000 +--- a/usr/share/rear/build/YUM/default/600_create_python_symlink.sh ++++ /dev/null +@@ -1,14 +0,0 @@ +-# Copied from ../../DUPLICITY/default/600_create_python_symlink.sh for YUM +-# make sure we have a symbolic link to the python binary +-( +- cd $ROOTFS_DIR/bin +- for py in $(find . -name "python*" ) +- do +- this_py=${py#./*} # should be without ./ +- case $this_py in +- python) break ;; +- python2*|python3*) ln -sf $v $this_py python >&2 ;; +- esac +- done +-) +- +diff --git a/usr/share/rear/build/YUM/default/600_create_python_symlink.sh b/usr/share/rear/build/YUM/default/600_create_python_symlink.sh +new file mode 120000 +index 00000000..d776e5aa +--- /dev/null ++++ b/usr/share/rear/build/YUM/default/600_create_python_symlink.sh +@@ -0,0 +1 @@ ++../../DUPLICITY/default/600_create_python_symlink.sh +\ No newline at end of file +diff --git a/usr/share/rear/lib/framework-functions.sh b/usr/share/rear/lib/framework-functions.sh +index f245861a..b5324747 100644 +--- a/usr/share/rear/lib/framework-functions.sh ++++ b/usr/share/rear/lib/framework-functions.sh +@@ -122,7 +122,7 @@ function cleanup_build_area_and_end_program () { + # Cleanup build area + Log "Finished in $((SECONDS-STARTTIME)) seconds" + if is_true "$KEEP_BUILD_DIR" ; then +- LogPrint "You should also rm -Rf $BUILD_DIR" ++ LogPrint "You should also rm -Rf --one-file-system $BUILD_DIR" + else + Log "Removing build area $BUILD_DIR" + rm -Rf $TMP_DIR +@@ -132,15 +132,11 @@ function cleanup_build_area_and_end_program () { + # in worst case it could not umount; so before remove the BUILD_DIR check if above outputfs is gone + if mountpoint -q "$BUILD_DIR/outputfs" ; then + # still mounted it seems +- LogPrint "Directory $BUILD_DIR/outputfs still mounted - trying lazy umount" + sleep 2 +- umount -f -l $BUILD_DIR/outputfs >&2 +- rm -Rf $v $BUILD_DIR/outputfs >&2 +- else +- # not mounted so we can safely delete $BUILD_DIR/outputfs +- rm -Rf $BUILD_DIR/outputfs ++ umount_mountpoint_lazy $BUILD_DIR/outputfs + fi +- rm -Rf $v $BUILD_DIR >&2 ++ remove_temporary_mountpoint '$BUILD_DIR/outputfs' || BugError "Directory $BUILD_DIR/outputfs not empty, can not remove" ++ rmdir $v $BUILD_DIR >&2 + fi + Log "End of program reached" + } +diff --git a/usr/share/rear/lib/global-functions.sh b/usr/share/rear/lib/global-functions.sh +index 4264bb53..a1aec604 100644 +--- a/usr/share/rear/lib/global-functions.sh ++++ b/usr/share/rear/lib/global-functions.sh +@@ -342,7 +342,44 @@ function url_path() { + echo /${url_without_scheme#*/} + } + +-backup_path() { ++### Returns true if one can upload files to the URL ++function scheme_accepts_files() { ++ local scheme=$1 ++ case $scheme in ++ (null|tape|obdr) ++ # tapes do not support uploading arbitrary files, one has to handle them ++ # as special case (usually passing the tape device as argument to tar) ++ # null means do not upload anything anywhere, leave the files under /var/lib/rear/output ++ return 1 ++ ;; ++ (*) ++ # most URL schemes support uploading files ++ return 0 ++ ;; ++ esac ++} ++ ++### Returns true if URLs with the given scheme corresponds to a path inside ++### a mountable fileystem and one can put files directly into it. ++### The actual path will be returned by backup_path() / output_path(). ++### If returns false, using backup_path() / output_path() has no sense ++### and one must use a scheme-specific method (like lftp or writing them to a tape) ++### to upload files to the destination instead of just "cp" or other direct filesystem access. ++### Returning true does not imply that the URL is currently mounted at a filesystem and usable, ++### only that it can be mounted (use mount_url() first) ++function scheme_supports_filesystem() { ++ local scheme=$1 ++ case $scheme in ++ (null|tape|obdr|rsync|fish|ftp|ftps|hftp|http|https|sftp) ++ return 1 ++ ;; ++ (*) ++ return 0 ++ ;; ++ esac ++} ++ ++function backup_path() { + local scheme=$1 + local path=$2 + case $scheme in +@@ -368,13 +405,21 @@ backup_path() { + echo "$path" + } + +-output_path() { ++function output_path() { + local scheme=$1 + local path=$2 ++ ++ # Abort for unmountable schemes ("tape-like" or "ftp-like" schemes). ++ # Returning an empty string for them is not satisfactory: it could lead to caller putting its files ++ # under / instead of the intended location if the result is not checked for emptiness. ++ # Returning ${BUILD_DIR}/outputfs/${OUTPUT_PREFIX} for unmountable URLs is also not satisfactory: ++ # caller could put its files there expecting them to be safely at their destination, ++ # but if the directory is not a mountpoint, they would get silently lost. ++ # The caller needs to check the URL/scheme using scheme_supports_filesystem() ++ # before calling this function. ++ scheme_supports_filesystem $scheme || BugError "output_path() called with scheme $scheme that does not support filesystem access" ++ + case $scheme in +- (null|tape) # no path for tape required +- path="" +- ;; + (file) # type file needs a local path (must be mounted by user) + path="$path/${OUTPUT_PREFIX}" + ;; +@@ -387,17 +432,33 @@ output_path() { + + + ### Mount URL $1 at mountpoint $2[, with options $3] +-mount_url() { ++function mount_url() { + local url=$1 + local mountpoint=$2 + local defaultoptions="rw,noatime" + local options=${3:-"$defaultoptions"} ++ local scheme ++ ++ scheme=$( url_scheme $url ) ++ ++ # The cases where we return 0 are those that do not need umount and also do not need ExitTask handling. ++ # They thus need to be kept in sync with umount_url() so that RemoveExitTasks is used ++ # iff AddExitTask was used in mount_url(). ++ ++ if ! scheme_supports_filesystem $scheme ; then ++ ### Stuff like null|tape|rsync|fish|ftp|ftps|hftp|http|https|sftp ++ ### Don't need to umount anything for these. ++ ### file: supports filesystem access, but is not mounted and unmounted, ++ ### so it has to be handled specially below. ++ ### Similarly for iso: which gets mounted and unmounted only during recovery. ++ return 0 ++ fi + + ### Generate a mount command + local mount_cmd +- case $(url_scheme $url) in +- (null|tape|file|rsync|fish|ftp|ftps|hftp|http|https|sftp) +- ### Don't need to mount anything for these ++ case $scheme in ++ (file) ++ ### Don't need to mount anything for file:, it is already mounted by user + return 0 + ;; + (iso) +@@ -558,22 +619,47 @@ mount_url() { + ;; + esac + ++ # create mount point ++ mkdir -p $v "$mountpoint" || Error "Could not mkdir '$mountpoint'" ++ AddExitTask "remove_temporary_mountpoint '$mountpoint'" ++ + Log "Mounting with '$mount_cmd'" + # eval is required when mount_cmd contains single quoted stuff (e.g. see the above mount_cmd for curlftpfs) + eval $mount_cmd || Error "Mount command '$mount_cmd' failed." + +- AddExitTask "umount -f $v '$mountpoint' >&2" ++ AddExitTask "perform_umount_url '$url' '$mountpoint' lazy" + return 0 + } + +-### Unmount url $1 at mountpoint $2 +-umount_url() { ++function remove_temporary_mountpoint() { ++ if test -d "$1" ; then ++ rmdir $v "$1" ++ fi ++} ++ ++### Unmount url $1 at mountpoint $2, perform mountpoint cleanup and exit task + error handling ++function umount_url() { + local url=$1 + local mountpoint=$2 ++ local scheme + +- case $(url_scheme $url) in +- (null|tape|file|rsync|fish|ftp|ftps|hftp|http|https|sftp) +- ### Don't need to umount anything for these ++ scheme=$( url_scheme $url ) ++ ++ # The cases where we return 0 are those that do not need umount and also do not need ExitTask handling. ++ # They thus need to be kept in sync with mount_url() so that RemoveExitTasks is used ++ # iff AddExitTask was used in mount_url(). ++ ++ if ! scheme_supports_filesystem $scheme ; then ++ ### Stuff like null|tape|rsync|fish|ftp|ftps|hftp|http|https|sftp ++ ### Don't need to umount anything for these. ++ ### file: supports filesystem access, but is not mounted and unmounted, ++ ### so it has to be handled specially below. ++ ### Similarly for iso: which gets mounted and unmounted only during recovery. ++ return 0 ++ fi ++ ++ case $scheme in ++ (file) + return 0 + ;; + (iso) +@@ -581,42 +667,106 @@ umount_url() { + return 0 + fi + ;; +- (sshfs) +- umount_cmd="fusermount -u $mountpoint" +- ;; +- (davfs) +- umount_cmd="umount $mountpoint" +- # Wait for 3 sek. then remove the cache-dir /var/cache/davfs +- sleep 30 +- # ToDo: put in here the cache-dir from /etc/davfs2/davfs.conf +- # and delete only the just used cache +- #rm -rf /var/cache/davfs2/** +- rm -rf /var/cache/davfs2/*outputfs* +- +- ;; +- (var) +- local var=$(url_host $url) +- umount_cmd="${!var} $mountpoint" ++ (*) ++ # Schemes that actually need nontrivial umount are handled below. ++ # We do not handle them in the default branch because in the case of iso: ++ # it depends on the current workflow whether umount is needed or not. ++ : ++ esac + +- Log "Unmounting with '$umount_cmd'" +- $umount_cmd +- StopIfError "Unmounting failed." ++ # umount_url() is a wrapper that takes care of exit tasks and error handling and mountpoint cleanup. ++ # Therefore it also determines if exit task and mountpoint handling is required and returns early if not. ++ # The actual umount job is performed inside perform_umount_url(). ++ # We do not request lazy umount here because we want umount errors to be reliably reported. ++ perform_umount_url $url $mountpoint || Error "Unmounting '$mountpoint' failed." + +- RemoveExitTask "umount -f $v '$mountpoint' >&2" +- return 0 ++ RemoveExitTask "perform_umount_url '$url' '$mountpoint' lazy" ++ ++ remove_temporary_mountpoint '$mountpoint' && RemoveExitTask "remove_temporary_mountpoint '$mountpoint'" ++ return 0 ++} ++ ++### Unmount url $1 at mountpoint $2 [ lazily if $3 is set to 'lazy' and normal unmount fails ] ++function perform_umount_url() { ++ local url=$1 ++ local mountpoint=$2 ++ local lazy=${3:-} ++ ++ if test $lazy ; then ++ if test $lazy != "lazy" ; then ++ BugError "lazy = $lazy, but it must have the value of 'lazy' or empty" ++ fi ++ fi ++ ++ case $(url_scheme $url) in ++ (sshfs) ++ # does ftpfs need this special case as well? ++ fusermount -u ${lazy:+'-z'} $mountpoint ++ ;; ++ (davfs) ++ umount_davfs $mountpoint $lazy ++ ;; ++ (var) ++ local var ++ var=$(url_host $url) ++ Log "Unmounting with '${!var} $mountpoint'" ++ # lazy unmount not supported with custom umount command ++ ${!var} $mountpoint + ;; ++ (*) ++ # usual umount command ++ umount_mountpoint $mountpoint $lazy + esac ++ # The switch above must be the last statement in this function and the umount commands must be ++ # the last commands (or part of) in each branch. This ensures proper exit code propagation ++ # to the caller even when set -e is used. ++} + +- umount_mountpoint $mountpoint +- StopIfError "Unmounting '$mountpoint' failed." ++### Helper which unmounts davfs mountpoint $1 and cleans up the cache, ++### performing lazy unmount if $2 = 'lazy' and normal unmount fails. ++function umount_davfs() { ++ local mountpoint=$1 ++ local lazy="${2:-}" + +- RemoveExitTask "umount -f $v '$mountpoint' >&2" +- return 0 ++ if test $lazy ; then ++ if test $lazy != "lazy" ; then ++ BugError "lazy = $lazy, but it must have the value of 'lazy' or empty" ++ fi ++ fi ++ ++ if umount_mountpoint $mountpoint ; then ++ # Wait for 3 sek. then remove the cache-dir /var/cache/davfs ++ sleep 30 ++ # TODO: put in here the cache-dir from /etc/davfs2/davfs.conf ++ # and delete only the just used cache ++ #rm -rf /var/cache/davfs2/** ++ rm -rf /var/cache/davfs2/*outputfs* ++ else ++ local retval=$? ++ ++ if test $lazy ; then ++ # try again to unmount lazily and this time do not delete the cache, it is still in use. ++ LogPrintError "davfs cache /var/cache/davfs2/*outputfs* needs to be cleaned up manually after the lazy unmount finishes" ++ umount_mountpoint_lazy $mountpoint ++ else ++ # propagate errors from umount ++ return $retval ++ fi ++ fi + } + +-### Unmount mountpoint $1 +-umount_mountpoint() { ++### Unmount mountpoint $1 [ lazily if $2 = 'lazy' ] ++### Default implementation for filesystems that don't need anything fancy ++### For special umount commands use perform_umount_url() ++function umount_mountpoint() { + local mountpoint=$1 ++ local lazy=${2:-} ++ ++ if test $lazy ; then ++ if test $lazy != "lazy" ; then ++ BugError "lazy = $lazy, but it must have the value of 'lazy' or empty" ++ fi ++ fi + + ### First, try a normal unmount, + Log "Unmounting '$mountpoint'" +@@ -636,7 +786,21 @@ umount_mountpoint() { + fi + + Log "Unmounting '$mountpoint' failed." +- return 1 ++ ++ if test $lazy ; then ++ umount_mountpoint_lazy $mountpoint ++ else ++ return 1 ++ fi ++} ++ ++### Unmount mountpoint $1 lazily ++### Preferably use "umount_mountpoint $mountpoint lazy", which attempts non-lazy unmount first. ++function umount_mountpoint_lazy() { ++ local mountpoint=$1 ++ ++ LogPrint "Directory $mountpoint still mounted - trying lazy umount" ++ umount $v -f -l $mountpoint >&2 + } + + # Change $1 to user input or leave default value on empty input +diff --git a/usr/share/rear/output/PXE/default/800_copy_to_tftp.sh b/usr/share/rear/output/PXE/default/800_copy_to_tftp.sh +index a43dff13..3e7512ee 100644 +--- a/usr/share/rear/output/PXE/default/800_copy_to_tftp.sh ++++ b/usr/share/rear/output/PXE/default/800_copy_to_tftp.sh +@@ -8,10 +8,12 @@ + if [[ ! -z "$PXE_TFTP_URL" ]] ; then + # E.g. PXE_TFTP_URL=nfs://server/export/nfs/tftpboot + local scheme=$( url_scheme $PXE_TFTP_URL ) +- local path=$( url_path $PXE_TFTP_URL ) +- mkdir -p $v "$BUILD_DIR/tftpbootfs" >&2 +- StopIfError "Could not mkdir '$BUILD_DIR/tftpbootfs'" +- AddExitTask "rm -Rf $v $BUILD_DIR/tftpbootfs >&2" ++ ++ # We need filesystem access to the destination (schemes like ftp:// are not supported) ++ if ! scheme_supports_filesystem $scheme ; then ++ Error "Scheme $scheme for PXE output not supported, use a scheme that supports mounting (like nfs: )" ++ fi ++ + mount_url $PXE_TFTP_URL $BUILD_DIR/tftpbootfs $BACKUP_OPTIONS + # However, we copy under $OUTPUT_PREFIX_PXE directory (usually HOSTNAME) to have different clients on one pxe server + PXE_TFTP_LOCAL_PATH=$BUILD_DIR/tftpbootfs +@@ -74,10 +76,6 @@ fi + if [[ ! -z "$PXE_TFTP_URL" ]] ; then + LogPrint "Copied kernel+initrd $( du -shc $KERNEL_FILE "$TMP_DIR/$REAR_INITRD_FILENAME" | tail -n 1 | tr -s "\t " " " | cut -d " " -f 1 ) to $PXE_TFTP_URL/$OUTPUT_PREFIX_PXE" + umount_url $PXE_TFTP_URL $BUILD_DIR/tftpbootfs +- rmdir $BUILD_DIR/tftpbootfs >&2 +- if [[ $? -eq 0 ]] ; then +- RemoveExitTask "rm -Rf $v $BUILD_DIR/tftpbootfs >&2" +- fi + else + # legacy way PXE_TFTP_PATH + LogPrint "Copied kernel+initrd $( du -shc $KERNEL_FILE "$TMP_DIR/$REAR_INITRD_FILENAME" | tail -n 1 | tr -s "\t " " " | cut -d " " -f 1 ) to $PXE_TFTP_PATH" +diff --git a/usr/share/rear/output/PXE/default/810_create_pxelinux_cfg.sh b/usr/share/rear/output/PXE/default/810_create_pxelinux_cfg.sh +index fce4bcf1..5041a3bc 100644 +--- a/usr/share/rear/output/PXE/default/810_create_pxelinux_cfg.sh ++++ b/usr/share/rear/output/PXE/default/810_create_pxelinux_cfg.sh +@@ -1,4 +1,4 @@ +-# 81_create_pxelinux_cfg.sh ++# 810_create_pxelinux_cfg.sh + # + # create pxelinux config on PXE server for Relax-and-Recover + # +@@ -11,10 +11,12 @@ if [[ ! -z "$PXE_CONFIG_URL" ]] ; then + # E.g. PXE_CONFIG_URL=nfs://server/export/nfs/tftpboot/pxelinux.cfg + # Better be sure that on 'server' the directory /export/nfs/tftpboot/pxelinux.cfg exists + local scheme=$( url_scheme $PXE_CONFIG_URL ) +- local path=$( url_path $PXE_CONFIG_URL ) +- mkdir -p $v "$BUILD_DIR/tftpbootfs" >&2 +- StopIfError "Could not mkdir '$BUILD_DIR/tftpbootfs'" +- AddExitTask "rm -Rf $v $BUILD_DIR/tftpbootfs >&2" ++ ++ # We need filesystem access to the destination (schemes like ftp:// are not supported) ++ if ! scheme_supports_filesystem $scheme ; then ++ Error "Scheme $scheme for PXE output not supported, use a scheme that supports mounting (like nfs: )" ++ fi ++ + mount_url $PXE_CONFIG_URL $BUILD_DIR/tftpbootfs $BACKUP_OPTIONS + PXE_LOCAL_PATH=$BUILD_DIR/tftpbootfs + else +@@ -105,10 +107,6 @@ popd >/dev/null + if [[ ! -z "$PXE_CONFIG_URL" ]] ; then + LogPrint "Created pxelinux config '${PXE_CONFIG_PREFIX}$HOSTNAME' and symlinks for $PXE_CREATE_LINKS adresses in $PXE_CONFIG_URL" + umount_url $PXE_TFTP_URL $BUILD_DIR/tftpbootfs +- rmdir $BUILD_DIR/tftpbootfs >&2 +- if [[ $? -eq 0 ]] ; then +- RemoveExitTask "rm -Rf $v $BUILD_DIR/tftpbootfs >&2" +- fi + else + LogPrint "Created pxelinux config '${PXE_CONFIG_PREFIX}$HOSTNAME' and symlinks for $PXE_CREATE_LINKS adresses in $PXE_CONFIG_PATH" + # Add to result files +diff --git a/usr/share/rear/output/PXE/default/820_copy_to_net.sh b/usr/share/rear/output/PXE/default/820_copy_to_net.sh +deleted file mode 100644 +index 39cd316d..00000000 +--- a/usr/share/rear/output/PXE/default/820_copy_to_net.sh ++++ /dev/null +@@ -1,41 +0,0 @@ +- +-# 820_copy_to_net.sh +- +-# Check if we have a target location OUTPUT_URL +-test "$OUTPUT_URL" || return 0 +- +-local scheme=$( url_scheme $OUTPUT_URL ) +-local result_file="" +-local path="" +- +-case "$scheme" in +- (nfs|cifs|usb|tape|file|davfs) +- # The ISO has already been transferred by NETFS. +- return 0 +- ;; +- (fish|ftp|ftps|hftp|http|https|sftp) +- LogPrint "Transferring PXE files to $OUTPUT_URL" +- for result_file in "${RESULT_FILES[@]}" ; do +- path=$(url_path $OUTPUT_URL) +- +- # Make sure that destination directory exists, otherwise lftp would copy +- # RESULT_FILES into last available directory in the path. +- # e.g. OUTPUT_URL=sftp:///iso/server1 and have "/iso/server1" +- # directory missing, would upload RESULT_FILES into sftp:///iso/ +- lftp -c "$OUTPUT_LFTP_OPTIONS; open $OUTPUT_URL; mkdir -fp ${path}" +- +- LogPrint "Transferring file: $result_file" +- lftp -c "$OUTPUT_LFTP_OPTIONS; open $OUTPUT_URL; mput $result_file" || Error "lftp failed to transfer '$result_file' to '$OUTPUT_URL' (lftp exit code: $?)" +- done +- ;; +- (rsync) +- LogPrint "Transferring PXE files to $OUTPUT_URL" +- for result_file in "${RESULT_FILES[@]}" ; do +- LogPrint "Transferring file: $result_file" +- rsync -a $v "$result_file" "$OUTPUT_URL" || Error "Problem transferring '$result_file' to $OUTPUT_URL" +- done +- ;; +- (*) Error "Invalid scheme '$scheme' in '$OUTPUT_URL'." +- ;; +-esac +- +diff --git a/usr/share/rear/output/default/100_mount_output_path.sh b/usr/share/rear/output/default/100_mount_output_path.sh +index 22ef36de..34ea8e5e 100644 +--- a/usr/share/rear/output/default/100_mount_output_path.sh ++++ b/usr/share/rear/output/default/100_mount_output_path.sh +@@ -1,9 +1,3 @@ +-# create mount point +-mkdir -p $v "$BUILD_DIR/outputfs" >&2 +-StopIfError "Could not mkdir '$BUILD_DIR/outputfs'" +- +-AddExitTask "rm -Rf $v $BUILD_DIR/outputfs >&2" +- + if [[ "$OUTPUT_MOUNTCMD" ]] ; then + OUTPUT_URL="var://$OUTPUT_MOUNTCMD" + fi +diff --git a/usr/share/rear/output/default/150_save_copy_of_prefix_dir.sh b/usr/share/rear/output/default/150_save_copy_of_prefix_dir.sh +index 00339a96..06326114 100644 +--- a/usr/share/rear/output/default/150_save_copy_of_prefix_dir.sh ++++ b/usr/share/rear/output/default/150_save_copy_of_prefix_dir.sh +@@ -3,22 +3,20 @@ + [ -z "${KEEP_OLD_OUTPUT_COPY}" ] && return + + # do not do this for tapes and special attention for file:///path +-url="$( echo $stage | tr '[:lower:]' '[:upper:]')_URL" +-local scheme=$(url_scheme ${!url}) +-local path=$(url_path ${!url}) +-local opath=$(output_path $scheme $path) ++local scheme=$( url_scheme $OUTPUT_URL ) ++local path=$( url_path $OUTPUT_URL ) + +-# if $opath is empty return silently (e.g. scheme tape) +-[ -z "$opath" ] && return 0 ++# if filesystem access to url is unsupported return silently (e.g. scheme tape) ++scheme_supports_filesystem $scheme || return 0 ++ ++local opath=$( output_path $scheme $path ) + + # an old lockfile from a previous run not cleaned up by output is possible + [[ -f ${opath}/.lockfile ]] && rm -f ${opath}/.lockfile >&2 + + if test -d "${opath}" ; then +- rm -rf $v "${opath}.old" >&2 +- StopIfError "Could not remove '${opath}.old'" ++ rm -rf $v "${opath}.old" || Error "Could not remove '${opath}.old'" + # below statement was 'cp -af' instead of 'mv -f' (see issue #192) +- mv -f $v "${opath}" "${opath}.old" >&2 +- StopIfError "Could not move '${opath}'" ++ mv -f $v "${opath}" "${opath}.old" || Error "Could not move '${opath}'" + fi + # the ${BUILD_DIR}/outputfs/${OUTPUT_PREFIX} will be created by output/default/200_make_prefix_dir.sh +diff --git a/usr/share/rear/output/default/200_make_prefix_dir.sh b/usr/share/rear/output/default/200_make_prefix_dir.sh +index b8892f2f..606e1c86 100644 +--- a/usr/share/rear/output/default/200_make_prefix_dir.sh ++++ b/usr/share/rear/output/default/200_make_prefix_dir.sh +@@ -3,25 +3,21 @@ + # The $OUTPUT_PREFIX directory defaults to $HOSTNAME. + # + # This happens usually under a mounted network filesystem share +-# e.g. in case of BACKUP_URL=nfs://NFS.server.IP.address/remote/nfs/share +-# but it is also happens for local stuff like BACKUP_URL=usb:///dev/disk/by-label/REAR-000 ++# e.g. in case of OUTPUT_URL=nfs://NFS.server.IP.address/remote/nfs/share ++# but it is also happens for local stuff like OUTPUT_URL=usb:///dev/disk/by-label/REAR-000 + # + # Do not do this for tapes and special attention for file:///path ++local scheme=$( url_scheme $OUTPUT_URL ) ++local path=$( url_path $OUTPUT_URL ) + +-# Generate url variable name that depends on the current stage, +-# e.g. BACKUP_URL or OUTPUT_URL: +-url="$( echo $stage | tr '[:lower:]' '[:upper:]' )_URL" ++# If filesystem access to url is unsupported return silently (e.g. scheme tape) ++scheme_supports_filesystem $scheme || return 0 + +-local scheme=$( url_scheme ${!url} ) +-local path=$( url_path ${!url} ) + local opath=$( output_path $scheme $path ) + +-# If $opath is empty return silently (e.g. scheme tape): +-test "$opath" || return 0 +- + # Create $OUTPUT_PREFIX sub-directory: + mkdir -p $v -m0750 "$opath" && return + +-# A failure to cerate the $OUTPUT_PREFIX sub-directory is fatal: +-Error "Failed to create '$opath' directory for $url=${!url}" ++# A failure to create the $OUTPUT_PREFIX sub-directory is fatal: ++Error "Failed to create '$opath' directory for OUTPUT_URL=$OUTPUT_URL" + +diff --git a/usr/share/rear/output/default/250_create_lock.sh b/usr/share/rear/output/default/250_create_lock.sh +index 49c75601..d792b036 100644 +--- a/usr/share/rear/output/default/250_create_lock.sh ++++ b/usr/share/rear/output/default/250_create_lock.sh +@@ -2,15 +2,14 @@ + # made by a previous mkrescue run when the variable KEEP_OLD_OUTPUT_COPY has been set + + # do not do this for tapes and special attention for file:///path +-url="$( echo $stage | tr '[:lower:]' '[:upper:]')_URL" +-local scheme=$(url_scheme ${!url}) +-local path=$(url_path ${!url}) +-local opath=$(output_path $scheme $path) ++local scheme=$( url_scheme $OUTPUT_URL ) ++local path=$( url_path $OUTPUT_URL ) + +-# if $opath is empty return silently (e.g. scheme tape) +-[ -z "$opath" ] && return 0 ++# if filesystem access to url is unsupported return silently (e.g. scheme tape) ++scheme_supports_filesystem $scheme || return 0 ++ ++local opath=$( output_path $scheme $path ) + + if test -d "${opath}" ; then +- > "${opath}/.lockfile" +- StopIfError "Could not create '${opath}/.lockfile'" ++ > "${opath}/.lockfile" || Error "Could not create '${opath}/.lockfile'" + fi +diff --git a/usr/share/rear/output/default/950_copy_result_files.sh b/usr/share/rear/output/default/950_copy_result_files.sh +index 545b3f7d..77f54d51 100644 +--- a/usr/share/rear/output/default/950_copy_result_files.sh ++++ b/usr/share/rear/output/default/950_copy_result_files.sh +@@ -5,16 +5,25 @@ + + # For example for "rear mkbackuponly" there are usually no result files + # that would need to be copied here to the output location: +-test "$RESULT_FILES" || return 0 ++test "${RESULT_FILES[*]:-}" || return 0 + + local scheme=$( url_scheme $OUTPUT_URL ) + local host=$( url_host $OUTPUT_URL ) + local path=$( url_path $OUTPUT_URL ) +-local opath=$( output_path $scheme $path ) + +-# if $opath is empty return silently (e.g. scheme tape) +-if [[ -z "$opath" || -z "$OUTPUT_URL" || "$scheme" == "obdr" || "$scheme" == "tape" ]] ; then +- return 0 ++if [ -z "$OUTPUT_URL" ] || ! scheme_accepts_files $scheme ; then ++ if [ "$scheme" == "null" -o -z "$OUTPUT_URL" ] ; then ++ # There are result files to copy, but OUTPUT_URL=null indicates that we are not interested in them ++ # TODO: empty OUTPUT_URL seems to be equivalent to null, should we continue to allow that, ++ # or enforce setting it explicitly? ++ return 0 ++ else ++ # There are files to copy, but schemes like tape: do not allow files to be stored. The files would be lost. ++ # Do not allow that. ++ # Schemes like obdr: that store the results themselves should clear RESULT_FILES to indicate that nothing is to be done. ++ # Is this considered a bug in ReaR (BugError), or a user misconfiguration (Error) when this happens? ++ BugError "Output scheme $scheme does not accept result files ${RESULT_FILES[*]}, use OUTPUT_URL=null if you don't want to copy them anywhere." ++ fi + fi + + LogPrint "Copying resulting files to $scheme location" +@@ -38,66 +47,76 @@ RESULT_FILES+=( "$TMP_DIR/$final_logfile_name" ) + LogPrint "Saving $RUNTIME_LOGFILE as $final_logfile_name to $scheme location" + + # The real work (actually copying resulting files to the output location): ++if scheme_supports_filesystem $scheme ; then ++ # We can access the destination as a mounted filesystem. Do nothing special, ++ # simply copy the output files there. (Covers stuff like nfs|cifs|usb|file|sshfs|ftpfs|davfs.) ++ # This won't work for iso:// , but iso can't be a OUTPUT_URL scheme, this is checked in ++ # prep/default/040_check_backup_and_output_scheme.sh ++ # This covers also unknown schemes, because mount_url() will attempt to mount them and fail if this is not possible, ++ # so if we got here, the URL had been mounted successfully. ++ local opath ++ opath=$( output_path $scheme $path ) ++ LogPrint "Copying result files '${RESULT_FILES[*]}' to $opath at $scheme location" ++ # Copy each result file one by one to avoid usually false error exits as in ++ # https://github.com/rear/rear/issues/1711#issuecomment-380009044 ++ # where in case of an improper RESULT_FILES array member 'cp' can error out with something like ++ # cp: will not overwrite just-created '/tmp/rear.XXX/outputfs/f121/rear-f121.log' with '/tmp/rear.XXX/tmp/rear-f121.log' ++ # See ++ # https://stackoverflow.com/questions/4669420/have-you-ever-got-this-message-when-moving-a-file-mv-will-not-overwrite-just-c ++ # which is about the same for 'mv', how to reproduce it: ++ # mkdir a b c ++ # touch a/f b/f ++ # mv a/f b/f c/ ++ # mv: will not overwrite just-created 'c/f' with 'b/f' ++ # It happens because two different files with the same name would be moved to the same place with only one command. ++ # The -f option won't help for this case, it only applies when there already is a target file that will be overwritten. ++ # Accordingly it is sufficient (even without '-f') to copy each result file one by one: ++ for result_file in "${RESULT_FILES[@]}" ; do ++ ++ # note: s390 kernel copy is only through nfs ++ # ++ # s390 optional naming override of initrd and kernel to match the s390 filesytem naming conventions ++ # on s390a there is an option to name the initrd and kernel in the form of ++ # file name on s390 are in the form of name type mode ++ # the name is the userid or vm name and the type is initrd or kernel ++ # if the vm name (cp q userid) is HOSTA then the files written will be HOSTA kernel and HOSTA initrd ++ # vars needed: ++ # ZVM_NAMING - set in local.conf, if Y then enable naming override ++ # ZVM_KERNEL_NAME - keeps track of kernel name in results array ++ # ARCH - override only if ARCH is Linux-s390 ++ # ++ # initrd name override is handled in 900_create_initramfs.sh ++ # kernel name override is handled in 400_guess_kernel.sh ++ # kernel name override is handled in 950_copy_result_files.sh ++ ++ if [[ "$ZVM_NAMING" == "Y" && "$ARCH" == "Linux-s390" ]] ; then ++ if [[ -z $opath ]] ; then ++ Error "Output path is not set, please check OUTPUT_URL in local.conf." ++ fi ++ ++ if [ "$ZVM_KERNEL_NAME" == "$result_file" ] ; then ++ VM_UID=$(vmcp q userid |awk '{ print $1 }') ++ ++ if [[ -z $VM_UID ]] ; then ++ Error "VM UID is not set, VM UID is set from call to vmcp. Please make sure vmcp is available and 'vmcp q userid' returns VM ID" ++ fi ++ ++ LogPrint "s390 kernel naming override: $result_file will be written as $VM_UID.kernel" ++ cp $v "$result_file" $opath/$VM_UID.kernel || Error "Could not copy result file $result_file to $opath/$VM_UID.kernel at $scheme location" ++ else ++ cp $v "$result_file" $opath/ || Error "Could not copy result file $result_file to $opath at $scheme location" ++ fi ++ else ++ cp $v "$result_file" $opath/ || Error "Could not copy result file $result_file to $opath at $scheme location" ++ fi ++ done ++ ++ return 0 ++fi ++ ++# Filesystem access to output destination not supported, use a scheme-specific tool (rsync, lftp) + case "$scheme" in +- (nfs|cifs|usb|file|sshfs|ftpfs|davfs) +- LogPrint "Copying result files '${RESULT_FILES[@]}' to $opath at $scheme location" +- # Copy each result file one by one to avoid usually false error exits as in +- # https://github.com/rear/rear/issues/1711#issuecomment-380009044 +- # where in case of an improper RESULT_FILES array member 'cp' can error out with something like +- # cp: will not overwrite just-created '/tmp/rear.XXX/outputfs/f121/rear-f121.log' with '/tmp/rear.XXX/tmp/rear-f121.log' +- # See +- # https://stackoverflow.com/questions/4669420/have-you-ever-got-this-message-when-moving-a-file-mv-will-not-overwrite-just-c +- # which is about the same for 'mv', how to reproduce it: +- # mkdir a b c +- # touch a/f b/f +- # mv a/f b/f c/ +- # mv: will not overwrite just-created 'c/f' with 'b/f' +- # It happens because two different files with the same name would be moved to the same place with only one command. +- # The -f option won't help for this case, it only applies when there already is a target file that will be overwritten. +- # Accordingly it is sufficient (even without '-f') to copy each result file one by one: +- for result_file in "${RESULT_FILES[@]}" ; do +- +- # note: s390 kernel copy is only through nfs +- # +- # s390 optional naming override of initrd and kernel to match the s390 filesytem naming conventions +- # on s390a there is an option to name the initrd and kernel in the form of +- # file name on s390 are in the form of name type mode +- # the name is the userid or vm name and the type is initrd or kernel +- # if the vm name (cp q userid) is HOSTA then the files written will be HOSTA kernel and HOSTA initrd +- # vars needed: +- # ZVM_NAMING - set in local.conf, if Y then enable naming override +- # ZVM_KERNEL_NAME - keeps track of kernel name in results array +- # ARCH - override only if ARCH is Linux-s390 +- # +- # initrd name override is handled in 900_create_initramfs.sh +- # kernel name override is handled in 400_guess_kernel.sh +- # kernel name override is handled in 950_copy_result_files.sh +- +- if [[ "$ZVM_NAMING" == "Y" && "$ARCH" == "Linux-s390" ]] ; then +- if [[ -z $opath ]] ; then +- Error "Output path is not set, please check OUTPUT_URL in local.conf." +- fi +- +- if [ "$ZVM_KERNEL_NAME" == "$result_file" ] ; then +- VM_UID=$(vmcp q userid |awk '{ print $1 }') +- +- if [[ -z $VM_UID ]] ; then +- Error "VM UID is not set, VM UID is set from call to vmcp. Please make sure vmcp is available and 'vmcp q userid' returns VM ID" +- fi +- +- LogPrint "s390 kernel naming override: $result_file will be written as $VM_UID.kernel" +- cp $v "$result_file" $opath/$VM_UID.kernel || Error "Could not copy result file $result_file to $opath/$VM_UID.kernel at $scheme location" +- else +- cp $v "$result_file" $opath/ || Error "Could not copy result file $result_file to $opath at $scheme location" +- fi +- else +- cp $v "$result_file" $opath/ || Error "Could not copy result file $result_file to $opath at $scheme location" +- fi +- done +- ;; + (fish|ftp|ftps|hftp|http|https|sftp) +- # FIXME: Verify if usage of $array[*] instead of "${array[@]}" is actually intended here +- # see https://github.com/rear/rear/issues/1068 + LogPrint "Copying result files '${RESULT_FILES[*]}' to $scheme location" + Log "lftp -c $OUTPUT_LFTP_OPTIONS; open $OUTPUT_URL; mput ${RESULT_FILES[*]}" + +@@ -111,12 +130,15 @@ case "$scheme" in + (rsync) + # If BACKUP = RSYNC output/RSYNC/default/900_copy_result_files.sh took care of it: + test "$BACKUP" = "RSYNC" && return 0 +- LogPrint "Copying result files '${RESULT_FILES[@]}' to $scheme location" +- Log "rsync -a $v ${RESULT_FILES[@]} ${host}:${path}" ++ LogPrint "Copying result files '${RESULT_FILES[*]}' to $scheme location" ++ Log "rsync -a $v ${RESULT_FILES[*]} ${host}:${path}" + rsync -a $v "${RESULT_FILES[@]}" "${host}:${path}" || Error "Problem transferring result files to $OUTPUT_URL" + ;; + (*) +- Error "Invalid scheme '$scheme' in '$OUTPUT_URL'." ++ # Should be unreachable, if we got here, it is a bug. ++ # Unknown schemes are handled in mount_url(), which tries to mount them and aborts if they are unsupported. ++ # If they can be mounted, they fall under the scheme_supports_filesystem branch above. ++ BugError "Invalid scheme '$scheme' in '$OUTPUT_URL'." + ;; + esac + +diff --git a/usr/share/rear/output/default/970_remove_lock.sh b/usr/share/rear/output/default/970_remove_lock.sh +index 56640839..3b1b97cc 100644 +--- a/usr/share/rear/output/default/970_remove_lock.sh ++++ b/usr/share/rear/output/default/970_remove_lock.sh +@@ -1,10 +1,11 @@ + # remove the lockfile + local scheme=$(url_scheme $OUTPUT_URL) + local path=$(url_path $OUTPUT_URL) +-local opath=$(output_path $scheme $path) + +-# if $opath is empty return silently (e.g. scheme tape) +-[ -z "$opath" ] && return 0 ++# if filesystem access to url is unsupported return silently (e.g. scheme tape) ++scheme_supports_filesystem $scheme || return 0 ++ ++local opath=$( output_path $scheme $path ) + + # when OUTPUT_URL=BACKUP_URL we keep the lockfile to avoid double moves of the directory + [[ "$OUTPUT_URL" != "$BACKUP_URL" ]] && rm -f $v "${opath}/.lockfile" >&2 +diff --git a/usr/share/rear/output/default/980_umount_output_dir.sh b/usr/share/rear/output/default/980_umount_output_dir.sh +index 9a9995bd..abf0cd53 100644 +--- a/usr/share/rear/output/default/980_umount_output_dir.sh ++++ b/usr/share/rear/output/default/980_umount_output_dir.sh +@@ -9,12 +9,3 @@ if [[ -z "$OUTPUT_URL" ]] ; then + fi + + umount_url $OUTPUT_URL $BUILD_DIR/outputfs +- +-[[ -d $BUILD_DIR/outputfs/$NETFS_PREFIX ]] && rm -rf $v $BUILD_DIR/outputfs/$NETFS_PREFIX +-[[ -d $BUILD_DIR/outputfs/$RSYNC_PREFIX ]] && rm -rf $v $BUILD_DIR/outputfs/$RSYNC_PREFIX +- +-rmdir $v $BUILD_DIR/outputfs >&2 +-if [[ $? -eq 0 ]] ; then +- # the argument to RemoveExitTask has to be identical to the one given to AddExitTask +- RemoveExitTask "rm -Rf $v $BUILD_DIR/outputfs >&2" +-fi +diff --git a/usr/share/rear/prep/BORG/default/250_mount_usb.sh b/usr/share/rear/prep/BORG/default/250_mount_usb.sh +index c13fd088..05be0179 100644 +--- a/usr/share/rear/prep/BORG/default/250_mount_usb.sh ++++ b/usr/share/rear/prep/BORG/default/250_mount_usb.sh +@@ -8,10 +8,5 @@ + # When BORGBACKUP_HOST is set, we don't need to mount anything as SSH + # backup destination will be handled internally by Borg it self. + if [[ -z $BORGBACKUP_HOST ]]; then +- # Has to be $verbose, not "$verbose", since it's used as option. +- # shellcheck disable=SC2086,SC2154 +- mkdir -p $verbose "$borg_dst_dev" >&2 +- StopIfError "Could not mkdir '$borg_dst_dev'" +- + mount_url "usb://$USB_DEVICE" "$borg_dst_dev" + fi +diff --git a/usr/share/rear/prep/YUM/default/070_set_backup_archive.sh b/usr/share/rear/prep/YUM/default/070_set_backup_archive.sh +deleted file mode 100644 +index 2fbcc6cd..00000000 +--- a/usr/share/rear/prep/YUM/default/070_set_backup_archive.sh ++++ /dev/null +@@ -1,300 +0,0 @@ +-# Copied from ../../NETFS/default/070_set_backup_archive.sh for YUM +-### Determine the name of the backup archive +-### This needs to be after we special case USB devices. +- +-# FIXME: backuparchive is no local variable (regardless that it is lowercased) +- +-# If TAPE_DEVICE is specified, use that: +-if test "$TAPE_DEVICE" ; then +- backuparchive="$TAPE_DEVICE" +- LogPrint "Using backup archive '$backuparchive'" +- return +-fi +- +-local backup_file_suffix="$BACKUP_PROG_SUFFIX$BACKUP_PROG_COMPRESS_SUFFIX" +-local backup_file_name="$BACKUP_PROG_ARCHIVE$backup_file_suffix" +- +-local scheme=$( url_scheme $BACKUP_URL ) +-local path=$( url_path $BACKUP_URL ) +-case "$scheme" in +- (file|iso) +- # Define the output path according to the scheme +- local outputpath=$( backup_path $scheme $path ) +- backuparchive="$outputpath/$backup_file_name" +- LogPrint "Using backup archive '$backuparchive'" +- return +- ;; +- (tape) +- # TODO: Check if that case is really needed. +- # Perhaps prep/default/030_translate_tape.sh does already all what is needed. +- backuparchive=$path +- LogPrint "Using backup archive '$backuparchive'" +- return +- ;; +-esac +- +-local backup_directory=$BUILD_DIR/outputfs/$NETFS_PREFIX +- +-# Normal (i.e. non-incremental/non-differential) backup: +-if ! test "incremental" = "$BACKUP_TYPE" -o "differential" = "$BACKUP_TYPE" ; then +- # In case of normal (i.e. non-incremental) backup there is only one restore archive +- # and its name is the same as the backup archive (usually 'backup.tar.gz'): +- backuparchive="$backup_directory/$backup_file_name" +- LogPrint "Using backup archive '$backuparchive'" +- # This script is also run during "rear recover/restoreonly" where RESTORE_ARCHIVES must be set. +- local backup_restore_workflows=( "recover" "restoreonly" ) +- if IsInArray $WORKFLOW ${backup_restore_workflows[@]} ; then +- # Only set RESTORE_ARCHIVES the backup archive is actually accessible +- # cf. https://github.com/rear/rear/issues/1166 +- if test -r "$backuparchive" ; then +- RESTORE_ARCHIVES=( "$backuparchive" ) +- else +- # In case of USB backup there is the subsequent 540_choose_backup_archive.sh script +- # that shows a backup selection dialog when RESTORE_ARCHIVES is not already set. +- if test "usb" = "$scheme" ; then +- LogPrint "Backup archive '$backuparchive' not readable. Need to select another one." +- else +- Error "Backup archive '$backuparchive' not readable." +- fi +- fi +- fi +- return +-fi +- +-# Incremental or differential backup: +-set -e -u -o pipefail +-# Incremental or differential backup only works for the NETFS backup method +-# and only with the 'tar' backup program: +-if ! test "NETFS" = "$BACKUP" -a "tar" = "$BACKUP_PROG" ; then +- Error "BACKUP_TYPE incremental or differential only works with BACKUP=NETFS and BACKUP_PROG=tar" +-fi +-# Incremental or differential backup is currently only known to work with BACKUP_URL=nfs://. +-# Other BACKUP_URL schemes may work and at least BACKUP_URL=usb:///... needs special setup +-# to work with incremental or differential backup (see https://github.com/rear/rear/issues/1145): +-if test "usb" = "$scheme" ; then +- # When USB_SUFFIX is set the compliance mode is used where +- # backup on USB works in compliance with backup on NFS which means +- # a fixed backup directory where incremental or differential backups work. +- # Use plain $USB_SUFFIX and not "$USB_SUFFIX" because when USB_SUFFIX contains only blanks +- # test "$USB_SUFFIX" would result true because test " " results true: +- test $USB_SUFFIX || Error "BACKUP_TYPE incremental or differential requires USB_SUFFIX for BACKUP_URL=usb" +-fi +-# Incremental or differential backup and keeping old backup contradict each other (mutual exclusive) +-# so that NETFS_KEEP_OLD_BACKUP_COPY must not be 'true' in case of incremental or differential backup: +-if test "$NETFS_KEEP_OLD_BACKUP_COPY" ; then +- NETFS_KEEP_OLD_BACKUP_COPY="" +- LogPrint "Disabled NETFS_KEEP_OLD_BACKUP_COPY because BACKUP_TYPE incremental or differential does not work with that" +-fi +-# For incremental or differential backup some date values (weekday, YYYY-MM-DD, HHMM) are needed +-# that must be consistent for one single point of the current time which means +-# one cannot call the 'date' command several times because then there would be +-# a small probability that e.g. weekday, YYYY-MM-DD, HHMM do not match +-# one single point in time (in particular when midnight passes in between). +-# Therefore the output of one single 'date' call is storend in an array and +-# the array elements are then assinged to individual variables as needed: +-local current_date_output=( $( date '+%a %Y-%m-%d %H%M' ) ) +-local current_weekday="${current_date_output[0]}" +-local current_yyyy_mm_dd="${current_date_output[1]}" +-local current_hhmm="${current_date_output[2]}" +-# The date FULLBACKUP_OUTDATED_DAYS ago is needed to check if the latest full backup is too old. +-# When the latest full backup is more than FULLBACKUP_OUTDATED_DAYS ago a new full backup is made. +-# This separated call of the 'date' command which is technically needed because it is +-# for another point in time (e.g. 7 days ago) is run after the above call of the 'date' +-# command for the current time to be on the safe side when midnight passes in between +-# both 'date' commands which would then result that a new full backup is made +-# when the latest full backup is basically right now FULLBACKUP_OUTDATED_DAYS ago because +-# the stored date of the latest full backup is the current date at the time when it was made. +-# Example (assuming FULLBACKUP_OUTDATED_DAYS=7 ): +-# The latest full backup was made on Sunday January 10 in 2016 (just before midnight). +-# One week later this script runs again while midnight passes between the two 'date' calls +-# so that current_date_output[@]="Sun 2016-01-17 0000" (still Sunday January 17 in 2016) +-# and yyyymmdd_max_days_ago=20160111 (already Monday January 11 in 2016), then +-# Sunday January 10 is older than Monday January 11 so that a new full backup is made: +-test "$FULLBACKUP_OUTDATED_DAYS" || FULLBACKUP_OUTDATED_DAYS="7" +-local yyyymmdd_max_days_ago=$( date '+%Y%m%d' --date="$FULLBACKUP_OUTDATED_DAYS days ago" ) +-# Full backup file names are of the form YYYY-MM-DD-HHMM-F.tar.gz +-# where the 'F' denotes a full backup: +-local full_backup_marker="F" +-# Incremental backup file names are of the form YYYY-MM-DD-HHMM-I.tar.gz +-# where the 'I' denotes an incremental backup: +-local incremental_backup_marker="I" +-# Differential backup file names are of the form YYYY-MM-DD-HHMM-D.tar.gz +-# where the last 'D' denotes a differential backup: +-local differential_backup_marker="D" +-# In case of incremental or differential backup the RESTORE_ARCHIVES contains +-# first the latest full backup file. +-# In case of incremental backup the RESTORE_ARCHIVES contains +-# after the latest full backup file each incremental backup +-# in the ordering how they must be restored. +-# For example when the latest full backup was made on Sunday +-# plus each subsequent weekday a separated incremental backup was made, +-# then during a "rear recover" on Wednesday morning +-# first the full backup from Sunday has to be restored, +-# then the incremental backup from Monday, and +-# finally the incremental backup from Tuesday. +-# In case of differential backup the RESTORE_ARCHIVES contains +-# after the latest full backup file the latest differential backup. +-# For example when the latest full backup was made on Sunday +-# plus each subsequent weekday a separated differential backup was made, +-# then during a "rear recover" on Wednesday morning +-# first the full backup from Sunday has to be restored, +-# and finally the differential backup from Tuesday +-# (i.e. the differential backup from Monday is skipped). +-# The date format YYYY-MM-DD that is used here is crucial. +-# It is the ISO 8601 format 'year-month-day' to specify a day of a year +-# that is accepted by 'tar' for the '--newer' option, +-# see the GNU tar manual section "Operating Only on New Files" +-# at https://www.gnu.org/software/tar/manual/html_node/after.html +-# and the GNU tar manual section "Calendar date items" +-# at https://www.gnu.org/software/tar/manual/html_node/Calendar-date-items.html#SEC124 +-local date_glob_regex="[0-9][0-9][0-9][0-9]-[0-9][0-9]-[0-9][0-9]" +-local date_time_glob_regex="$date_glob_regex-[0-9][0-9][0-9][0-9]" +-# Determine what kind of backup must be created, 'full' or 'incremental' or 'differential' +-# (the empty default means it is undecided what kind of backup must be created): +-local create_backup_type="" +-# Code regarding creating a backup is useless during "rear recover" and +-# messages about creating a backup are misleading during "rear recover": +-local recovery_workflows=( "recover" "layoutonly" "restoreonly" ) +-if ! IsInArray $WORKFLOW ${recovery_workflows[@]} ; then +- # When today is a specified full backup day, do a full backup in any case +- # (regardless if there is already a full backup of this day): +- if IsInArray "$current_weekday" "${FULLBACKUPDAY[@]}" ; then +- create_backup_type="full" +- LogPrint "Today's weekday ('$current_weekday') is a full backup day that triggers a new full backup in any case" +- fi +-fi +-# Get the latest full backup (if exists): +-local full_backup_glob_regex="$date_time_glob_regex-$full_backup_marker$backup_file_suffix" +-# Here things like 'find /path/to/dir -name '*.tar.gz' | sort' are used because +-# one cannot use bash globbing via commands like 'ls /path/to/dir/*.tar.gz' +-# because /usr/sbin/rear sets the nullglob bash option which leads to plain 'ls' +-# when '/path/to/dir/*.tar.gz' matches nothing (i.e. when no backup file exists) +-# so that then plain 'ls' would result nonsense. +-local latest_full_backup=$( find $backup_directory -name "$full_backup_glob_regex" | sort | tail -n1 ) +-# A latest full backup is found: +-if test "$latest_full_backup" ; then +- local latest_full_backup_file_name=$( basename "$latest_full_backup" ) +- # The full_or_incremental_backup_glob_regex is also needed below for non-"recover" WORKFLOWs +- # to set the right variables for creating an incremental backup: +- local full_or_incremental_backup_glob_regex="$date_time_glob_regex-[$full_backup_marker$incremental_backup_marker]$backup_file_suffix" +- # Code regarding creating a backup is useless during "rear recover" and +- # messages about creating a backup are misleading during "rear recover": +- if ! IsInArray $WORKFLOW ${recovery_workflows[@]} ; then +- # There is nothing to do here if it is already decided that +- # a full backup must be created (see "full backup day" above"): +- if ! test "full" = "$create_backup_type" ; then +- local latest_full_backup_date=$( echo $latest_full_backup_file_name | grep -o "$date_glob_regex" ) +- local yyyymmdd_latest_full_backup=$( echo $latest_full_backup_date | tr -d '-' ) +- # Check if the latest full backup is too old: +- if test $yyyymmdd_latest_full_backup -lt $yyyymmdd_max_days_ago ; then +- create_backup_type="full" +- LogPrint "Latest full backup date '$latest_full_backup_date' too old (more than $FULLBACKUP_OUTDATED_DAYS days ago) triggers new full backup" +- else +- # When a latest full backup is found that is not too old +- # a BACKUP_TYPE (incremental or differential) backup will be created: +- create_backup_type="$BACKUP_TYPE" +- LogPrint "Latest full backup found ($latest_full_backup_file_name) triggers $BACKUP_TYPE backup" +- fi +- fi +- else +- # This script is also run during "rear recover" where RESTORE_ARCHIVES must be set: +- case "$BACKUP_TYPE" in +- (incremental) +- # When a latest full backup is found use that plus all later incremental backups for restore: +- # The following command is a bit tricky: +- # It lists all YYYY-MM-DD-HHMM-F.tar.gz and all YYYY-MM-DD-HHMM-I.tar.gz files in the backup directory and sorts them +- # and finally it outputs only those that match the latest full backup file name and incremental backups that got sorted after that +- # where it is mandatory that the backup file names sort by date (i.e. date must be the leading part of the backup file names): +- RESTORE_ARCHIVES=( $( find $backup_directory -name "$full_or_incremental_backup_glob_regex" | sort | sed -n -e "/$latest_full_backup_file_name/,\$p" ) ) +- ;; +- (differential) +- # For differential backup use the latest full backup plus the one latest differential backup for restore: +- # The following command is a bit tricky: +- # It lists all YYYY-MM-DD-HHMM-F.tar.gz and all YYYY-MM-DD-HHMM-D.tar.gz files in the backup directory and sorts them +- # then it outputs only those that match the latest full backup file name and all differential backups that got sorted after that +- # and then it outputs only the first line (i.e. the full backup) and the last line (i.e. the latest differential backup) +- # but when no differential backup exists (i.e. when only the full backup exists) the first line is also the last line +- # so that "sed -n -e '1p;$p'" outputs the full backup twice which is corrected by the final "sort -u": +- local full_or_differential_backup_glob_regex="$date_time_glob_regex-[$full_backup_marker$differential_backup_marker]$backup_file_suffix" +- RESTORE_ARCHIVES=( $( find $backup_directory -name "$full_or_differential_backup_glob_regex" | sort | sed -n -e "/$latest_full_backup_file_name/,\$p" | sed -n -e '1p;$p' | sort -u ) ) +- ;; +- (*) +- BugError "Unexpected BACKUP_TYPE '$BACKUP_TYPE'" +- ;; +- esac +- # Tell the user what will be restored: +- local restore_archives_file_names="" +- for restore_archive in "${RESTORE_ARCHIVES[@]}" ; do +- restore_archives_file_names="$restore_archives_file_names $( basename "$restore_archive" )" +- done +- LogPrint "For backup restore using $restore_archives_file_names" +- fi +-# No latest full backup is found: +-else +- # Code regarding creating a backup is useless during "rear recover" and +- # messages about creating a backup are misleading during "rear recover": +- if ! IsInArray $WORKFLOW ${recovery_workflows[@]} ; then +- # If no latest full backup is found create one during "rear mkbackup": +- create_backup_type="full" +- LogPrint "No full backup found (YYYY-MM-DD-HHMM-F.tar.gz) triggers full backup" +- else +- # This script is also run during "rear recover" where RESTORE_ARCHIVES must be set: +- # If no latest full backup is found (i.e. no file name matches the YYYY-MM-DD-HHMM-F.tar.gz form) +- # fall back to what is done in case of normal (i.e. non-incremental/non-differential) backup +- # and hope for the best (i.e. that a backup_directory/backup_file_name actually exists). +- # In case of normal (i.e. non-incremental/non-differential) backup there is only one restore archive +- # and its name is the same as the backup archive (usually 'backup.tar.gz'). +- # This is only a fallback setting to be more on the safe side for "rear recover". +- # Initially for the very fist run of incremental backup during "rear mkbackup" +- # a full backup file of the YYYY-MM-DD-HHMM-F.tar.gz form will be created. +- RESTORE_ARCHIVES=( "$backup_directory/$backup_file_name" ) +- LogPrint "Using $backup_file_name for backup restore" +- fi +-fi +-# Code regarding creating a backup is useless during "rear recover" and +-# messages about creating a backup are misleading during "rear recover": +-if ! IsInArray $WORKFLOW ${recovery_workflows[@]} ; then +- # Set the right variables for creating a backup (but do not actually do anything at this point): +- case "$create_backup_type" in +- (full) +- local new_full_backup_file_name="$current_yyyy_mm_dd-$current_hhmm-$full_backup_marker$backup_file_suffix" +- backuparchive="$backup_directory/$new_full_backup_file_name" +- BACKUP_PROG_CREATE_NEWER_OPTIONS="-V $new_full_backup_file_name" +- LogPrint "Performing full backup using backup archive '$new_full_backup_file_name'" +- ;; +- (incremental) +- local new_incremental_backup_file_name="$current_yyyy_mm_dd-$current_hhmm-$incremental_backup_marker$backup_file_suffix" +- backuparchive="$backup_directory/$new_incremental_backup_file_name" +- # Get the latest latest incremental backup that is based on the latest full backup (if exists): +- local incremental_backup_glob_regex="$date_time_glob_regex-$incremental_backup_marker$backup_file_suffix" +- # First get the latest full backup plus all later incremental backups (cf. how RESTORE_ARCHIVES is set in case of incremental backup) +- # then grep only the incremental backups and from the incremental backups use only the last one (if exists): +- local latest_incremental_backup=$( find $backup_directory -name "$full_or_incremental_backup_glob_regex" | sort | sed -n -e "/$latest_full_backup_file_name/,\$p" | grep "$incremental_backup_glob_regex" | tail -n1 ) +- if test "$latest_incremental_backup" ; then +- # A latest incremental backup that is based on the latest full backup is found: +- local latest_incremental_backup_file_name=$( basename $latest_incremental_backup ) +- LogPrint "Latest incremental backup found ($latest_incremental_backup_file_name) that is newer than the latest full backup" +- local latest_incremental_backup_date=$( echo $latest_incremental_backup_file_name | grep -o "$date_glob_regex" ) +- BACKUP_PROG_CREATE_NEWER_OPTIONS="--newer=$latest_incremental_backup_date -V $latest_incremental_backup_file_name" +- LogPrint "Performing incremental backup for files newer than $latest_incremental_backup_date using backup archive '$new_incremental_backup_file_name'" +- else +- # When there is not yet an incremental backup that is based on the latest full backup +- # the new created incremental backup must be based on the latest full backup: +- BACKUP_PROG_CREATE_NEWER_OPTIONS="--newer=$latest_full_backup_date -V $latest_full_backup_file_name" +- LogPrint "Performing incremental backup for files newer than $latest_full_backup_date using backup archive '$new_incremental_backup_file_name'" +- fi +- ;; +- (differential) +- local new_differential_backup_file_name="$current_yyyy_mm_dd-$current_hhmm-$differential_backup_marker$backup_file_suffix" +- backuparchive="$backup_directory/$new_differential_backup_file_name" +- BACKUP_PROG_CREATE_NEWER_OPTIONS="--newer=$latest_full_backup_date -V $latest_full_backup_file_name" +- LogPrint "Performing differential backup for files newer than $latest_full_backup_date using backup archive '$new_differential_backup_file_name'" +- ;; +- (*) +- BugError "Unexpected create_backup_type '$create_backup_type'" +- ;; +- esac +-fi +-# Go back from "set -e -u -o pipefail" to the defaults: +-apply_bash_flags_and_options_commands "$DEFAULT_BASH_FLAGS_AND_OPTIONS_COMMANDS" +- +diff --git a/usr/share/rear/prep/YUM/default/070_set_backup_archive.sh b/usr/share/rear/prep/YUM/default/070_set_backup_archive.sh +new file mode 120000 +index 00000000..cdbdc31f +--- /dev/null ++++ b/usr/share/rear/prep/YUM/default/070_set_backup_archive.sh +@@ -0,0 +1 @@ ++../../NETFS/default/070_set_backup_archive.sh +\ No newline at end of file +diff --git a/usr/share/rear/restore/DUPLICITY/default/100_mount_duplicity_path.sh b/usr/share/rear/restore/DUPLICITY/default/100_mount_duplicity_path.sh +deleted file mode 100644 +index 64b7a792..00000000 +--- a/usr/share/rear/restore/DUPLICITY/default/100_mount_duplicity_path.sh ++++ /dev/null +@@ -1,15 +0,0 @@ +-# create mount point +-if [ -n "$BACKUP_DUPLICITY_NETFS_URL" -o -n "$BACKUP_DUPLICITY_NETFS_MOUNTCMD" ]; then +- mkdir -p $v "$BUILD_DIR/outputfs" >&2 +- StopIfError "Could not mkdir '$BUILD_DIR/outputfs'" +- +- AddExitTask "rmdir $v $BUILD_DIR/outputfs >&2" +- +- if [[ "$BACKUP_DUPLICITY_NETFS_MOUNTCMD" ]] ; then +- BACKUP_DUPLICITY_NETFS_URL="var://BACKUP_DUPLICITY_NETFS_MOUNTCMD" +- fi +- +- mount_url $BACKUP_DUPLICITY_NETFS_URL $BUILD_DIR/outputfs $BACKUP_DUPLICITY_NETFS_OPTIONS +- +- BACKUP_DUPLICITY_URL="file://$BUILD_DIR/outputfs" +-fi +diff --git a/usr/share/rear/restore/DUPLICITY/default/100_mount_duplicity_path.sh b/usr/share/rear/restore/DUPLICITY/default/100_mount_duplicity_path.sh +new file mode 120000 +index 00000000..7f558c5d +--- /dev/null ++++ b/usr/share/rear/restore/DUPLICITY/default/100_mount_duplicity_path.sh +@@ -0,0 +1 @@ ++../../../backup/DUPLICITY/default/100_mount_duplicity_path.sh +\ No newline at end of file +diff --git a/usr/share/rear/restore/DUPLICITY/default/980_unmount_duplicity_path.sh b/usr/share/rear/restore/DUPLICITY/default/980_unmount_duplicity_path.sh +deleted file mode 100644 +index 60aa811e..00000000 +--- a/usr/share/rear/restore/DUPLICITY/default/980_unmount_duplicity_path.sh ++++ /dev/null +@@ -1,15 +0,0 @@ +-# umount mountpoint +-if [ -n "$BACKUP_DUPLICITY_NETFS_URL" -o -n "$BACKUP_DUPLICITY_NETFS_UMOUNTCMD" ]; then +- +- if [[ "$BACKUP_DUPLICITY_NETFS_UMOUNTCMD" ]] ; then +- BACKUP_DUPLICITY_NETFS_URL="var://BACKUP_DUPLICITY_NETFS_UMOUNTCMD" +- fi +- +- umount_url $BACKUP_DUPLICITY_NETFS_URL $BUILD_DIR/outputfs +- +- rmdir $v $BUILD_DIR/outputfs >&2 +- if [[ $? -eq 0 ]] ; then +- # the argument to RemoveExitTask has to be identical to the one given to AddExitTask +- RemoveExitTask "rmdir $v $BUILD_DIR/outputfs >&2" +- fi +-fi +diff --git a/usr/share/rear/restore/DUPLICITY/default/980_unmount_duplicity_path.sh b/usr/share/rear/restore/DUPLICITY/default/980_unmount_duplicity_path.sh +new file mode 120000 +index 00000000..b7e47be1 +--- /dev/null ++++ b/usr/share/rear/restore/DUPLICITY/default/980_unmount_duplicity_path.sh +@@ -0,0 +1 @@ ++../../../backup/DUPLICITY/default/980_unmount_duplicity_path.sh +\ No newline at end of file +diff --git a/usr/share/rear/restore/YUM/default/100_mount_YUM_path.sh b/usr/share/rear/restore/YUM/default/100_mount_YUM_path.sh +deleted file mode 100644 +index 7de92af4..00000000 +--- a/usr/share/rear/restore/YUM/default/100_mount_YUM_path.sh ++++ /dev/null +@@ -1,13 +0,0 @@ +-# Copied from ../../NETFS/default/100_mount_NETFS_path.sh a.k.a. ../../../backup/NETFS/default/100_mount_NETFS_path.sh for YUM +- +-# create mount point +-mkdir -p $v "$BUILD_DIR/outputfs" >&2 +-StopIfError "Could not mkdir '$BUILD_DIR/outputfs'" +- +-AddExitTask "rmdir $v $BUILD_DIR/outputfs >&2" +- +-if [[ "$BACKUP_MOUNTCMD" ]] ; then +- BACKUP_URL="var://BACKUP_MOUNTCMD" +-fi +- +-mount_url $BACKUP_URL $BUILD_DIR/outputfs $BACKUP_OPTIONS +diff --git a/usr/share/rear/restore/YUM/default/100_mount_YUM_path.sh b/usr/share/rear/restore/YUM/default/100_mount_YUM_path.sh +new file mode 120000 +index 00000000..60e0f83f +--- /dev/null ++++ b/usr/share/rear/restore/YUM/default/100_mount_YUM_path.sh +@@ -0,0 +1 @@ ++../../NETFS/default/100_mount_NETFS_path.sh +\ No newline at end of file +diff --git a/usr/share/rear/restore/YUM/default/980_umount_YUM_dir.sh b/usr/share/rear/restore/YUM/default/980_umount_YUM_dir.sh +deleted file mode 100644 +index d02dcf34..00000000 +--- a/usr/share/rear/restore/YUM/default/980_umount_YUM_dir.sh ++++ /dev/null +@@ -1,15 +0,0 @@ +-# Copied from ../../../backup/NETFS/default/980_umount_NETFS_dir.sh for YUM +- +-# umount NETFS mountpoint +- +-if [[ "$BACKUP_UMOUNTCMD" ]] ; then +- BACKUP_URL="var://BACKUP_UMOUNTCMD" +-fi +- +-umount_url $BACKUP_URL $BUILD_DIR/outputfs +- +-rmdir $v $BUILD_DIR/outputfs >&2 +-if [[ $? -eq 0 ]] ; then +- # the argument to RemoveExitTask has to be identical to the one given to AddExitTask +- RemoveExitTask "rmdir $v $BUILD_DIR/outputfs >&2" +-fi +diff --git a/usr/share/rear/restore/YUM/default/980_umount_YUM_dir.sh b/usr/share/rear/restore/YUM/default/980_umount_YUM_dir.sh +new file mode 120000 +index 00000000..2c29cb57 +--- /dev/null ++++ b/usr/share/rear/restore/YUM/default/980_umount_YUM_dir.sh +@@ -0,0 +1 @@ ++../../NETFS/default/980_umount_NETFS_dir.sh +\ No newline at end of file +diff --git a/usr/share/rear/verify/DUPLICITY/default/100_mount_duplicity_path.sh b/usr/share/rear/verify/DUPLICITY/default/100_mount_duplicity_path.sh +deleted file mode 100644 +index 64b7a792..00000000 +--- a/usr/share/rear/verify/DUPLICITY/default/100_mount_duplicity_path.sh ++++ /dev/null +@@ -1,15 +0,0 @@ +-# create mount point +-if [ -n "$BACKUP_DUPLICITY_NETFS_URL" -o -n "$BACKUP_DUPLICITY_NETFS_MOUNTCMD" ]; then +- mkdir -p $v "$BUILD_DIR/outputfs" >&2 +- StopIfError "Could not mkdir '$BUILD_DIR/outputfs'" +- +- AddExitTask "rmdir $v $BUILD_DIR/outputfs >&2" +- +- if [[ "$BACKUP_DUPLICITY_NETFS_MOUNTCMD" ]] ; then +- BACKUP_DUPLICITY_NETFS_URL="var://BACKUP_DUPLICITY_NETFS_MOUNTCMD" +- fi +- +- mount_url $BACKUP_DUPLICITY_NETFS_URL $BUILD_DIR/outputfs $BACKUP_DUPLICITY_NETFS_OPTIONS +- +- BACKUP_DUPLICITY_URL="file://$BUILD_DIR/outputfs" +-fi +diff --git a/usr/share/rear/verify/DUPLICITY/default/100_mount_duplicity_path.sh b/usr/share/rear/verify/DUPLICITY/default/100_mount_duplicity_path.sh +new file mode 120000 +index 00000000..7f558c5d +--- /dev/null ++++ b/usr/share/rear/verify/DUPLICITY/default/100_mount_duplicity_path.sh +@@ -0,0 +1 @@ ++../../../backup/DUPLICITY/default/100_mount_duplicity_path.sh +\ No newline at end of file +diff --git a/usr/share/rear/verify/DUPLICITY/default/980_unmount_duplicity_path.sh b/usr/share/rear/verify/DUPLICITY/default/980_unmount_duplicity_path.sh +deleted file mode 100644 +index 60aa811e..00000000 +--- a/usr/share/rear/verify/DUPLICITY/default/980_unmount_duplicity_path.sh ++++ /dev/null +@@ -1,15 +0,0 @@ +-# umount mountpoint +-if [ -n "$BACKUP_DUPLICITY_NETFS_URL" -o -n "$BACKUP_DUPLICITY_NETFS_UMOUNTCMD" ]; then +- +- if [[ "$BACKUP_DUPLICITY_NETFS_UMOUNTCMD" ]] ; then +- BACKUP_DUPLICITY_NETFS_URL="var://BACKUP_DUPLICITY_NETFS_UMOUNTCMD" +- fi +- +- umount_url $BACKUP_DUPLICITY_NETFS_URL $BUILD_DIR/outputfs +- +- rmdir $v $BUILD_DIR/outputfs >&2 +- if [[ $? -eq 0 ]] ; then +- # the argument to RemoveExitTask has to be identical to the one given to AddExitTask +- RemoveExitTask "rmdir $v $BUILD_DIR/outputfs >&2" +- fi +-fi +diff --git a/usr/share/rear/verify/DUPLICITY/default/980_unmount_duplicity_path.sh b/usr/share/rear/verify/DUPLICITY/default/980_unmount_duplicity_path.sh +new file mode 120000 +index 00000000..b7e47be1 +--- /dev/null ++++ b/usr/share/rear/verify/DUPLICITY/default/980_unmount_duplicity_path.sh +@@ -0,0 +1 @@ ++../../../backup/DUPLICITY/default/980_unmount_duplicity_path.sh +\ No newline at end of file +diff --git a/usr/share/rear/verify/YUM/default/050_check_YUM_requirements.sh b/usr/share/rear/verify/YUM/default/050_check_YUM_requirements.sh +deleted file mode 100644 +index cfd70026..00000000 +--- a/usr/share/rear/verify/YUM/default/050_check_YUM_requirements.sh ++++ /dev/null +@@ -1,116 +0,0 @@ +-# Copied from ../../../prep/NETFS/default/050_check_NETFS_requirements.sh for YUM +-# BACKUP_URL=[proto]://[host]/[share] +-# example: nfs://lucky/temp/backup +-# example: cifs://lucky/temp +-# example: usb:///dev/sdb1 +-# example: tape:///dev/nst0 +-# example: file:///path +-# example: iso://backup/ +-# example: sshfs://user@host/G/rear/ +-# example: ftpfs://user:password@host/rear/ (the password part is optional) +- +-[[ "$BACKUP_URL" || "$BACKUP_MOUNTCMD" ]] +-# FIXME: The above test does not match the error message below. +-# To match the the error message the test should be +-# [[ "$BACKUP_URL" || ( "$BACKUP_MOUNTCMD" && "$BACKUP_UMOUNTCMD" ) ]] +-# but I cannot decide if there is a subtle reason for the omission. +-StopIfError "You must specify either BACKUP_URL or BACKUP_MOUNTCMD and BACKUP_UMOUNTCMD !" +- +-if [[ "$BACKUP_URL" ]] ; then +- local scheme=$( url_scheme $BACKUP_URL ) +- local hostname=$( url_hostname $BACKUP_URL ) +- local path=$( url_path $BACKUP_URL ) +- +- ### check for vaild BACKUP_URL schemes +- ### see https://github.com/rear/rear/issues/842 +- case $scheme in +- (nfs|cifs|usb|tape|file|iso|sshfs|ftpfs) +- # do nothing for vaild BACKUP_URL schemes +- : +- ;; +- (*) +- Error "Invalid scheme '$scheme' in BACKUP_URL '$BACKUP_URL' valid schemes: nfs cifs usb tape file iso sshfs ftpfs" +- ;; +- esac +- +- ### set other variables from BACKUP_URL +- if [[ "usb" = "$scheme" ]] ; then +- # if USB_DEVICE is not explicitly specified it is the path from BACKUP_URL +- [[ -z "$USB_DEVICE" ]] && USB_DEVICE="$path" +- fi +- +- ### check if host is reachable +- if [[ "$PING" && "$hostname" ]] ; then +- # Only LogPrintIfError but no StopIfError because it is not a fatal error +- # (i.e. not a reason to abort) when a host does not respond to a 'ping' +- # because hosts can be accessible via certain ports but do not respond to a 'ping' +- # cf. https://bugzilla.opensuse.org/show_bug.cgi?id=616706 +- # TODO: it would be better to test if it is accessible via the actually needed port(s) +- ping -c 2 "$hostname" >/dev/null +- LogPrintIfError "Host '$hostname' in BACKUP_URL '$BACKUP_URL' does not respond to a 'ping'." +- else +- Log "Skipping 'ping' test for host '$hostname' in BACKUP_URL '$BACKUP_URL'" +- fi +- +-fi +- +-# some backup progs require a different backuparchive name +-case "$(basename $BACKUP_PROG)" in +- (rsync) +- # rsync creates a target directory instead of a file +- BACKUP_PROG_SUFFIX= +- BACKUP_PROG_COMPRESS_SUFFIX= +- ;; +- (*) +- : +- ;; +-esac +- +-# include required programs +-# the code below includes mount.* and umount.* programs for all non-empty schemes +-# (i.e. for any non-empty BACKUP_URL like usb tape file sshfs ftpfs) +-# and it includes 'mount.' for empty schemes (e.g. if BACKUP_URL is not set) +-# which is o.k. because it is a catch all rule so we do not miss any +-# important executable needed a certain scheme and it does not hurt +-# see https://github.com/rear/rear/pull/859 +-PROGS+=( +-showmount +-mount.$(url_scheme $BACKUP_URL) +-umount.$(url_scheme $BACKUP_URL) +-$( test "$BACKUP_MOUNTCMD" && echo "${BACKUP_MOUNTCMD%% *}" ) +-$( test "$BACKUP_UMOUNTCMD" && echo "${BACKUP_UMOUNTCMD%% *}" ) +-$BACKUP_PROG +-gzip +-bzip2 +-xz +-) +- +-# include required stuff for sshfs or ftpfs (via CurlFtpFS) +-if [[ "sshfs" = "$scheme" || "ftpfs" = "$scheme" ]] ; then +- # both sshfs and ftpfs (via CurlFtpFS) are based on FUSE +- PROGS+=( fusermount mount.fuse ) +- MODULES+=( fuse ) +- MODULES_LOAD+=( fuse ) +- COPY_AS_IS+=( /etc/fuse.conf ) +- # include what is specific for sshfs +- if [[ "sshfs" = "$scheme" ]] ; then +- # see http://sourceforge.net/apps/mediawiki/fuse/index.php?title=SshfsFaq +- REQUIRED_PROGS+=( sshfs ssh ) +- # relying on 500_ssh.sh to take a long the SSH related files +- fi +- # include what is specific for ftpfs +- if [[ "ftpfs" = "$scheme" ]] ; then +- # see http://curlftpfs.sourceforge.net/ +- # and https://github.com/rear/rear/issues/845 +- REQUIRED_PROGS+=( curlftpfs ) +- fi +-fi +- +-# include required modules, like nfs cifs ... +-# the code below includes modules for all non-empty schemes +-# (i.e. for any non-empty BACKUP_URL like usb tape file sshfs ftpfs) +-# which is o.k. because this must been seen as a catch all rule +-# (one never knows what one could miss) +-# see https://github.com/rear/rear/pull/859 +-MODULES+=( $(url_scheme $BACKUP_URL) ) +- +diff --git a/usr/share/rear/verify/YUM/default/050_check_YUM_requirements.sh b/usr/share/rear/verify/YUM/default/050_check_YUM_requirements.sh +new file mode 120000 +index 00000000..af1512d6 +--- /dev/null ++++ b/usr/share/rear/verify/YUM/default/050_check_YUM_requirements.sh +@@ -0,0 +1 @@ ++../../NETFS/default/050_check_NETFS_requirements.sh +\ No newline at end of file +diff --git a/usr/share/rear/verify/YUM/default/060_mount_YUM_path.sh b/usr/share/rear/verify/YUM/default/060_mount_YUM_path.sh +deleted file mode 100644 +index f7e31ed6..00000000 +--- a/usr/share/rear/verify/YUM/default/060_mount_YUM_path.sh ++++ /dev/null +@@ -1,12 +0,0 @@ +-# Copied from ../../../backup/NETFS/default/100_mount_NETFS_path.sh for YUM +-# create mount point +-mkdir -p $v "$BUILD_DIR/outputfs" >&2 +-StopIfError "Could not mkdir '$BUILD_DIR/outputfs'" +- +-AddExitTask "rmdir $v $BUILD_DIR/outputfs >&2" +- +-if [[ "$BACKUP_MOUNTCMD" ]] ; then +- BACKUP_URL="var://BACKUP_MOUNTCMD" +-fi +- +-mount_url $BACKUP_URL $BUILD_DIR/outputfs $BACKUP_OPTIONS +diff --git a/usr/share/rear/verify/YUM/default/060_mount_YUM_path.sh b/usr/share/rear/verify/YUM/default/060_mount_YUM_path.sh +new file mode 120000 +index 00000000..73dd4697 +--- /dev/null ++++ b/usr/share/rear/verify/YUM/default/060_mount_YUM_path.sh +@@ -0,0 +1 @@ ++../../../restore/YUM/default/100_mount_YUM_path.sh +\ No newline at end of file +diff --git a/usr/share/rear/verify/YUM/default/070_set_backup_archive.sh b/usr/share/rear/verify/YUM/default/070_set_backup_archive.sh +deleted file mode 100644 +index 86d1708d..00000000 +--- a/usr/share/rear/verify/YUM/default/070_set_backup_archive.sh ++++ /dev/null +@@ -1,300 +0,0 @@ +-# Copied from ../../../prep/NETFS/default/070_set_backup_archive.sh for YUM +-### Determine the name of the backup archive +-### This needs to be after we special case USB devices. +- +-# FIXME: backuparchive is no local variable (regardless that it is lowercased) +- +-# If TAPE_DEVICE is specified, use that: +-if test "$TAPE_DEVICE" ; then +- backuparchive="$TAPE_DEVICE" +- LogPrint "Using backup archive '$backuparchive'" +- return +-fi +- +-local backup_file_suffix="$BACKUP_PROG_SUFFIX$BACKUP_PROG_COMPRESS_SUFFIX" +-local backup_file_name="$BACKUP_PROG_ARCHIVE$backup_file_suffix" +- +-local scheme=$( url_scheme $BACKUP_URL ) +-local path=$( url_path $BACKUP_URL ) +-case "$scheme" in +- (file|iso) +- # Define the output path according to the scheme +- local outputpath=$( backup_path $scheme $path ) +- backuparchive="$outputpath/$backup_file_name" +- LogPrint "Using backup archive '$backuparchive'" +- return +- ;; +- (tape) +- # TODO: Check if that case is really needed. +- # Perhaps prep/default/030_translate_tape.sh does already all what is needed. +- backuparchive=$path +- LogPrint "Using backup archive '$backuparchive'" +- return +- ;; +-esac +- +-local backup_directory=$BUILD_DIR/outputfs/$NETFS_PREFIX +- +-# Normal (i.e. non-incremental/non-differential) backup: +-if ! test "incremental" = "$BACKUP_TYPE" -o "differential" = "$BACKUP_TYPE" ; then +- # In case of normal (i.e. non-incremental) backup there is only one restore archive +- # and its name is the same as the backup archive (usually 'backup.tar.gz'): +- backuparchive="$backup_directory/$backup_file_name" +- LogPrint "Using backup archive '$backuparchive'" +- # This script is also run during "rear recover/restoreonly" where RESTORE_ARCHIVES must be set. +- local backup_restore_workflows=( "recover" "restoreonly" ) +- if IsInArray $WORKFLOW ${backup_restore_workflows[@]} ; then +- # Only set RESTORE_ARCHIVES the backup archive is actually accessible +- # cf. https://github.com/rear/rear/issues/1166 +- if test -r "$backuparchive" ; then +- RESTORE_ARCHIVES=( "$backuparchive" ) +- else +- # In case of USB backup there is the subsequent 540_choose_backup_archive.sh script +- # that shows a backup selection dialog when RESTORE_ARCHIVES is not already set. +- if test "usb" = "$scheme" ; then +- LogPrint "Backup archive '$backuparchive' not readable. Need to select another one." +- else +- Error "Backup archive '$backuparchive' not readable." +- fi +- fi +- fi +- return +-fi +- +-# Incremental or differential backup: +-set -e -u -o pipefail +-# Incremental or differential backup only works for the NETFS backup method +-# and only with the 'tar' backup program: +-if ! test "NETFS" = "$BACKUP" -a "tar" = "$BACKUP_PROG" ; then +- Error "BACKUP_TYPE incremental or differential only works with BACKUP=NETFS and BACKUP_PROG=tar" +-fi +-# Incremental or differential backup is currently only known to work with BACKUP_URL=nfs://. +-# Other BACKUP_URL schemes may work and at least BACKUP_URL=usb:///... needs special setup +-# to work with incremental or differential backup (see https://github.com/rear/rear/issues/1145): +-if test "usb" = "$scheme" ; then +- # When USB_SUFFIX is set the compliance mode is used where +- # backup on USB works in compliance with backup on NFS which means +- # a fixed backup directory where incremental or differential backups work. +- # Use plain $USB_SUFFIX and not "$USB_SUFFIX" because when USB_SUFFIX contains only blanks +- # test "$USB_SUFFIX" would result true because test " " results true: +- test $USB_SUFFIX || Error "BACKUP_TYPE incremental or differential requires USB_SUFFIX for BACKUP_URL=usb" +-fi +-# Incremental or differential backup and keeping old backup contradict each other (mutual exclusive) +-# so that NETFS_KEEP_OLD_BACKUP_COPY must not be 'true' in case of incremental or differential backup: +-if test "$NETFS_KEEP_OLD_BACKUP_COPY" ; then +- NETFS_KEEP_OLD_BACKUP_COPY="" +- LogPrint "Disabled NETFS_KEEP_OLD_BACKUP_COPY because BACKUP_TYPE incremental or differential does not work with that" +-fi +-# For incremental or differential backup some date values (weekday, YYYY-MM-DD, HHMM) are needed +-# that must be consistent for one single point of the current time which means +-# one cannot call the 'date' command several times because then there would be +-# a small probability that e.g. weekday, YYYY-MM-DD, HHMM do not match +-# one single point in time (in particular when midnight passes in between). +-# Therefore the output of one single 'date' call is storend in an array and +-# the array elements are then assinged to individual variables as needed: +-local current_date_output=( $( date '+%a %Y-%m-%d %H%M' ) ) +-local current_weekday="${current_date_output[0]}" +-local current_yyyy_mm_dd="${current_date_output[1]}" +-local current_hhmm="${current_date_output[2]}" +-# The date FULLBACKUP_OUTDATED_DAYS ago is needed to check if the latest full backup is too old. +-# When the latest full backup is more than FULLBACKUP_OUTDATED_DAYS ago a new full backup is made. +-# This separated call of the 'date' command which is technically needed because it is +-# for another point in time (e.g. 7 days ago) is run after the above call of the 'date' +-# command for the current time to be on the safe side when midnight passes in between +-# both 'date' commands which would then result that a new full backup is made +-# when the latest full backup is basically right now FULLBACKUP_OUTDATED_DAYS ago because +-# the stored date of the latest full backup is the current date at the time when it was made. +-# Example (assuming FULLBACKUP_OUTDATED_DAYS=7 ): +-# The latest full backup was made on Sunday January 10 in 2016 (just before midnight). +-# One week later this script runs again while midnight passes between the two 'date' calls +-# so that current_date_output[@]="Sun 2016-01-17 0000" (still Sunday January 17 in 2016) +-# and yyyymmdd_max_days_ago=20160111 (already Monday January 11 in 2016), then +-# Sunday January 10 is older than Monday January 11 so that a new full backup is made: +-test "$FULLBACKUP_OUTDATED_DAYS" || FULLBACKUP_OUTDATED_DAYS="7" +-local yyyymmdd_max_days_ago=$( date '+%Y%m%d' --date="$FULLBACKUP_OUTDATED_DAYS days ago" ) +-# Full backup file names are of the form YYYY-MM-DD-HHMM-F.tar.gz +-# where the 'F' denotes a full backup: +-local full_backup_marker="F" +-# Incremental backup file names are of the form YYYY-MM-DD-HHMM-I.tar.gz +-# where the 'I' denotes an incremental backup: +-local incremental_backup_marker="I" +-# Differential backup file names are of the form YYYY-MM-DD-HHMM-D.tar.gz +-# where the last 'D' denotes a differential backup: +-local differential_backup_marker="D" +-# In case of incremental or differential backup the RESTORE_ARCHIVES contains +-# first the latest full backup file. +-# In case of incremental backup the RESTORE_ARCHIVES contains +-# after the latest full backup file each incremental backup +-# in the ordering how they must be restored. +-# For example when the latest full backup was made on Sunday +-# plus each subsequent weekday a separated incremental backup was made, +-# then during a "rear recover" on Wednesday morning +-# first the full backup from Sunday has to be restored, +-# then the incremental backup from Monday, and +-# finally the incremental backup from Tuesday. +-# In case of differential backup the RESTORE_ARCHIVES contains +-# after the latest full backup file the latest differential backup. +-# For example when the latest full backup was made on Sunday +-# plus each subsequent weekday a separated differential backup was made, +-# then during a "rear recover" on Wednesday morning +-# first the full backup from Sunday has to be restored, +-# and finally the differential backup from Tuesday +-# (i.e. the differential backup from Monday is skipped). +-# The date format YYYY-MM-DD that is used here is crucial. +-# It is the ISO 8601 format 'year-month-day' to specify a day of a year +-# that is accepted by 'tar' for the '--newer' option, +-# see the GNU tar manual section "Operating Only on New Files" +-# at https://www.gnu.org/software/tar/manual/html_node/after.html +-# and the GNU tar manual section "Calendar date items" +-# at https://www.gnu.org/software/tar/manual/html_node/Calendar-date-items.html#SEC124 +-local date_glob_regex="[0-9][0-9][0-9][0-9]-[0-9][0-9]-[0-9][0-9]" +-local date_time_glob_regex="$date_glob_regex-[0-9][0-9][0-9][0-9]" +-# Determine what kind of backup must be created, 'full' or 'incremental' or 'differential' +-# (the empty default means it is undecided what kind of backup must be created): +-local create_backup_type="" +-# Code regarding creating a backup is useless during "rear recover" and +-# messages about creating a backup are misleading during "rear recover": +-local recovery_workflows=( "recover" "layoutonly" "restoreonly" ) +-if ! IsInArray $WORKFLOW ${recovery_workflows[@]} ; then +- # When today is a specified full backup day, do a full backup in any case +- # (regardless if there is already a full backup of this day): +- if IsInArray "$current_weekday" "${FULLBACKUPDAY[@]}" ; then +- create_backup_type="full" +- LogPrint "Today's weekday ('$current_weekday') is a full backup day that triggers a new full backup in any case" +- fi +-fi +-# Get the latest full backup (if exists): +-local full_backup_glob_regex="$date_time_glob_regex-$full_backup_marker$backup_file_suffix" +-# Here things like 'find /path/to/dir -name '*.tar.gz' | sort' are used because +-# one cannot use bash globbing via commands like 'ls /path/to/dir/*.tar.gz' +-# because /usr/sbin/rear sets the nullglob bash option which leads to plain 'ls' +-# when '/path/to/dir/*.tar.gz' matches nothing (i.e. when no backup file exists) +-# so that then plain 'ls' would result nonsense. +-local latest_full_backup=$( find $backup_directory -name "$full_backup_glob_regex" | sort | tail -n1 ) +-# A latest full backup is found: +-if test "$latest_full_backup" ; then +- local latest_full_backup_file_name=$( basename "$latest_full_backup" ) +- # The full_or_incremental_backup_glob_regex is also needed below for non-"recover" WORKFLOWs +- # to set the right variables for creating an incremental backup: +- local full_or_incremental_backup_glob_regex="$date_time_glob_regex-[$full_backup_marker$incremental_backup_marker]$backup_file_suffix" +- # Code regarding creating a backup is useless during "rear recover" and +- # messages about creating a backup are misleading during "rear recover": +- if ! IsInArray $WORKFLOW ${recovery_workflows[@]} ; then +- # There is nothing to do here if it is already decided that +- # a full backup must be created (see "full backup day" above"): +- if ! test "full" = "$create_backup_type" ; then +- local latest_full_backup_date=$( echo $latest_full_backup_file_name | grep -o "$date_glob_regex" ) +- local yyyymmdd_latest_full_backup=$( echo $latest_full_backup_date | tr -d '-' ) +- # Check if the latest full backup is too old: +- if test $yyyymmdd_latest_full_backup -lt $yyyymmdd_max_days_ago ; then +- create_backup_type="full" +- LogPrint "Latest full backup date '$latest_full_backup_date' too old (more than $FULLBACKUP_OUTDATED_DAYS days ago) triggers new full backup" +- else +- # When a latest full backup is found that is not too old +- # a BACKUP_TYPE (incremental or differential) backup will be created: +- create_backup_type="$BACKUP_TYPE" +- LogPrint "Latest full backup found ($latest_full_backup_file_name) triggers $BACKUP_TYPE backup" +- fi +- fi +- else +- # This script is also run during "rear recover" where RESTORE_ARCHIVES must be set: +- case "$BACKUP_TYPE" in +- (incremental) +- # When a latest full backup is found use that plus all later incremental backups for restore: +- # The following command is a bit tricky: +- # It lists all YYYY-MM-DD-HHMM-F.tar.gz and all YYYY-MM-DD-HHMM-I.tar.gz files in the backup directory and sorts them +- # and finally it outputs only those that match the latest full backup file name and incremental backups that got sorted after that +- # where it is mandatory that the backup file names sort by date (i.e. date must be the leading part of the backup file names): +- RESTORE_ARCHIVES=( $( find $backup_directory -name "$full_or_incremental_backup_glob_regex" | sort | sed -n -e "/$latest_full_backup_file_name/,\$p" ) ) +- ;; +- (differential) +- # For differential backup use the latest full backup plus the one latest differential backup for restore: +- # The following command is a bit tricky: +- # It lists all YYYY-MM-DD-HHMM-F.tar.gz and all YYYY-MM-DD-HHMM-D.tar.gz files in the backup directory and sorts them +- # then it outputs only those that match the latest full backup file name and all differential backups that got sorted after that +- # and then it outputs only the first line (i.e. the full backup) and the last line (i.e. the latest differential backup) +- # but when no differential backup exists (i.e. when only the full backup exists) the first line is also the last line +- # so that "sed -n -e '1p;$p'" outputs the full backup twice which is corrected by the final "sort -u": +- local full_or_differential_backup_glob_regex="$date_time_glob_regex-[$full_backup_marker$differential_backup_marker]$backup_file_suffix" +- RESTORE_ARCHIVES=( $( find $backup_directory -name "$full_or_differential_backup_glob_regex" | sort | sed -n -e "/$latest_full_backup_file_name/,\$p" | sed -n -e '1p;$p' | sort -u ) ) +- ;; +- (*) +- BugError "Unexpected BACKUP_TYPE '$BACKUP_TYPE'" +- ;; +- esac +- # Tell the user what will be restored: +- local restore_archives_file_names="" +- for restore_archive in "${RESTORE_ARCHIVES[@]}" ; do +- restore_archives_file_names="$restore_archives_file_names $( basename "$restore_archive" )" +- done +- LogPrint "For backup restore using $restore_archives_file_names" +- fi +-# No latest full backup is found: +-else +- # Code regarding creating a backup is useless during "rear recover" and +- # messages about creating a backup are misleading during "rear recover": +- if ! IsInArray $WORKFLOW ${recovery_workflows[@]} ; then +- # If no latest full backup is found create one during "rear mkbackup": +- create_backup_type="full" +- LogPrint "No full backup found (YYYY-MM-DD-HHMM-F.tar.gz) triggers full backup" +- else +- # This script is also run during "rear recover" where RESTORE_ARCHIVES must be set: +- # If no latest full backup is found (i.e. no file name matches the YYYY-MM-DD-HHMM-F.tar.gz form) +- # fall back to what is done in case of normal (i.e. non-incremental/non-differential) backup +- # and hope for the best (i.e. that a backup_directory/backup_file_name actually exists). +- # In case of normal (i.e. non-incremental/non-differential) backup there is only one restore archive +- # and its name is the same as the backup archive (usually 'backup.tar.gz'). +- # This is only a fallback setting to be more on the safe side for "rear recover". +- # Initially for the very fist run of incremental backup during "rear mkbackup" +- # a full backup file of the YYYY-MM-DD-HHMM-F.tar.gz form will be created. +- RESTORE_ARCHIVES=( "$backup_directory/$backup_file_name" ) +- LogPrint "Using $backup_file_name for backup restore" +- fi +-fi +-# Code regarding creating a backup is useless during "rear recover" and +-# messages about creating a backup are misleading during "rear recover": +-if ! IsInArray $WORKFLOW ${recovery_workflows[@]} ; then +- # Set the right variables for creating a backup (but do not actually do anything at this point): +- case "$create_backup_type" in +- (full) +- local new_full_backup_file_name="$current_yyyy_mm_dd-$current_hhmm-$full_backup_marker$backup_file_suffix" +- backuparchive="$backup_directory/$new_full_backup_file_name" +- BACKUP_PROG_CREATE_NEWER_OPTIONS="-V $new_full_backup_file_name" +- LogPrint "Performing full backup using backup archive '$new_full_backup_file_name'" +- ;; +- (incremental) +- local new_incremental_backup_file_name="$current_yyyy_mm_dd-$current_hhmm-$incremental_backup_marker$backup_file_suffix" +- backuparchive="$backup_directory/$new_incremental_backup_file_name" +- # Get the latest latest incremental backup that is based on the latest full backup (if exists): +- local incremental_backup_glob_regex="$date_time_glob_regex-$incremental_backup_marker$backup_file_suffix" +- # First get the latest full backup plus all later incremental backups (cf. how RESTORE_ARCHIVES is set in case of incremental backup) +- # then grep only the incremental backups and from the incremental backups use only the last one (if exists): +- local latest_incremental_backup=$( find $backup_directory -name "$full_or_incremental_backup_glob_regex" | sort | sed -n -e "/$latest_full_backup_file_name/,\$p" | grep "$incremental_backup_glob_regex" | tail -n1 ) +- if test "$latest_incremental_backup" ; then +- # A latest incremental backup that is based on the latest full backup is found: +- local latest_incremental_backup_file_name=$( basename $latest_incremental_backup ) +- LogPrint "Latest incremental backup found ($latest_incremental_backup_file_name) that is newer than the latest full backup" +- local latest_incremental_backup_date=$( echo $latest_incremental_backup_file_name | grep -o "$date_glob_regex" ) +- BACKUP_PROG_CREATE_NEWER_OPTIONS="--newer=$latest_incremental_backup_date -V $latest_incremental_backup_file_name" +- LogPrint "Performing incremental backup for files newer than $latest_incremental_backup_date using backup archive '$new_incremental_backup_file_name'" +- else +- # When there is not yet an incremental backup that is based on the latest full backup +- # the new created incremental backup must be based on the latest full backup: +- BACKUP_PROG_CREATE_NEWER_OPTIONS="--newer=$latest_full_backup_date -V $latest_full_backup_file_name" +- LogPrint "Performing incremental backup for files newer than $latest_full_backup_date using backup archive '$new_incremental_backup_file_name'" +- fi +- ;; +- (differential) +- local new_differential_backup_file_name="$current_yyyy_mm_dd-$current_hhmm-$differential_backup_marker$backup_file_suffix" +- backuparchive="$backup_directory/$new_differential_backup_file_name" +- BACKUP_PROG_CREATE_NEWER_OPTIONS="--newer=$latest_full_backup_date -V $latest_full_backup_file_name" +- LogPrint "Performing differential backup for files newer than $latest_full_backup_date using backup archive '$new_differential_backup_file_name'" +- ;; +- (*) +- BugError "Unexpected create_backup_type '$create_backup_type'" +- ;; +- esac +-fi +-# Go back from "set -e -u -o pipefail" to the defaults: +-apply_bash_flags_and_options_commands "$DEFAULT_BASH_FLAGS_AND_OPTIONS_COMMANDS" +- +diff --git a/usr/share/rear/verify/YUM/default/070_set_backup_archive.sh b/usr/share/rear/verify/YUM/default/070_set_backup_archive.sh +new file mode 120000 +index 00000000..b8de3d9e +--- /dev/null ++++ b/usr/share/rear/verify/YUM/default/070_set_backup_archive.sh +@@ -0,0 +1 @@ ++../../../prep/YUM/default/070_set_backup_archive.sh +\ No newline at end of file +diff --git a/usr/share/rear/verify/YUM/default/980_umount_YUM_dir.sh b/usr/share/rear/verify/YUM/default/980_umount_YUM_dir.sh +deleted file mode 100644 +index dc719e38..00000000 +--- a/usr/share/rear/verify/YUM/default/980_umount_YUM_dir.sh ++++ /dev/null +@@ -1,14 +0,0 @@ +-# Copied from ../../../backup/NETFS/default/980_umount_NETFS_dir.sh for YUM +-# umount NETFS mountpoint +- +-if [[ "$BACKUP_UMOUNTCMD" ]] ; then +- BACKUP_URL="var://BACKUP_UMOUNTCMD" +-fi +- +-umount_url $BACKUP_URL $BUILD_DIR/outputfs +- +-rmdir $v $BUILD_DIR/outputfs >&2 +-if [[ $? -eq 0 ]] ; then +- # the argument to RemoveExitTask has to be identical to the one given to AddExitTask +- RemoveExitTask "rmdir $v $BUILD_DIR/outputfs >&2" +-fi +diff --git a/usr/share/rear/verify/YUM/default/980_umount_YUM_dir.sh b/usr/share/rear/verify/YUM/default/980_umount_YUM_dir.sh +new file mode 120000 +index 00000000..ada5ea50 +--- /dev/null ++++ b/usr/share/rear/verify/YUM/default/980_umount_YUM_dir.sh +@@ -0,0 +1 @@ ++../../../restore/YUM/default/980_umount_YUM_dir.sh +\ No newline at end of file diff --git a/SOURCES/rear-bz1983013.patch b/SOURCES/rear-bz1983013.patch new file mode 100644 index 0000000..f8032bb --- /dev/null +++ b/SOURCES/rear-bz1983013.patch @@ -0,0 +1,68 @@ +diff --git a/usr/share/rear/conf/Linux-ppc64.conf b/usr/share/rear/conf/Linux-ppc64.conf +index 7e20ddc7..d7774062 100644 +--- a/usr/share/rear/conf/Linux-ppc64.conf ++++ b/usr/share/rear/conf/Linux-ppc64.conf +@@ -1,18 +1,26 @@ +-REQUIRED_PROGS+=( sfdisk ) ++REQUIRED_PROGS+=( sfdisk ofpathname ) + + PROGS+=( + mkofboot + ofpath + ybin + yabootconfig +-bootlist + pseries_platform + nvram +-ofpathname + bc + agetty + ) + ++if grep -q "emulated by qemu" /proc/cpuinfo ; then ++ # Qemu/KVM virtual machines don't need bootlist - don't complain if ++ # it is missing ++ PROGS+=( bootlist ) ++else ++ # PowerVM environment, we need to run bootlist, otherwise ++ # we can't make the system bpotable. Be strict about requiring it ++ REQUIRED_PROGS+=( bootlist ) ++fi ++ + COPY_AS_IS+=( + /usr/lib/yaboot/yaboot + /usr/lib/yaboot/ofboot +diff --git a/usr/share/rear/conf/Linux-ppc64le.conf b/usr/share/rear/conf/Linux-ppc64le.conf +index d00154a2..df8066ea 100644 +--- a/usr/share/rear/conf/Linux-ppc64le.conf ++++ b/usr/share/rear/conf/Linux-ppc64le.conf +@@ -1,10 +1,8 @@ + REQUIRED_PROGS+=( sfdisk ) + + PROGS+=( +-bootlist + pseries_platform + nvram +-ofpathname + bc + agetty + ) +@@ -17,4 +15,18 @@ agetty + if [[ $(awk '/platform/ {print $NF}' < /proc/cpuinfo) != PowerNV ]] ; then + # No firmware files when ppc64le Linux is not run in BareMetal Mode (PowerNV): + test "${FIRMWARE_FILES[*]}" || FIRMWARE_FILES=( 'no' ) ++ # grub2-install for powerpc-ieee1275 calls ofpathname, so without it, ++ # the rescue system can't make the recovered system bootable ++ REQUIRED_PROGS+=( ofpathname ) ++ if grep -q "emulated by qemu" /proc/cpuinfo ; then ++ # Qemu/KVM virtual machines don't need bootlist - don't complain if ++ # it is missing ++ PROGS+=( bootlist ) ++ else ++ # PowerVM environment, we need to run bootlist, otherwise ++ # we can't make the system bpotable. Be strict about requiring it ++ REQUIRED_PROGS+=( bootlist ) ++ fi ++else ++ PROGS+=( ofpathname bootlist ) + fi diff --git a/SOURCES/rear-bz1993296.patch b/SOURCES/rear-bz1993296.patch new file mode 100644 index 0000000..15e65a2 --- /dev/null +++ b/SOURCES/rear-bz1993296.patch @@ -0,0 +1,34 @@ +From 4233fe30b315737ac8c4d857e2b04e021c2e2886 Mon Sep 17 00:00:00 2001 +From: Pavel Cahyna +Date: Mon, 16 Aug 2021 10:10:38 +0300 +Subject: [PATCH] Revert the main part of PR #2299 + +multipath -l is very slow with many multipath devices. As it will be +called for every multipath device, it leads to quadratic time complexity +in the number of multipath devices. For thousands of devices, ReaR can +take hours to scan and exclude them. We therefore have to comment +multipath -l out, as it is a huge performance regression, and find +another solution to bug #2298. +--- + usr/share/rear/lib/layout-functions.sh | 5 ++++- + 1 file changed, 4 insertions(+), 1 deletion(-) + +diff --git a/usr/share/rear/lib/layout-functions.sh b/usr/share/rear/lib/layout-functions.sh +index cdd81a14..8c8be74b 100644 +--- a/usr/share/rear/lib/layout-functions.sh ++++ b/usr/share/rear/lib/layout-functions.sh +@@ -771,7 +771,10 @@ function is_multipath_path { + # so that no "multipath -l" output could clutter the log (the "multipath -l" output is irrelevant here) + # in contrast to e.g. test "$( multipath -l )" that would falsely succeed with blank output + # and the output would appear in the log in 'set -x' debugscript mode: +- multipath -l | grep -q '[[:alnum:]]' || return 1 ++ # ++ # Unfortunately, multipat -l is quite slow with many multipath devices ++ # and becomes a performance bottleneck, so we must comment it out for now. ++ #multipath -l | grep -q '[[:alnum:]]' || return 1 + # Check if a block device should be a path in a multipath device: + multipath -c /dev/$1 &>/dev/null + } +-- +2.26.3 + diff --git a/SOURCES/rear-bz2035939.patch b/SOURCES/rear-bz2035939.patch new file mode 100644 index 0000000..30771c9 --- /dev/null +++ b/SOURCES/rear-bz2035939.patch @@ -0,0 +1,56 @@ +diff --git a/usr/share/rear/conf/default.conf b/usr/share/rear/conf/default.conf +index 0c230f38..f231bf3d 100644 +--- a/usr/share/rear/conf/default.conf ++++ b/usr/share/rear/conf/default.conf +@@ -2707,6 +2707,15 @@ WARN_MISSING_VOL_ID=1 + USE_CFG2HTML= + # The SKIP_CFG2HTML variable is no longer supported since ReaR 1.18 + ++# IP addresses that are present on the system but must be excluded when ++# building the network configuration used in recovery mode; this is typically ++# used when floating IP addresses are used on the system ++EXCLUDE_IP_ADDRESSES=() ++ ++# Network interfaces that are present on the system but must be excluded when ++# building the network configuration used in recovery mode ++EXCLUDE_NETWORK_INTERFACES=() ++ + # Simplify bonding setups by configuring always the first active device of a + # bond, except when mode is 4 (IEEE 802.3ad policy) + SIMPLIFY_BONDING=no +diff --git a/usr/share/rear/rescue/GNU/Linux/310_network_devices.sh b/usr/share/rear/rescue/GNU/Linux/310_network_devices.sh +index f806bfbf..2385f5b6 100644 +--- a/usr/share/rear/rescue/GNU/Linux/310_network_devices.sh ++++ b/usr/share/rear/rescue/GNU/Linux/310_network_devices.sh +@@ -355,6 +355,11 @@ function is_interface_up () { + local network_interface=$1 + local sysfspath=/sys/class/net/$network_interface + ++ if IsInArray "$network_interface" "${EXCLUDE_NETWORK_INTERFACES[@]}"; then ++ LogPrint "Excluding '$network_interface' per EXCLUDE_NETWORK_INTERFACES directive." ++ return 1 ++ fi ++ + local state=$( cat $sysfspath/operstate ) + if [ "$state" = "down" ] ; then + return 1 +@@ -403,11 +408,19 @@ function ipaddr_setup () { + if [ -n "$ipaddrs" ] ; then + # If some IP is found for the network interface, then use them + for ipaddr in $ipaddrs ; do ++ if IsInArray "${ipaddr%%/*}" "${EXCLUDE_IP_ADDRESSES[@]}"; then ++ LogPrint "Excluding IP address '$ipaddr' per EXCLUDE_IP_ADDRESSES directive even through it's defined in mapping file '$CONFIG_DIR/mappings/ip_addresses'." ++ continue ++ fi + echo "ip addr add $ipaddr dev $mapped_as" + done + else + # Otherwise, collect IP addresses for the network interface on the system + for ipaddr in $( ip a show dev $network_interface scope global | grep "inet.*\ " | tr -s " " | cut -d " " -f 3 ) ; do ++ if IsInArray "${ipaddr%%/*}" "${EXCLUDE_IP_ADDRESSES[@]}"; then ++ LogPrint "Excluding IP address '$ipaddr' per EXCLUDE_IP_ADDRESSES directive." ++ continue ++ fi + echo "ip addr add $ipaddr dev $mapped_as" + done + fi diff --git a/SOURCES/rear-bz2048454.patch b/SOURCES/rear-bz2048454.patch new file mode 100644 index 0000000..428505e --- /dev/null +++ b/SOURCES/rear-bz2048454.patch @@ -0,0 +1,78 @@ +diff --git a/usr/share/rear/layout/save/GNU/Linux/220_lvm_layout.sh b/usr/share/rear/layout/save/GNU/Linux/220_lvm_layout.sh +index 35be1721..d3c9ae86 100644 +--- a/usr/share/rear/layout/save/GNU/Linux/220_lvm_layout.sh ++++ b/usr/share/rear/layout/save/GNU/Linux/220_lvm_layout.sh +@@ -103,12 +103,7 @@ local lvs_exit_code + pdev=$( get_device_name $pdev ) + + # Output lvmdev entry to DISKLAYOUT_FILE: +- # With the above example the output is: +- # lvmdev /dev/system /dev/sda1 7wwpcO-KmNN-qsTE-7sp7-JBJS-vBdC-Zyt1W7 41940992 +- echo "lvmdev /dev/$vgrp $pdev $uuid $size" +- +- # After the 'lvmdev' line was written to disklayout.conf so that the user can inspect it +- # check that the required positional parameters in the 'lvmdev' line are non-empty ++ # Check that the required positional parameters in the 'lvmdev' line are non-empty + # because an empty positional parameter would result an invalid 'lvmdev' line + # which would cause invalid parameters are 'read' as input during "rear recover" + # cf. "Verifying ... 'lvm...' entries" in layout/save/default/950_verify_disklayout_file.sh +@@ -117,13 +112,24 @@ local lvs_exit_code + # so that this also checks that the variables do not contain blanks or more than one word + # because blanks (actually $IFS characters) are used as field separators in disklayout.conf + # which means the positional parameter values must be exactly one non-empty word. +- # Two separated simple 'test $vgrp && test $pdev' commands are used here because +- # 'test $vgrp -a $pdev' does not work when $vgrp is empty or only blanks +- # because '-a' has two different meanings: "EXPR1 -a EXPR2" and "-a FILE" (see "help test") +- # so that when $vgrp is empty 'test $vgrp -a $pdev' tests if file $pdev exists +- # which is usually true because $pdev is usually a partition device node (e.g. /dev/sda1) +- # so that when $vgrp is empty 'test $vgrp -a $pdev' would falsely succeed: +- test $vgrp && test $pdev || Error "LVM 'lvmdev' entry in $DISKLAYOUT_FILE where volume_group or device is empty or more than one word" ++ test $pdev || Error "Cannot make 'lvmdev' entry in disklayout.conf (PV device '$pdev' empty or more than one word)" ++ if ! test $vgrp ; then ++ # Valid $pdev but invalid $vgrp (empty or more than one word): ++ # When $vgrp is empty it means it is a PV that is not part of a VG so the PV exists but it is not used. ++ # PVs that are not part of a VG are documented as comment in disklayout.conf but they are not recreated ++ # because they were not used on the original system so there is no need to recreate them by "rear recover" ++ # (the user can manually recreate them later in his recreated system when needed) ++ # cf. https://github.com/rear/rear/issues/2596 ++ DebugPrint "Skipping PV $pdev that is not part of a valid VG (VG '$vgrp' empty or more than one word)" ++ echo "# Skipping PV $pdev that is not part of a valid VG (VG '$vgrp' empty or more than one word):" ++ contains_visible_char "$vgrp" || vgrp='' ++ echo "# lvmdev /dev/$vgrp $pdev $uuid $size" ++ # Continue with the next line in the output of "lvm pvdisplay -c" ++ continue ++ fi ++ # With the above example the output is: ++ # lvmdev /dev/system /dev/sda1 7wwpcO-KmNN-qsTE-7sp7-JBJS-vBdC-Zyt1W7 41940992 ++ echo "lvmdev /dev/$vgrp $pdev $uuid $size" + + done + # Check the exit code of "lvm pvdisplay -c" +@@ -161,8 +167,15 @@ local lvs_exit_code + # lvmgrp /dev/system 4096 5119 20967424 + echo "lvmgrp /dev/$vgrp $extentsize $nrextents $size" + +- # Check that the required positional parameters in the 'lvmgrp' line are non-empty +- # cf. the code above to "check that the required positional parameters in the 'lvmdev' line are non-empty": ++ # Check that the required positional parameters in the 'lvmgrp' line are non-empty. ++ # The tested variables are intentionally not quoted here, cf. the code above to ++ # "check that the required positional parameters in the 'lvmdev' line are non-empty". ++ # Two separated simple 'test $vgrp && test $extentsize' commands are used here because ++ # 'test $vgrp -a $extentsize' does not work when $vgrp is empty or only blanks ++ # because '-a' has two different meanings: "EXPR1 -a EXPR2" and "-a FILE" (see "help test") ++ # so with empty $vgrp it becomes 'test -a $extentsize' that tests if a file $extentsize exists ++ # which is unlikely to be true but it is not impossible that a file $extentsize exists ++ # so when $vgrp is empty (or blanks) 'test $vgrp -a $extentsize' might falsely succeed: + test $vgrp && test $extentsize || Error "LVM 'lvmgrp' entry in $DISKLAYOUT_FILE where volume_group or extentsize is empty or more than one word" + + done +@@ -305,7 +318,8 @@ local lvs_exit_code + fi + already_processed_lvs+=( "$vg/$lv" ) + # Check that the required positional parameters in the 'lvmvol' line are non-empty +- # cf. the code above to "check that the required positional parameters in the 'lvmdev' line are non-empty": ++ # cf. the code above to "check that the required positional parameters in the 'lvmdev' line are non-empty" ++ # and the code above to "check that the required positional parameters in the 'lvmgrp' line are non-empty": + test $vg && test $lv && test $size && test $layout || Error "LVM 'lvmvol' entry in $DISKLAYOUT_FILE where volume_group or name or size or layout is empty or more than one word" + fi + diff --git a/SOURCES/rear-bz2049091.patch b/SOURCES/rear-bz2049091.patch new file mode 100644 index 0000000..9f5e12d --- /dev/null +++ b/SOURCES/rear-bz2049091.patch @@ -0,0 +1,25 @@ +diff --git a/usr/share/rear/layout/save/default/335_remove_excluded_multipath_vgs.sh b/usr/share/rear/layout/save/default/335_remove_excluded_multipath_vgs.sh +index 040e9eec..e731c994 100644 +--- a/usr/share/rear/layout/save/default/335_remove_excluded_multipath_vgs.sh ++++ b/usr/share/rear/layout/save/default/335_remove_excluded_multipath_vgs.sh +@@ -19,9 +19,9 @@ while read lvmdev name mpdev junk ; do + # Remember, multipath devices from a volume group that is "excluded" should be 'commented out' + device=$(echo $mpdev | cut -c1-45) + while read LINE ; do +- # Now we need to comment all lines that contain "$devices" in the LAYOUT_FILE ++ # Now we need to comment all lines that contain "$device" in the LAYOUT_FILE + sed -i "s|^$LINE|\#$LINE|" "$LAYOUT_FILE" +- done < <(grep "$device" $LAYOUT_FILE | grep -v "^#") ++ done < <(grep " $device " $LAYOUT_FILE | grep -v "^#") + Log "Excluding multipath device $device" + done < <(grep "^#lvmdev" $LAYOUT_FILE) + +@@ -31,7 +31,7 @@ done < <(grep "^#lvmdev" $LAYOUT_FILE) + while read LINE ; do + # multipath /dev/mapper/360060e8007e2e3000030e2e300002065 /dev/sdae,/dev/sdat,/dev/sdbi,/dev/sdp + device=$(echo $LINE | awk '{print $2}' | cut -c1-45) +- num=$(grep "$device" $LAYOUT_FILE | grep -v "^#" | wc -l) ++ num=$(grep " $device " $LAYOUT_FILE | grep -v "^#" | wc -l) + if [ $num -lt 2 ] ; then + # If the $device is only seen once (in a uncommented line) then the multipath is not in use + sed -i "s|^$LINE|\#$LINE|" "$LAYOUT_FILE" diff --git a/SOURCES/rear-pr2675.patch b/SOURCES/rear-pr2675.patch new file mode 100644 index 0000000..7d11071 --- /dev/null +++ b/SOURCES/rear-pr2675.patch @@ -0,0 +1,60 @@ +diff --git a/usr/share/rear/lib/framework-functions.sh b/usr/share/rear/lib/framework-functions.sh +index 4878216b..e919bdbf 100644 +--- a/usr/share/rear/lib/framework-functions.sh ++++ b/usr/share/rear/lib/framework-functions.sh +@@ -121,7 +121,7 @@ function cleanup_build_area_and_end_program () { + sleep 2 + umount_mountpoint_lazy $BUILD_DIR/outputfs + fi +- remove_temporary_mountpoint '$BUILD_DIR/outputfs' || BugError "Directory $BUILD_DIR/outputfs not empty, can not remove" ++ remove_temporary_mountpoint "$BUILD_DIR/outputfs" || BugError "Directory $BUILD_DIR/outputfs not empty, can not remove" + rmdir $v $BUILD_DIR >&2 + fi + Log "End of program reached" +diff --git a/usr/share/rear/lib/global-functions.sh b/usr/share/rear/lib/global-functions.sh +index c1a11615..0f8f362d 100644 +--- a/usr/share/rear/lib/global-functions.sh ++++ b/usr/share/rear/lib/global-functions.sh +@@ -317,7 +317,20 @@ function url_path() { + + ### Returns true if one can upload files to the URL + function scheme_accepts_files() { +- local scheme=$1 ++ # Be safe against 'set -eu' which would exit 'rear' with "bash: $1: unbound variable" ++ # when scheme_accepts_files is called without an argument ++ # by bash parameter expansion with using an empty default value if $1 is unset or null. ++ # Bash parameter expansion with assigning a default value ${1:=} does not work ++ # (then it would still exit with "bash: $1: cannot assign in this way") ++ # but using a default value is practicable here because $1 is used only once ++ # cf. https://github.com/rear/rear/pull/2675#discussion_r705018956 ++ local scheme=${1:-} ++ # Return false if scheme is empty or blank (e.g. when OUTPUT_URL is unset or empty or blank) ++ # cf. https://github.com/rear/rear/issues/2676 ++ # and https://github.com/rear/rear/issues/2667#issuecomment-914447326 ++ # also return false if scheme is more than one word (so no quoted "$scheme" here) ++ # cf. https://github.com/rear/rear/pull/2675#discussion_r704401462 ++ test $scheme || return 1 + case $scheme in + (null|tape|obdr) + # tapes do not support uploading arbitrary files, one has to handle them +@@ -341,7 +354,10 @@ function scheme_accepts_files() { + ### Returning true does not imply that the URL is currently mounted at a filesystem and usable, + ### only that it can be mounted (use mount_url() first) + function scheme_supports_filesystem() { +- local scheme=$1 ++ # Be safe against 'set -eu' exit if scheme_supports_filesystem is called without argument ++ local scheme=${1:-} ++ # Return false if scheme is empty or blank or more than one word, cf. scheme_accepts_files() above ++ test $scheme || return 1 + case $scheme in + (null|tape|obdr|rsync|fish|ftp|ftps|hftp|http|https|sftp) + return 1 +@@ -560,7 +576,7 @@ function umount_url() { + + RemoveExitTask "perform_umount_url '$url' '$mountpoint' lazy" + +- remove_temporary_mountpoint '$mountpoint' && RemoveExitTask "remove_temporary_mountpoint '$mountpoint'" ++ remove_temporary_mountpoint "$mountpoint" && RemoveExitTask "remove_temporary_mountpoint '$mountpoint'" + return 0 + } + diff --git a/SOURCES/rear-sfdc02772301.patch b/SOURCES/rear-sfdc02772301.patch new file mode 100644 index 0000000..74456dd --- /dev/null +++ b/SOURCES/rear-sfdc02772301.patch @@ -0,0 +1,38 @@ +diff --git a/usr/share/rear/conf/default.conf b/usr/share/rear/conf/default.conf +index 9ada92c3..455aa3ce 100644 +--- a/usr/share/rear/conf/default.conf ++++ b/usr/share/rear/conf/default.conf +@@ -1813,7 +1813,7 @@ OBDR_BLOCKSIZE=2048 + # BACKUP=NBU stuff (Symantec/Veritas NetBackup) + ## + # +-COPY_AS_IS_NBU=( /usr/openv/bin/vnetd /usr/openv/bin/vopied /usr/openv/lib /usr/openv/netbackup /usr/openv/var/auth/[mn]*.txt ) ++COPY_AS_IS_NBU=( /usr/openv/bin/vnetd /usr/openv/bin/vopied /usr/openv/lib /usr/openv/netbackup /usr/openv/var/auth/[mn]*.txt /opt/VRTSpbx /etc/vx/VxICS /etc/vx/vrtslog.conf ) + COPY_AS_IS_EXCLUDE_NBU=( /usr/openv/netbackup/logs "/usr/openv/netbackup/bin/bpjava*" /usr/openv/netbackup/bin/xbp /usr/openv/netbackup/bin/private /usr/openv/lib/java /usr/openv/lib/shared/vddk /usr/openv/netbackup/baremetal ) + # See https://github.com/rear/rear/issues/2105 why /usr/openv/netbackup/sec/at/lib/ is needed: + NBU_LD_LIBRARY_PATH="/usr/openv/lib:/usr/openv/netbackup/sec/at/lib/" +diff --git a/usr/share/rear/rescue/NBU/default/450_prepare_netbackup.sh b/usr/share/rear/rescue/NBU/default/450_prepare_netbackup.sh +index cd48b8d9..ae5a3ccc 100644 +--- a/usr/share/rear/rescue/NBU/default/450_prepare_netbackup.sh ++++ b/usr/share/rear/rescue/NBU/default/450_prepare_netbackup.sh +@@ -7,6 +7,12 @@ + + [[ $NBU_version -lt 7 ]] && return # NBU is using xinetd when version <7.x + ++if [ -e "/etc/init.d/vxpbx_exchanged" ]; then ++ cp $v /etc/init.d/vxpbx_exchanged $ROOTFS_DIR/etc/scripts/system-setup.d/vxpbx_exchanged.real ++ chmod $v +x $ROOTFS_DIR/etc/scripts/system-setup.d/vxpbx_exchanged.real ++ echo "( /etc/scripts/system-setup.d/vxpbx_exchanged.real start )" > $ROOTFS_DIR/etc/scripts/system-setup.d/89-vxpbx_exchanged.sh ++fi ++ + if [ -e "/etc/init.d/netbackup" ]; then + cp $v /etc/init.d/netbackup $ROOTFS_DIR/etc/scripts/system-setup.d/netbackup.real + chmod $v +x $ROOTFS_DIR/etc/scripts/system-setup.d/netbackup.real +diff --git a/usr/share/rear/skel/NBU/usr/openv/tmp/.gitignore b/usr/share/rear/skel/NBU/usr/openv/tmp/.gitignore +new file mode 100644 +index 00000000..d6b7ef32 +--- /dev/null ++++ b/usr/share/rear/skel/NBU/usr/openv/tmp/.gitignore +@@ -0,0 +1,2 @@ ++* ++!.gitignore diff --git a/SOURCES/rear-tmpdir.patch b/SOURCES/rear-tmpdir.patch new file mode 100644 index 0000000..ec5ba71 --- /dev/null +++ b/SOURCES/rear-tmpdir.patch @@ -0,0 +1,37 @@ +diff --git a/usr/share/rear/conf/default.conf b/usr/share/rear/conf/default.conf +index 9ada92c3..3bdb5497 100644 +--- a/usr/share/rear/conf/default.conf ++++ b/usr/share/rear/conf/default.conf +@@ -57,10 +57,16 @@ + # + # where /prefix/for/rear/working/directory must already exist. + # This is useful for example when there is not sufficient free space +-# in /tmp or $TMPDIR for the ISO image or even the backup archive. +-# TMPDIR cannot be set to a default value here, otherwise /usr/sbin/rear ++# in /var/tmp or $TMPDIR for the ISO image or even the backup archive. ++# TMPDIR cannot be set to a default value here unconditionally but only ++# if it is not set before calling the program, otherwise /usr/sbin/rear + # would not work in compliance with the Linux/Unix standards regarding TMPDIR + # see https://github.com/rear/rear/issues/968 ++# The default is /var/tmp instead of the more usual /tmp (the system default), ++# because /tmp is not intended for such large amounts of data that ReaR usually ++# produces when creating the image (see file-hierarchy(7)). In particular, ++# /tmp can be a tmpfs, and thus restricted by the available RAM/swap. ++export TMPDIR="${TMPDIR-/var/tmp}" + + ## + # ROOT_HOME_DIR +diff --git a/usr/share/rear/rescue/GNU/Linux/600_unset_TMPDIR_in_rescue_conf.sh b/usr/share/rear/rescue/GNU/Linux/600_unset_TMPDIR_in_rescue_conf.sh +deleted file mode 100644 +index 84d0cabb..00000000 +--- a/usr/share/rear/rescue/GNU/Linux/600_unset_TMPDIR_in_rescue_conf.sh ++++ /dev/null +@@ -1,8 +0,0 @@ +-cat - <> "$ROOTFS_DIR/etc/rear/rescue.conf" +-# TMPDIR variable may be defined in local.conf file as prefix dir for mktemp command +-# e.g. by defining TMPDIR=/var we would get our BUILD_DIR=/var/tmp/rear.XXXXXXXXXXXX +-# However, in rescue we want our BUILD_DIR=/tmp/rear.XXXXXXX as we are not sure that +-# the user defined TMPDIR would exist in our rescue image +-# by 'unset TMPDIR' we achieve above goal (as rescue.conf is read after local.conf)! +-unset TMPDIR +-EOF diff --git a/SOURCES/rear.cron b/SOURCES/rear.cron new file mode 100644 index 0000000..b4dbce1 --- /dev/null +++ b/SOURCES/rear.cron @@ -0,0 +1,4 @@ +# cronjob for ReaR +# periodically check if disk layout has changed and update +# the rescue image +30 1 * * * root test -f /var/lib/rear/layout/disklayout.conf && /usr/sbin/rear checklayout || /usr/sbin/rear mkrescue diff --git a/SOURCES/rear.service b/SOURCES/rear.service new file mode 100644 index 0000000..e1e359a --- /dev/null +++ b/SOURCES/rear.service @@ -0,0 +1,6 @@ +[Unit] +Description=Update ReaR rescue image + +[Service] +Type=oneshot +ExecStart=/usr/sbin/rear checklayout || /usr/sbin/rear mkrescue diff --git a/SOURCES/rear.timer b/SOURCES/rear.timer new file mode 100644 index 0000000..6012724 --- /dev/null +++ b/SOURCES/rear.timer @@ -0,0 +1,10 @@ +[Unit] +Description=Update ReaR rescue image + +[Timer] +OnCalendar=daily +RandomizedDelaySec=1h +Persistent=true + +[Install] +WantedBy=timers.target diff --git a/SPECS/rear.spec b/SPECS/rear.spec new file mode 100644 index 0000000..b7564a5 --- /dev/null +++ b/SPECS/rear.spec @@ -0,0 +1,237 @@ +# this is purely a shell script, so no debug packages +%global debug_package %{nil} + +Name: rear +Version: 2.6 +Release: 11%{?dist} +Summary: Relax-and-Recover is a Linux disaster recovery and system migration tool +URL: http://relax-and-recover.org/ +License: GPLv3 + +Source0: https://github.com/rear/rear/archive/%{version}.tar.gz#/rear-%{version}.tar.gz +# Add cronjob and systemd timer as documentation +Source1: rear.cron +Source2: rear.service +Source3: rear.timer +# Skip buildin modules, RHBZ#1831311 +Patch0: 0001-skip-kernel-buildin-modules.patch +Patch4: rear-bz1492177-warning.patch +Patch29: rear-bz1832394.patch +Patch30: rear-sfdc02772301.patch +Patch31: rear-bz1945869.patch +Patch32: rear-bz1958247.patch +Patch33: rear-bz1930662.patch +Patch34: rear-tmpdir.patch +Patch35: rear-bz1983013.patch +Patch36: rear-bz1993296.patch +Patch37: rear-bz1747468.patch +Patch38: rear-bz2049091.patch +Patch39: rear-pr2675.patch +Patch40: rear-bz2048454.patch +Patch41: rear-bz2035939.patch + +# rear contains only bash scripts plus documentation so that on first glance it could be "BuildArch: noarch" +# but actually it is not "noarch" because it only works on those architectures that are explicitly supported. +# Of course the rear bash scripts can be installed on any architecture just as any binaries can be installed on any architecture. +# But the meaning of architecture dependent packages should be on what architectures they will work. +# Therefore only those architectures that are actually supported are explicitly listed. +# This avoids that rear can be "just installed" on architectures that are actually not supported (e.g. ARM): +ExclusiveArch: %ix86 x86_64 ppc ppc64 ppc64le ia64 s390x +# Furthermore for some architectures it requires architecture dependent packages (like syslinux for x86 and x86_64) +# so that rear must be architecture dependent because ifarch conditions never match in case of "BuildArch: noarch" +# see the GitHub issue https://github.com/rear/rear/issues/629 +%ifarch %ix86 x86_64 +Requires: syslinux +%endif +%ifarch ppc ppc64 ppc64le +# Called by grub2-install (except on PowerNV) +Requires: /usr/sbin/ofpathname +# Needed to make PowerVM LPARs bootable +Requires: /usr/sbin/bootlist +%endif +# In the end this should tell the user that rear is known to work only on ix86 x86_64 ppc ppc64 ppc64le ia64 +# and on ix86 x86_64 syslinux is explicitly required to make the bootable ISO image +# (in addition to the default installed bootloader grub2) while on ppc ppc64 the +# default installed bootloader yaboot is also useed to make the bootable ISO image. + +# Required for HTML user guide +BuildRequires: make +BuildRequires: asciidoctor + +### Mandatory dependencies: +Requires: binutils +Requires: ethtool +Requires: gzip +Requires: iputils +Requires: parted +Requires: tar +Requires: openssl +Requires: gawk +Requires: attr +Requires: bc +Requires: iproute +# No ISO image support on s390x (may change when we add support for LPARs) +%ifnarch s390x +Requires: xorriso +%endif +Requires: file +Requires: dhcp-client +%if 0%{?rhel} +Requires: util-linux +%endif + +%description +Relax-and-Recover is the leading Open Source disaster recovery and system +migration solution. It comprises of a modular +frame-work and ready-to-go workflows for many common situations to produce +a bootable image and restore from backup using this image. As a benefit, +it allows to restore to different hardware and can therefore be used as +a migration tool as well. + +Currently Relax-and-Recover supports various boot media (incl. ISO, PXE, +OBDR tape, USB or eSATA storage), a variety of network protocols (incl. +sftp, ftp, http, nfs, cifs) as well as a multitude of backup strategies +(incl. IBM TSM, MircroFocus Data Protector, Symantec NetBackup, EMC NetWorker, +Bacula, Bareos, BORG, Duplicity, rsync). + +Relax-and-Recover was designed to be easy to set up, requires no maintenance +and is there to assist when disaster strikes. Its setup-and-forget nature +removes any excuse for not having a disaster recovery solution implemented. + +Professional services and support are available. + +#-- PREP, BUILD & INSTALL -----------------------------------------------------# +%prep +%autosetup -p1 + +### Add a specific os.conf so we do not depend on LSB dependencies +%{?fedora:echo -e "OS_VENDOR=Fedora\nOS_VERSION=%{?fedora}" >etc/rear/os.conf} +%{?rhel:echo -e "OS_VENDOR=RedHatEnterpriseServer\nOS_VERSION=%{?rhel}" >etc/rear/os.conf} + +# Change /lib to /usr/lib for COPY_AS_IS +sed -E -e "s:([\"' ])/lib:\1/usr/lib:g" \ + -i usr/share/rear/prep/GNU/Linux/*include*.sh + +# Same for Linux.conf +sed -e 's:/lib/:/usr/lib/:g' \ + -e 's:/lib\*/:/usr/lib\*/:g' \ + -e 's:/usr/usr/lib:/usr/lib:g' \ + -i 'usr/share/rear/conf/GNU/Linux.conf' + +%build +# build HTML user guide +# asciidoc writes a timestamp to files it produces, based on the last +# modified date of the source file, but is sensitive to the timezone. +# This makes the results differ according to the timezone of the build machine +# and spurious changes will be seen. +# Set the timezone to UTC as a workaround. +# https://wiki.debian.org/ReproducibleBuilds/TimestampsInDocumentationGeneratedByAsciidoc +TZ=UTC make doc + +%install +%{make_install} +install -p -d %{buildroot}%{_docdir}/%{name}/ +install -m 0644 %{SOURCE1} %{buildroot}%{_docdir}/%{name}/ +install -m 0644 %{SOURCE2} %{buildroot}%{_docdir}/%{name}/ +install -m 0644 %{SOURCE3} %{buildroot}%{_docdir}/%{name}/ + +#-- FILES ---------------------------------------------------------------------# +%files +%doc MAINTAINERS COPYING README.adoc doc/*.txt doc/user-guide/*.html +%doc %{_mandir}/man8/rear.8* +%doc %{_docdir}/%{name}/rear.* +%config(noreplace) %{_sysconfdir}/rear/ +%{_datadir}/rear/ +%{_sharedstatedir}/rear/ +%{_sbindir}/rear + +#-- CHANGELOG -----------------------------------------------------------------# +%changelog +* Sun Feb 27 2022 Pavel Cahyna - 2.6-11 +- Apply PR2675 to fix leftover temp dir bug (introduced in backported PR2625) +- Apply PR2603 to ignore unused PV devices +- Apply upstream PR2750 to avoid exclusion of wanted multipath devices +- Remove unneeded xorriso dep on s390x (no ISO image support there) +- Apply upstream PR2736 to add the EXCLUDE_{IP_ADDRESSES,NETWORK_INTERFACES} + options +- Add patch for better handling of thin pools and other LV types not supported + by vgcfgrestore + +* Mon Aug 16 2021 Pavel Cahyna - 2.6-10 +- Sync spec changes and downstream patches from RHEL 8 rear-2.6-2 + - Fix multipath performance regression in 2.6, introduced by upstream PR #2299. + Resolves: rhbz1993296 + - On POWER add bootlist & ofpathname to the list of required programs + conditionally (bootlist only if running under PowerVM, ofpathname + always except on PowerNV) - upstream PR2665, add them to package + dependencies + Resolves: rhbz1983013 + - Backport PR2608: + Fix setting boot path in case of UEFI partition (ESP) on MD RAID + Resolves: rhbz1945869 + - Backport PR2625 + Prevents accidental backup removal in case of errors + Resolves: rhbz1958247 + - Fix rsync error and option handling + Resolves: rhbz1930662 + +* Wed Aug 11 2021 Pavel Cahyna - 2.6-9 +- Put TMPDIR on /var/tmp by default, otherwise it may lack space + RHBZ #1988420, upstream PR2664 + +* Tue Aug 10 2021 Mohan Boddu - 2.6-8 +- Rebuilt for IMA sigs, glibc 2.34, aarch64 flags + Related: rhbz#1991688 + +* Wed Jun 30 2021 Pavel Cahyna - 2.6-7 +- Sync spec changes and downstream patches from RHEL 8 + - Require xorriso instead of genisoimage + - Add S/390 support and forgotten dependency on the file utility + - Backport upstream code related to LUKS2 support + - Modify the cron command to avoid an e-mail with error message after + ReaR is installed but not properly configured when the cron command + is triggered for the first time + - Changes for NetBackup (NBU) support, upstream PR2544 +- Add dependency on dhcp-client, RHBZ #1926451 + +* Fri Apr 16 2021 Mohan Boddu - 2.6-6 +- Rebuilt for RHEL 9 BETA on Apr 15th 2021. Related: rhbz#1947937 + +* Fri Feb 26 2021 Christopher Engelhard - 2.6-5 +- Change /lib to /usr/lib in scripts to fix RHBZ #1931112 + +* Wed Jan 27 2021 Fedora Release Engineering - 2.6-4 +- Rebuilt for https://fedoraproject.org/wiki/Fedora_34_Mass_Rebuild + +* Wed Sep 23 2020 Christopher Engelhard - 2.6-3 +- Stop auto-creating a cronjob, but ship example cronjob/ + systemd timer units in docdir instead (upstream issue #1829) +- Build & ship HTML user guide +- Remove %pre scriptlet, as it was introduced only to fix a + specific upgrade issue with v1.15 in 2014 + +* Tue Sep 22 2020 Christopher Engelhard - 2.6-2 +- Backport upstream PR#2469 to fix RHBZ #1831311 + +* Tue Sep 22 2020 Christopher Engelhard - 2.6-1 +- Update to 2.6 +- Streamline & clean up spec file + +* Wed Jul 29 2020 Fedora Release Engineering - 2.4-6 +- Rebuilt for https://fedoraproject.org/wiki/Fedora_33_Mass_Rebuild + +* Thu Jan 30 2020 Fedora Release Engineering - 2.4-5 +- Rebuilt for https://fedoraproject.org/wiki/Fedora_32_Mass_Rebuild + +* Fri Jul 26 2019 Fedora Release Engineering - 2.4-4 +- Rebuilt for https://fedoraproject.org/wiki/Fedora_31_Mass_Rebuild + +* Sat Feb 02 2019 Fedora Release Engineering - 2.4-3 +- Rebuilt for https://fedoraproject.org/wiki/Fedora_30_Mass_Rebuild + +* Sat Jul 14 2018 Fedora Release Engineering - 2.4-2 +- Rebuilt for https://fedoraproject.org/wiki/Fedora_29_Mass_Rebuild + +* Thu Jul 30 2015 Johannes Meixner +- For a changelog see the rear-release-notes.txt file. +