backupninja/handlers/rsnap
Elijah Saxon 824ef5ad9a added
2006-07-10 20:03:45 +00:00

245 lines
5.9 KiB
Plaintext

# -*- mode: sh; sh-basic-offset: 3; indent-tabs-mode: nil; -*-
#
# rsync backup handler for backupninja
# requires rsync and optional freedups
#
# freedups:
# http://www.stearns.org/freedups/
# http://freshmeat.net/projects/freedups/
#
# rsync:
# http://samba.anu.edu.au/rsync/
# exit on error
#set -e
# System commands used by this script
# replace with absolute path's if neccecary
getconf rm rm
getconf cp cp
getconf touch touch
getconf mv mv
getconf ssh ssh
getconf tr tr
getconf rsync $RSYNC
setsection options
getconf options
getconf label
getconf nicelevel 0
getconf keep 60
setsection source
getconf testconnect no
getconf srchost localhost
getconf compress 1
getconf sshoptions
getconf bandwidthlimit 1000
getconf remote_rsync rsync
getconf numericids 1
getconf include
getconf vsnames all
getconf vsinclude
getconf include
getconf exclude
setsection dest
getconf directory
getconf enable_mv_timestamp_bug no
getconf freedups freedups
getconf enable_freedups no
getconf incremental yes
# Apparently, a bug in some Linux kernels between 2.4.4 and 2.4.9 causes mv to update timestamps;
# this may result in inaccurate timestamps on the snapshot directories.
# Set enable_mv_timestamp_bug=1 to enable this workaround
if [ $enable_mv_timestamp_bug == "yes" ]; then
mv=my_mv
fi;
function my_mv() {
ref=/tmp/makesnapshot-mymv-$$;
$touch -r $1 $ref;
$mv $1 $2;
$touch -r $ref $2;
$rm $ref;
}
if [ $enable_freedups == "yes" ]; then
# $freedups
debug "Not implemented yet!"
fi;
[ "$directory" != "" ] || fatal "Destination directory not set"
[ "$include" != "" ] || fatal "No source includes specified"
### vservers stuff ###
# If vservers are configured, check that the ones listed in $vsnames do exist.
local usevserver=no
if [ $vservers_are_available = yes ]; then
if [ "$vsnames" = all ]; then
vsnames="$found_vservers"
else
if ! vservers_exist "$vsnames" ; then
fatal "At least one of the vservers listed in vsnames ($vsnames) does not exist."
fi
fi
if [ -n "$vsinclude" ]; then
info "Using vservers '$vsnames'"
usevserver=yes
fi
else
[ -z "$vsinclude" ] || warning 'vservers support disabled in backupninja.conf, vsincludes configuration lines will be ignored'
[ -z "$vsnames" ] || warning 'vservers support disabled in backupninja.conf, vsnames configuration line will be ignored'
fi
### see if we can login ###
if [ "$testconnect" == "yes" ]; then
debug "$ssh $sshoptions -o PasswordAuthentication=no $srchost 'echo -n 1'"
if [ ! $test ]; then
result=`ssh $sshoptions -o PasswordAuthentication=no $srchost 'echo -n 1'`
if [ "$result" != "1" ]; then
fatal "Can't connect to $srchost."
else
debug "Connected to $srchost successfully"
fi
fi
fi
### COMMAND-LINE MANGLING ###
[ "$bandwidthlimit" == 1000 ] || options="$options --bwlimit=$bandwidthlimit"
[ "$numericids" == 1 ] || options="$options --numeric-ids "
[ "$compress" == 1 ] || options="$options --compress "
[ "$remote_rsync" == "rsync" ] || options="$options --rsync-path=$remote_rsync"
if [ "$nicelevel" -ne 0 ]; then
nice="nice -n $nicelevel" ;
else
nice="";
fi
execstr="$options --exclude '/' --delete-during --delete-excluded --archive $sshoptions "
if [ "$incremental" == "no" ]; then
execstr="${execstr} --whole-file "
fi
execstr_serverpart="$srchost:/"
### SOURCE ###
set -o noglob
# excludes
for i in $exclude; do
str="${i//__star__/*}"
#execstr="${execstr}--exclude '$str' "
execstr="${execstr}--exclude $str "
done
# includes
for i in $include; do
str="${i//__star__/*}"
#execstr="${execstr}--include '$str' "
execstr="${execstr}--include $str "
done
# vsincludes
if [ $usevserver = yes ]; then
for vserver in $vsnames; do
for vi in $vsinclude; do
str="${vi//__star__/*}"
execstr="${execstr}--include '$label/$vserver$str' "
done
done
fi
### SNAPSHOT ROTATION ###
if [ "$incremental" == "yes" ]; then
debug "starting to rotate the old dirs"
# rotating snapshots
# delete the oldest snapshot, if it exists:
debug "does $directory/$label/$keep exist?"
if [ -d "$directory/$label/$keep" ] ; then
debug "$rm -rf $directory/$label/$keep"
if [ !$test ]; then
#$rm -rf "$directory/$label/$keep" ;
debug "$rm -rf $directory/$label/$keep";
fi;
fi;
# shift the snapshots(s) back by one, if they exist
for (( i=$keep; $i>=0; i--)) ; do
debug "does $directory/$label/$i exist?"
if [ -d "$directory/$label/$i" ] ; then
debug "$mv $directory/$label/$i $directory/$label/$(($i + 1))"
if [ !$test ]; then
$mv "$directory/$label/$i" "$directory/$label/$(($i + 1))"
fi;
fi;
done
# make a hard-link-only (except for dirs) copy of
# assuming that exists, into the new dir
if [ -d "$directory/$label/1" ]; then
debug "$cp -al $directory/$label/1 $directory/$label/0"
if [ !$test ]; then
$cp -al $directory/$label/1 $directory/$label/0 ;
fi;
fi;
fi
set +o noglob
### EXECUTE ###
# exclude everything else, start with root
#execstr="${execstr}--exclude '*' "
# include client-part and server-part
#execstr="$execstr $execstr_serverpart"
execstr=${execstr//\\*/\\\\\\*}
if [ "$debug" == "1" ]; then
execstr=" --verbose $execstr";
# execstr=" --verbose --dry-run $execstr";
else
execstr=" --quiet $execstr";
fi;
debug "$rsync $execstr $execstr_serverpart $directory/$label/0"
# rsync from the system into the latest snapshot (notice that
# rsync behaves like cp --remove-destination by default, so the destination
# is unlinked first. If it were not so, this would copy over the other
# snapshot(s) too!
output=`$nice $rsync $execstr $execstr_serverpart $directory/$label/0 2>&1`
code=$?
# update the mtime of the 0 dir to reflect the snapshot time
$touch $directory/$label/0
if [ $code -eq 0 ]; then
debug $output
info "rsync finished successfully.";
else
debug "returncode $code : $output "
#fatal "rsync failed.";
warning "rsync failed.";
fi;
return 0;