Merge pull request #78 from Hurricos/lvm

[WIP] feat: lvm: Add lvm-based backup functionality
This commit is contained in:
L3D 2023-08-10 11:42:06 +02:00 committed by GitHub
commit 7ec0667ac6
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
2 changed files with 98 additions and 9 deletions

View file

@ -1,8 +1,15 @@
[Unit] [Unit]
Description=Backup {{ item.name }} using restic Description=Backup {{ item.name }} using restic
{% if item.lvm is defined %}
Conflicts=fstrim.service
After=fstrim.timer
{% endif %}
[Service] [Service]
Type=oneshot Type=oneshot
{% if item.lvm is defined %}
PrivateMounts=on
{% endif %}
ExecStart={{ restic_script_dir }}/backup-{{ item.name }}.sh ExecStart={{ restic_script_dir }}/backup-{{ item.name }}.sh
TimeoutStartSec=0 TimeoutStartSec=0
Environment="CRON=true" Environment="CRON=true"

View file

@ -67,6 +67,77 @@ export B2_ACCOUNT_KEY={{ restic_repos[item.repo].b2_account_key }}
BACKUP_SOURCE={{ item.src }} BACKUP_SOURCE={{ item.src }}
{% endif %} {% endif %}
{% if item.lvm is defined %}
# Set up functions for LVM.
function mount_opt_map {
mount_type="$1"
case "$mount_type" in
xfs)
echo "noatime,nouuid"
;;
ext4)
echo "noatime"
;;
*)
echo "noatime"
esac
}
function prepare_vol {
local path="$1"
[ -d "$path" ] || path="$(dirname "$path")"
if [ "$path" == '/' ] ; then
mkdir -p /rootfs;
newpath='/rootfs';
else
newpath="$path";
fi
{
local source="$(findmnt -J -T ${path} | jq -r '.filesystems[0].source')"
local target="$(findmnt -J -T ${path} | jq -r '.filesystems[0].target')"
subdir=${path##$target}
echo "Creating snapshot ..."
lvcreate -y -L "${size:-10G}" -s -n "${source}_snap" "${source}"
local tmpdir="$(mktemp -d)"
local fs="$(lsblk -J --fs "$source" | jq -r '.blockdevices[0]|.fstype')"
echo "Identified fstype: $fs; using opts $(mount_opt_map "$fs") ..."
mount -t "$fs" \
-o "$(mount_opt_map "$fs")" \
--make-private \
"${source}_snap" "${tmpdir}"
mount --bind --make-private "${tmpdir}/${subdir}" "${newpath}"
}
}
function cleanup_vol {
local path="$1"
[ -d "$path" ] || path="$(dirname "$path")"
if [ "$path" == '/' ] ; then
newpath='/rootfs';
else
newpath="$path";
fi
{
local source="$(findmnt -v -J -T "${newpath}" | jq -r '.filesystems[]|.source' | grep '_snap$')"
if ! grep -q '_snap$' <<< $source; then
echo "Snapshot for ${path} could not be found (found: ${source}). Exiting!" && return 1;
fi
echo "Cleaning up mount ..."
umount "${newpath}"
echo "Cleaning up snapshot ..."
umount "${source}"
lvremove -y "${source}";
}
}
{% endif %}
set -uxo pipefail set -uxo pipefail
{# {#
@ -155,25 +226,36 @@ fi
{% if item.exclude is defined %}{{ exclude(item.exclude) }}{% endif %} \ {% if item.exclude is defined %}{{ exclude(item.exclude) }}{% endif %} \
$@ \ $@ \
{% else %} {% else %}
{{ restic_install_path }}/restic backup $BACKUP_SOURCE $MODE_TAG \ {
{% if item.lvm is defined %}prepare_vol $BACKUP_SOURCE &&{% endif %}
{{ restic_install_path }}/restic backup {% if item.lvm is defined and item.src == '/' %}/rootfs{% endif %}$BACKUP_SOURCE $MODE_TAG \
{{ tags(item.tags) }} \ {{ tags(item.tags) }} \
{% if item.exclude is defined %}{{ exclude(item.exclude) }}{% endif %} \ {% if item.exclude is defined %}{{ exclude(item.exclude) }}{% endif %} \
$@ \ $@
} \
{% endif %} {{ backup_output_log }} {% endif %} {{ backup_output_log }}
if [[ $? -eq 0 ]]
then case $? in
echo "$(date -u '+%Y-%m-%d %H:%M:%S') OK" {{ backup_result_log }} 0)
else echo "$(date -u '+%Y-%m-%d %H:%M:%S') OK" {{ backup_result_log }}
echo "$(date -u '+%Y-%m-%d %H:%M:%S') ERROR" {{ backup_result_log }} ;;
3)
echo "$(date -u '+%Y-%m-%d %H:%M:%S') WARNING" {{ backup_result_log }}
;;
*)
echo "$(date -u '+%Y-%m-%d %H:%M:%S') ERROR" {{ backup_result_log }}
{% if item.mail_on_error is defined and item.mail_on_error == true %} {% if item.mail_on_error is defined and item.mail_on_error == true %}
mail -s "restic backup failed on {{ ansible_hostname }}" {{ item.mail_address }} <<< "Something went wrong while running restic backup script running at {{ ansible_hostname }} at $(date -u '+%Y-%m-%d %H:%M:%S'). mail -s "restic backup failed on {{ ansible_hostname }}" {{ item.mail_address }} <<< "Something went wrong while running restic backup script running at {{ ansible_hostname }} at $(date -u '+%Y-%m-%d %H:%M:%S').
{%- if item.src is defined -%} {%- if item.src is defined -%}
{{ ' ' }}We tried to backup '{{ item.src }}'. {{ ' ' }}We tried to backup '{{ item.src }}'.
{%- endif -%} {%- endif -%}
{{ ' ' }}Please repair the restic-{{ item.name | replace(' ', '') }} job." {{ ' ' }}Please repair the restic-{{ item.name | replace(' ', '') }} job."
{% endif %} {% endif %}
fi esac
{% if item.lvm is defined %}
cleanup_vol $BACKUP_SOURCE
{% endif %}
{# {#