2016-08-03 11:55:03 -07:00
|
|
|
#!/bin/bash
|
|
|
|
# backup-zfs: use zfs send/recv to push/pull snapshots
|
2016-08-04 16:24:08 -07:00
|
|
|
prog="$(basename "$0")"
|
2016-08-03 11:55:03 -07:00
|
|
|
|
|
|
|
usage() {
|
2016-08-04 16:29:28 -07:00
|
|
|
cat >&2 <<-EOF
|
|
|
|
usage: $prog [-hvq] [-t tag] [-k keep] [-d dateopts] src dest
|
|
|
|
use zfs send/recv to push/pull snapshots
|
|
|
|
|
|
|
|
src the source fs, specified as [host:]pool/path/to/fs
|
|
|
|
dest the destination fs parent, specified as [host:]pool/path/to/fs
|
|
|
|
(the final path component of src will be appended to dest)
|
2019-11-11 15:19:24 +01:00
|
|
|
-p ssh port
|
2016-08-04 16:29:28 -07:00
|
|
|
-h help
|
|
|
|
-v verbose mode
|
|
|
|
-q quiet mode
|
|
|
|
-t tag tag to use for naming snapshots (default: backup-zfs)
|
|
|
|
-k keep number of snapshots to keep on src (default: 5)
|
|
|
|
-d dateopts options for date(1) - used to name the snapshots (default: +%F_%T)
|
2017-03-07 09:43:08 -08:00
|
|
|
-s store mode - output snaps from local fs to ssh server
|
|
|
|
-r read mode - read snaps from ssh server to local fs
|
2017-03-07 15:23:20 -08:00
|
|
|
-g gpg-id gpg recipient key id (store mode only)
|
2016-08-04 16:29:28 -07:00
|
|
|
EOF
|
2016-08-03 11:55:03 -07:00
|
|
|
exit $1
|
|
|
|
}
|
|
|
|
|
|
|
|
# log to syslog; if verbose or on a tty, also to stdout
|
|
|
|
# usage: log msg
|
|
|
|
log() {
|
2016-08-04 16:24:08 -07:00
|
|
|
logger -t "$prog" -- "$@"
|
2016-08-03 15:58:03 -07:00
|
|
|
if ! $quiet && [[ -t 1 ]] || $verbose ; then
|
2016-08-03 11:55:03 -07:00
|
|
|
echo "$@" >&2
|
|
|
|
fi
|
|
|
|
}
|
|
|
|
|
|
|
|
# exit with a code & message
|
|
|
|
# usage: die $exitcode msg
|
|
|
|
die() {
|
|
|
|
code="$1"
|
|
|
|
shift
|
|
|
|
if [[ $code -ne 0 ]] ; then
|
|
|
|
verbose=true log "FATAL: $@"
|
|
|
|
else
|
|
|
|
log "$@"
|
|
|
|
fi
|
|
|
|
exit $code
|
|
|
|
}
|
|
|
|
|
|
|
|
# run zfs(1) command either locally or via ssh
|
|
|
|
# usage: ZFS "$host" command args...
|
|
|
|
ZFS() {
|
|
|
|
host="$1"
|
|
|
|
shift
|
|
|
|
if [[ -n $host ]] ; then
|
|
|
|
log "remote ($host): zfs $@"
|
2019-11-11 15:19:24 +01:00
|
|
|
ssh -C -p $port "$host" zfs "$@"
|
2016-08-03 11:55:03 -07:00
|
|
|
else
|
|
|
|
log "local: zfs $@"
|
2019-11-11 15:19:24 +01:00
|
|
|
zfs "$@"
|
2016-08-03 11:55:03 -07:00
|
|
|
fi
|
|
|
|
}
|
|
|
|
|
|
|
|
###
|
|
|
|
### defaults
|
|
|
|
###
|
2016-08-04 16:24:08 -07:00
|
|
|
tag="$prog"
|
2016-08-04 16:23:02 -07:00
|
|
|
dateopts="+%F_%T"
|
2016-08-03 11:55:03 -07:00
|
|
|
keep=5
|
|
|
|
verbose=false
|
2016-08-03 15:58:03 -07:00
|
|
|
quiet=false
|
2017-03-07 09:43:08 -08:00
|
|
|
tossh=false
|
|
|
|
fromssh=false
|
2019-11-11 15:19:24 +01:00
|
|
|
port=22
|
2016-08-03 11:55:03 -07:00
|
|
|
|
|
|
|
###
|
|
|
|
### parse options
|
|
|
|
###
|
2019-11-11 15:19:24 +01:00
|
|
|
while getopts "hvqk:p:t:d:srg:" opt ; do
|
2016-08-03 11:55:03 -07:00
|
|
|
case $opt in
|
|
|
|
h) usage 0 ;;
|
2016-10-10 09:56:22 -07:00
|
|
|
v)
|
|
|
|
verbose=true
|
|
|
|
send_opts="-v"
|
|
|
|
recv_opts="-v"
|
|
|
|
;;
|
2016-08-03 15:58:03 -07:00
|
|
|
q) quiet=true ;;
|
2016-08-03 11:55:03 -07:00
|
|
|
k) keep=$OPTARG ;;
|
2019-11-11 15:19:24 +01:00
|
|
|
p) port=$OPTARG ;;
|
2016-08-03 11:55:03 -07:00
|
|
|
t) tag=$OPTARG ;;
|
2016-08-04 16:23:02 -07:00
|
|
|
d) dateopts=$OPTARG ;;
|
2017-03-07 09:43:08 -08:00
|
|
|
s) tossh=true ;;
|
|
|
|
r) fromssh=true ;;
|
2017-03-07 15:23:20 -08:00
|
|
|
g) gpgid="$OPTARG" ;;
|
2016-08-03 11:55:03 -07:00
|
|
|
*) usage 1 ;;
|
|
|
|
esac
|
|
|
|
done
|
|
|
|
shift $((OPTIND-1))
|
2016-08-04 16:23:02 -07:00
|
|
|
date="$(date $dateopts)"
|
2017-03-07 09:43:08 -08:00
|
|
|
$tossh && $fromssh && die 1 "-s and -r are mutually exclusive"
|
2017-03-07 15:23:20 -08:00
|
|
|
if ! $tossh && [[ -n $gpgid ]] ; then
|
|
|
|
die 1 "-g can only be used with -s"
|
|
|
|
fi
|
2016-08-03 11:55:03 -07:00
|
|
|
|
|
|
|
###
|
|
|
|
### parse src & dest host/fs info
|
|
|
|
###
|
|
|
|
# fail if there's ever >1 colon
|
|
|
|
if [[ $1 =~ :.*: || $2 =~ :.*: ]] ; then
|
|
|
|
die 1 "invalid fsspec: '$1' or '$2'"
|
|
|
|
fi
|
|
|
|
|
|
|
|
# fail if src or dest isn't specified
|
|
|
|
if [[ -z $1 || -z $2 ]] ; then
|
|
|
|
usage 1
|
|
|
|
fi
|
|
|
|
src="$1"
|
|
|
|
dest="$2"
|
|
|
|
|
2017-03-07 09:43:08 -08:00
|
|
|
###
|
|
|
|
### ssh mode - output snaps from local fs to ssh or read snaps from ssh to local fs
|
|
|
|
if $tossh ; then
|
|
|
|
log "sending from local zfs filesystem to SSH server"
|
|
|
|
|
|
|
|
# make sure src exists
|
|
|
|
if [[ $src =~ : ]] ; then
|
|
|
|
die 1 "$src must be a local zfs filesystem"
|
|
|
|
elif [[ $(ZFS "" list -H -o name "$src" 2>/dev/null) != $src ]] ; then
|
|
|
|
die 1 "$src must be a local zfs filesystem"
|
|
|
|
fi
|
|
|
|
|
|
|
|
# split dest to components
|
|
|
|
if [[ $dest =~ : ]] ; then
|
|
|
|
desthost="${dest%:*}"
|
|
|
|
destpath="${dest#*:}"
|
|
|
|
else
|
|
|
|
die 1 "$dest must be ssh host:path"
|
|
|
|
fi
|
|
|
|
|
|
|
|
# get the last src component
|
|
|
|
srcbase="${src##*/}"
|
|
|
|
|
|
|
|
###
|
|
|
|
### create new snapshot on src
|
|
|
|
###
|
2017-03-07 15:23:40 -08:00
|
|
|
snap="${tag}_$date"
|
|
|
|
cur="$src@$snap"
|
2017-03-07 09:43:08 -08:00
|
|
|
ZFS "$srchost" snapshot -r "$cur" || die $? "zfs snapshot failed"
|
|
|
|
|
|
|
|
###
|
|
|
|
### get newest snapshot on dest - it must exist on src
|
|
|
|
###
|
|
|
|
#last="$(ZFS "$desthost" list -d 1 -t snapshot -H -S creation -o name $destfs/$srcbase | head -n1 | cut -f2 -d@)"
|
2017-03-09 11:55:57 -08:00
|
|
|
last="$(ssh "$desthost" zfslast)"
|
2017-03-07 09:43:08 -08:00
|
|
|
|
|
|
|
###
|
|
|
|
### send
|
|
|
|
###
|
|
|
|
# refuse to send without a valid .last maker
|
|
|
|
if [[ -z $last ]] ; then
|
|
|
|
die 1 "ssh path contains no .last file"
|
|
|
|
# special case: tagged snapshots exist on dest, but src has rotated through all
|
|
|
|
elif ! ZFS "$srchost" list $src@$last &>/dev/null ; then
|
|
|
|
die 1 "no incremental path from from $src to $dest"
|
|
|
|
# normal case: send incremental
|
|
|
|
else
|
2017-03-07 15:23:20 -08:00
|
|
|
log "sending $([[ -n $gpgid ]] && echo "encrypted ")incremental snapshot from $src to $dest (${last#${tag}_}..${cur#*@${tag}_})"
|
2019-11-11 15:19:24 +01:00
|
|
|
#ZFS "$srchost" send $send_opts -R -i "$last" "$cur" | ZFS "$desthost" receive $recv_opts -Fue "$destfs" || die $? "zfs incremental send failed"
|
2017-03-07 15:23:20 -08:00
|
|
|
if [[ -n $gpgid ]] ; then
|
2019-11-11 15:19:24 +01:00
|
|
|
ZFS "$srchost" send $send_opts -R -i "$last" "$cur" \
|
2017-03-07 15:23:20 -08:00
|
|
|
| gpg --trust-model always --encrypt --recipient "$gpgid" \
|
2017-03-09 11:55:57 -08:00
|
|
|
| ssh "$desthost" zfswrite "${tag}_$date.zfssnap.gpg" \
|
2017-03-07 15:23:20 -08:00
|
|
|
|| die $? "zfs incremental send failed"
|
2017-03-09 11:55:57 -08:00
|
|
|
ssh "$desthost" zfslast "$snap"
|
2017-03-07 15:23:20 -08:00
|
|
|
else
|
2019-11-11 15:19:24 +01:00
|
|
|
ZFS "$srchost" send $send_opts -R -i "$last" "$cur" \
|
2017-03-09 11:55:57 -08:00
|
|
|
| ssh "$desthost" zfswrite "${tag}_$date.zfssnap" \
|
2017-03-07 15:23:20 -08:00
|
|
|
|| die $? "zfs incremental send failed"
|
2017-03-09 11:55:57 -08:00
|
|
|
ssh "$desthost" zfslast "$snap"
|
2017-03-07 15:23:20 -08:00
|
|
|
fi
|
2017-03-07 09:43:08 -08:00
|
|
|
fi
|
|
|
|
|
|
|
|
exit
|
|
|
|
elif $fromssh ; then
|
|
|
|
log "receving from SSH server to local zfs filesystem"
|
|
|
|
|
|
|
|
# make sure dest exists
|
|
|
|
if [[ $dest =~ : ]] ; then
|
|
|
|
die 1 "$dest must be a local zfs filesystem"
|
|
|
|
elif [[ $(ZFS "" list -H -o name "$dest" 2>/dev/null) != $dest ]] ; then
|
|
|
|
die 1 "$dest must be a local zfs filesystem"
|
|
|
|
fi
|
|
|
|
|
|
|
|
# split src into components
|
|
|
|
if [[ $src =~ : ]] ; then
|
|
|
|
srchost="${src%:*}"
|
|
|
|
srcpath="${src#*:}"
|
|
|
|
else
|
|
|
|
die 1 "$src must be ssh host:path"
|
|
|
|
fi
|
|
|
|
|
|
|
|
###
|
|
|
|
### receive
|
|
|
|
###
|
|
|
|
log "receiving incremental snapshot from $src to $dest"
|
2019-11-11 15:19:24 +01:00
|
|
|
#ZFS "$srchost" send $send_opts -R -i "$last" "$cur" | ZFS "$desthost" receive $recv_opts -Fue "$destfs" || die $? "zfs incremental send failed"
|
2017-03-10 10:10:57 -08:00
|
|
|
for file in $(ssh "$srchost" zfsfind | sort) ; do
|
2017-08-01 14:53:58 -07:00
|
|
|
log "receiving $file from $srchost"
|
2017-03-07 15:23:20 -08:00
|
|
|
if [[ $file =~ \.gpg$ ]] ; then
|
2017-03-09 11:55:57 -08:00
|
|
|
ssh "$srchost" zfsget "$file" | gpg | ZFS "$desthost" receive $recv_opts -Fue "$dest" \
|
|
|
|
&& ssh "$srchost" rm "$file"
|
2017-03-07 15:23:20 -08:00
|
|
|
else
|
2017-03-09 11:55:57 -08:00
|
|
|
ssh "$srchost" zfsget "$file" | ZFS "$desthost" receive $recv_opts -Fue "$dest" \
|
|
|
|
&& ssh "$srchost" rm "$file"
|
2017-03-07 15:23:20 -08:00
|
|
|
fi
|
2017-03-07 09:43:08 -08:00
|
|
|
done
|
|
|
|
|
|
|
|
exit
|
|
|
|
fi
|
|
|
|
|
2016-08-03 11:55:03 -07:00
|
|
|
# discard anything before a colon to get the fs
|
|
|
|
srcfs="${src#*:}"
|
|
|
|
destfs="${dest#*:}"
|
|
|
|
|
|
|
|
# iff there is a colon, discard everything after it to get the host
|
|
|
|
[[ $src =~ : ]] && srchost="${src%:*}"
|
|
|
|
[[ $dest =~ : ]] && desthost="${dest%:*}"
|
|
|
|
|
|
|
|
# get the last src component
|
|
|
|
srcbase="${srcfs##*/}"
|
|
|
|
|
2016-09-12 11:28:48 -07:00
|
|
|
# ensure the destination fs exists before proceeding
|
|
|
|
if [[ $(ZFS "$desthost" list -H -o name "$destfs" 2>/dev/null) != $destfs ]] ; then
|
|
|
|
die 1 "destination fs '$destfs' doesn't exist"
|
|
|
|
fi
|
|
|
|
|
2016-08-03 11:55:03 -07:00
|
|
|
###
|
|
|
|
### create new snapshot on src
|
|
|
|
###
|
|
|
|
cur="$srcfs@${tag}_$date"
|
2016-09-12 11:39:29 -07:00
|
|
|
ZFS "$srchost" snapshot -r "$cur" || die $? "zfs snapshot failed"
|
2016-08-03 11:55:03 -07:00
|
|
|
|
|
|
|
###
|
2016-09-16 15:50:26 -07:00
|
|
|
### get newest snapshot on dest - it must exist on src
|
2016-08-03 11:55:03 -07:00
|
|
|
###
|
2016-09-16 15:50:26 -07:00
|
|
|
last="$(ZFS "$desthost" list -d 1 -t snapshot -H -S creation -o name $destfs/$srcbase | head -n1 | cut -f2 -d@)"
|
2016-08-03 11:55:03 -07:00
|
|
|
|
|
|
|
###
|
|
|
|
### send & receive
|
|
|
|
###
|
|
|
|
# 1st time: send full snapshot
|
|
|
|
if [[ -z $last ]] ; then
|
|
|
|
log "sending full recursive snapshot from $src to $dest"
|
2016-10-10 09:56:22 -07:00
|
|
|
ZFS "$srchost" send $send_opts -R "$cur" | ZFS "$desthost" receive $recv_opts -Fue "$destfs" || die $? "zfs full send failed"
|
2016-08-03 11:55:03 -07:00
|
|
|
# special case: tagged snapshots exist on dest, but src has rotated through all
|
|
|
|
elif ! ZFS "$srchost" list $srcfs@$last &>/dev/null ; then
|
|
|
|
die 1 "no incremental path from from $src to $dest"
|
|
|
|
# normal case: send incremental
|
|
|
|
else
|
|
|
|
log "sending incremental snapshot from $src to $dest (${last#${tag}_}..${cur#*@${tag}_})"
|
2019-11-11 15:19:24 +01:00
|
|
|
ZFS "$srchost" send $send_opts -R -i "$last" "$cur" | ZFS "$desthost" receive $recv_opts -Fue "$destfs" || die $? "zfs incremental send failed"
|
2016-08-03 11:55:03 -07:00
|
|
|
fi
|
|
|
|
|
|
|
|
###
|
|
|
|
### clean up old snapshots
|
|
|
|
###
|
|
|
|
for snap in $(ZFS "$srchost" list -d 1 -t snapshot -H -S creation -o name $srcfs \
|
|
|
|
| grep -F "@${tag}_" | cut -f2 -d@ | tail -n+$((keep+1)) ) ;
|
|
|
|
do
|
|
|
|
ZFS "$srchost" destroy -r $srcfs@$snap
|
|
|
|
done
|