making snap-and-mount more maintainable
This commit is contained in:
parent
86674c4821
commit
8561a16219
3 changed files with 46 additions and 179 deletions
|
@ -18,153 +18,7 @@ let
|
||||||
];
|
];
|
||||||
|
|
||||||
text = ''
|
text = ''
|
||||||
# Check if running as root
|
${builtins.readFile ./snap-and-mount.sh} "${cfg.mountPath}" "${cfg.zfsDataset}" "${cfg.snapshotName}"
|
||||||
if [ "$(id -u)" -ne 0 ]; then
|
|
||||||
echo "Error: This script must be run as root."
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
BACKUP_DIRECTORY="${cfg.mountPath}"
|
|
||||||
ZFS_DATASET="${cfg.zfsDataset}"
|
|
||||||
SNAPSHOT_NAME="${cfg.snapshotName}"
|
|
||||||
|
|
||||||
# functions sourced from: https://github.com/Jip-Hop/zfs-backup-snapshots
|
|
||||||
# some enhancements made to the original code to adhere to best practices
|
|
||||||
# mounts all zfs filesystems under $ZFS_DATASET
|
|
||||||
function mount_dataset() {
|
|
||||||
# ensure BACKUP_DIRECTORY exists
|
|
||||||
mkdir -p "$BACKUP_DIRECTORY"
|
|
||||||
# get list of all zfs filesystems under $ZFS_DATASET
|
|
||||||
# exclude if mountpoint "legacy" and "none" mountpoint
|
|
||||||
# order by shallowest mountpoint first (determined by number of slashes)
|
|
||||||
mapfile -t fs < <(zfs list "$ZFS_DATASET" -r -H -o name,mountpoint | grep -E "(legacy)$|(none)$" -v | awk '{print gsub("/","/", $2), $1}' | sort -n | cut -d' ' -f2-)
|
|
||||||
|
|
||||||
for fs in "''${fs[@]}"; do
|
|
||||||
mount_latest_snap "''${fs}" "$BACKUP_DIRECTORY"
|
|
||||||
done
|
|
||||||
return 0
|
|
||||||
}
|
|
||||||
|
|
||||||
# umounts and cleans up the backup directory
|
|
||||||
# usage: zfs_backup_cleanup BACKUP_DIRECTORY
|
|
||||||
function zfs_backup_cleanup() {
|
|
||||||
# get all filesystems mounted within the backup directory
|
|
||||||
mapfile -t fs < <(tac /etc/mtab | cut -d " " -f 2 | grep "''${1}")
|
|
||||||
|
|
||||||
# umount said filesystems
|
|
||||||
for i in "''${fs[@]}"; do
|
|
||||||
echo "Unmounting $i"
|
|
||||||
umount "$i"
|
|
||||||
done
|
|
||||||
|
|
||||||
# delete empty directories from within the backup directory
|
|
||||||
find "''${1}" -type d -empty -delete
|
|
||||||
}
|
|
||||||
|
|
||||||
# gets the name of the newest snapshot given a zfs filesystem
|
|
||||||
# usage: get_latest_snap filesystem
|
|
||||||
function zfs_latest_snap() {
|
|
||||||
snapshot=$(zfs list -H -t snapshot -o name -S creation -d1 "''${1}" | head -1 | cut -d '@' -f 2)
|
|
||||||
if [[ -z $snapshot ]]; then
|
|
||||||
# if there's no snapshot then let's ignore it
|
|
||||||
echo "No snapshot exists for ''${1}, it will not be backed up."
|
|
||||||
return 1
|
|
||||||
fi
|
|
||||||
echo "$snapshot"
|
|
||||||
}
|
|
||||||
|
|
||||||
# gets the path of a snapshot given a zfs filesystem and a snapshot name
|
|
||||||
# usage zfs_snapshot_mountpoint filesystem snapshot
|
|
||||||
function zfs_snapshot_mountpoint() {
|
|
||||||
# get mountpoint for filesystem
|
|
||||||
mountpoint=$(zfs list -H -o mountpoint "''${1}")
|
|
||||||
|
|
||||||
# exit if filesystem doesn't exist
|
|
||||||
if [[ $? == 1 ]]; then
|
|
||||||
return 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
# build out path
|
|
||||||
path="''${mountpoint}/.zfs/snapshot/''${2}"
|
|
||||||
|
|
||||||
# check to make sure path exists
|
|
||||||
if stat "''${path}" &> /dev/null; then
|
|
||||||
echo "''${path}"
|
|
||||||
return 0
|
|
||||||
else
|
|
||||||
return 1
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
|
|
||||||
# mounts latest snapshot in directory
|
|
||||||
# usage: mount_latest_snap filesystem BACKUP_DIRECTORY
|
|
||||||
function mount_latest_snap() {
|
|
||||||
local mount_point="''${2}"
|
|
||||||
local filesystem="''${1}"
|
|
||||||
|
|
||||||
# get name of latest snapshot
|
|
||||||
snapshot=$(zfs_latest_snap "''${filesystem}")
|
|
||||||
|
|
||||||
# if there's no snapshot then let's ignore it
|
|
||||||
if [[ $? == 1 ]]; then
|
|
||||||
echo "No snapshot exists for ''${filesystem}, it will not be backed up."
|
|
||||||
return 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
sourcepath=$(zfs_snapshot_mountpoint "''${filesystem}" "''${snapshot}")
|
|
||||||
# if the filesystem is not mounted/path doesn't exist then let's ignore as well
|
|
||||||
if [[ $? == 1 ]]; then
|
|
||||||
echo "Cannot find snapshot ''${snapshot} for ''${filesystem}, perhaps it's not mounted? Anyways, it will not be backed up."
|
|
||||||
return 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
# mountpath may be inside a previously mounted snapshot
|
|
||||||
mountpath="$mount_point/''${filesystem}"
|
|
||||||
|
|
||||||
# mount to backup directory using a bind filesystem
|
|
||||||
mkdir -p "''${mountpath}"
|
|
||||||
echo "mount ''${sourcepath} => ''${mountpath}"
|
|
||||||
mount --bind --read-only "''${sourcepath}" "''${mountpath}"
|
|
||||||
return 0
|
|
||||||
}
|
|
||||||
|
|
||||||
# Unmount and cleanup if necessary
|
|
||||||
zfs_backup_cleanup "$BACKUP_DIRECTORY"
|
|
||||||
|
|
||||||
# Check if snapshot exists
|
|
||||||
echo "Previous snapshot:"
|
|
||||||
zfs list -t snapshot | grep "$ZFS_DATASET@$SNAPSHOT_NAME" || true
|
|
||||||
|
|
||||||
# Attempt to destroy existing snapshot
|
|
||||||
echo "Attempting to destroy existing snapshot..."
|
|
||||||
if zfs list -t snapshot | grep -q "$ZFS_DATASET@$SNAPSHOT_NAME"; then
|
|
||||||
if zfs destroy -r "$ZFS_DATASET@$SNAPSHOT_NAME"; then
|
|
||||||
echo "Successfully destroyed old snapshot"
|
|
||||||
else
|
|
||||||
echo "Failed to destroy existing snapshot"
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
else
|
|
||||||
echo "No existing snapshot found"
|
|
||||||
fi
|
|
||||||
|
|
||||||
# Create new snapshot
|
|
||||||
if ! zfs snapshot -r "$ZFS_DATASET@$SNAPSHOT_NAME"; then
|
|
||||||
echo "Failed to create snapshot"
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
echo "New snapshot created:"
|
|
||||||
zfs list -t snapshot | grep "$ZFS_DATASET@$SNAPSHOT_NAME"
|
|
||||||
|
|
||||||
# Mount the snapshot
|
|
||||||
if ! mount_dataset; then
|
|
||||||
echo "Failed to mount snapshot"
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
echo "Successfully created and mounted snapshot at $BACKUP_DIRECTORY"
|
|
||||||
mount | grep "$BACKUP_DIRECTORY"
|
|
||||||
'';
|
'';
|
||||||
};
|
};
|
||||||
in
|
in
|
||||||
|
|
15
nixos/modules/nixos/services/zfs-nightly-snap/snap-and-mount-standalone.sh
Executable file
15
nixos/modules/nixos/services/zfs-nightly-snap/snap-and-mount-standalone.sh
Executable file
|
@ -0,0 +1,15 @@
|
||||||
|
#!/usr/bin/env nix-shell
|
||||||
|
#!nix-shell -I nixpkgs=/etc/nix/inputs/nixpkgs -i bash -p busybox zfs
|
||||||
|
# shellcheck disable=SC1008
|
||||||
|
|
||||||
|
set -e # Exit on error
|
||||||
|
|
||||||
|
BACKUP_DIRECTORY="/mnt/restic_nightly_backup"
|
||||||
|
ZFS_DATASET="nahar/containers/volumes"
|
||||||
|
SNAPSHOT_NAME="restic_nightly_snap"
|
||||||
|
|
||||||
|
# Execute the main script with our parameters
|
||||||
|
./snap-and-mount.sh \
|
||||||
|
"$BACKUP_DIRECTORY" \
|
||||||
|
"$ZFS_DATASET" \
|
||||||
|
"$SNAPSHOT_NAME"
|
62
nixos/modules/nixos/services/zfs-nightly-snap/snap-and-mount.sh
Executable file → Normal file
62
nixos/modules/nixos/services/zfs-nightly-snap/snap-and-mount.sh
Executable file → Normal file
|
@ -1,14 +1,16 @@
|
||||||
#!/usr/bin/env nix-shell
|
|
||||||
#!nix-shell -I nixpkgs=/etc/nix/inputs/nixpkgs -i bash -p busybox zfs
|
|
||||||
# shellcheck disable=SC1008
|
# shellcheck disable=SC1008
|
||||||
|
|
||||||
set -e # Exit on error
|
# Command line arguments
|
||||||
|
BACKUP_DIRECTORY="$1"
|
||||||
BACKUP_DIRECTORY="/mnt/restic_nightly_backup"
|
ZFS_DATASET="$2"
|
||||||
ZFS_DATASET="nahar/containers/volumes"
|
SNAPSHOT_NAME="$3"
|
||||||
SNAPSHOT_NAME="restic_nightly_snap"
|
|
||||||
|
|
||||||
|
if [ -z "$BACKUP_DIRECTORY" ] || [ -z "$ZFS_DATASET" ] || [ -z "$SNAPSHOT_NAME" ]; then
|
||||||
|
echo "Usage: $0 <backup_directory> <zfs_dataset> <snapshot_name>"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
# Check if running as root
|
# Check if running as root
|
||||||
|
|
||||||
if [ "$(id -u)" -ne 0 ]; then
|
if [ "$(id -u)" -ne 0 ]; then
|
||||||
echo "Error: This script must be run as root."
|
echo "Error: This script must be run as root."
|
||||||
exit 1
|
exit 1
|
||||||
|
@ -19,7 +21,7 @@ fi
|
||||||
# mounts all zfs filesystems under $ZFS_DATASET
|
# mounts all zfs filesystems under $ZFS_DATASET
|
||||||
function mount_dataset() {
|
function mount_dataset() {
|
||||||
# ensure BACKUP_DIRECTORY exists
|
# ensure BACKUP_DIRECTORY exists
|
||||||
mkdir -p $BACKUP_DIRECTORY
|
mkdir -p "$BACKUP_DIRECTORY"
|
||||||
# get list of all zfs filesystems under $ZFS_DATASET
|
# get list of all zfs filesystems under $ZFS_DATASET
|
||||||
# exclude if mountpoint "legacy" and "none" mountpoint
|
# exclude if mountpoint "legacy" and "none" mountpoint
|
||||||
# order by shallowest mountpoint first (determined by number of slashes)
|
# order by shallowest mountpoint first (determined by number of slashes)
|
||||||
|
@ -44,7 +46,7 @@ function zfs_backup_cleanup() {
|
||||||
done
|
done
|
||||||
|
|
||||||
# delete empty directories from within the backup directory
|
# delete empty directories from within the backup directory
|
||||||
find "${1}" -type d -empty -delete
|
find "${1}" -type d -empty -delete 2>/dev/null || true
|
||||||
}
|
}
|
||||||
|
|
||||||
# gets the name of the newest snapshot given a zfs filesystem
|
# gets the name of the newest snapshot given a zfs filesystem
|
||||||
|
@ -62,24 +64,24 @@ function zfs_latest_snap() {
|
||||||
# gets the path of a snapshot given a zfs filesystem and a snapshot name
|
# gets the path of a snapshot given a zfs filesystem and a snapshot name
|
||||||
# usage zfs_snapshot_mountpoint filesystem snapshot
|
# usage zfs_snapshot_mountpoint filesystem snapshot
|
||||||
function zfs_snapshot_mountpoint() {
|
function zfs_snapshot_mountpoint() {
|
||||||
# get mountpoint for filesystem
|
# get mountpoint for filesystem
|
||||||
mountpoint=$(zfs list -H -o mountpoint "${1}")
|
mountpoint=$(zfs list -H -o mountpoint "${1}")
|
||||||
|
|
||||||
# exit if filesystem doesn't exist
|
# exit if filesystem doesn't exist
|
||||||
if [[ $? == 1 ]]; then
|
if [[ $? == 1 ]]; then
|
||||||
return 1
|
return 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# build out path
|
# build out path
|
||||||
path="${mountpoint}/.zfs/snapshot/${2}"
|
path="${mountpoint}/.zfs/snapshot/${2}"
|
||||||
|
|
||||||
# check to make sure path exists
|
# check to make sure path exists
|
||||||
if stat "${path}" &> /dev/null; then
|
if stat "${path}" &>/dev/null; then
|
||||||
echo "${path}"
|
echo "${path}"
|
||||||
return 0
|
return 0
|
||||||
else
|
else
|
||||||
return 1
|
return 1
|
||||||
fi
|
fi
|
||||||
}
|
}
|
||||||
|
|
||||||
# mounts latest snapshot in directory
|
# mounts latest snapshot in directory
|
||||||
|
@ -123,15 +125,11 @@ zfs list -t snapshot | grep "$ZFS_DATASET@$SNAPSHOT_NAME" || true
|
||||||
|
|
||||||
# Attempt to destroy existing snapshot
|
# Attempt to destroy existing snapshot
|
||||||
echo "Attempting to destroy existing snapshot..."
|
echo "Attempting to destroy existing snapshot..."
|
||||||
if zfs list -t snapshot | grep -q "$ZFS_DATASET@$SNAPSHOT_NAME"; then
|
if zfs destroy -r "$ZFS_DATASET@$SNAPSHOT_NAME"; then
|
||||||
if zfs destroy -r "$ZFS_DATASET@$SNAPSHOT_NAME"; then
|
echo "Successfully destroyed old snapshot"
|
||||||
echo "Successfully destroyed old snapshot"
|
|
||||||
else
|
|
||||||
echo "Failed to destroy existing snapshot"
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
else
|
else
|
||||||
echo "No existing snapshot found"
|
echo "Failed to destroy existing snapshot"
|
||||||
|
exit 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# Create new snapshot
|
# Create new snapshot
|
||||||
|
|
Loading…
Reference in a new issue