2 # backupninja handler to do incremental backups using
3 # rsync and hardlinks, based on
5 # http://www.mikerubel.org/computers/rsync_snapshots/
7 # feedback: rhatto at riseup.net | gpl
8 # lot of enhancements grabbed from "rsnap" handler by paulv at bikkel.org
14 # log = rsync log file
15 # partition = partition where the backup lives
16 # fscheck = set to 1 if fsck should run on $partition after the backup is made
17 # read_only = set to 1 if $partition is mounted read-only
18 # mountpoint = backup partition mountpoint or backup main folder
19 # backupdir = folder relative do $mountpoint where the backup should be stored
20 # days = number of backup increments (min = 5)
21 # lockfile = lockfile to be kept during backup execution
22 # enable_mv_timestamp_bug = set to "yes" if your system isnt handling timestamps correctly
26 # from = local or remote
27 # testconnect = when "yes", test the connection for a remote source before backup
28 # include = include folder on backup
29 # exclude = exclude folder on backup
30 # ssh = ssh command line (remote only)
31 # rsync = rsync program
32 # rsync_options = rsync command options
33 # exclude_vserver = vserver-name (valid only if vservers = yes on backupninja.conf)
34 # numericids = when set to 1, use numeric ids instead of user/group mappings on rsync
35 # compress = if set to 1, compress data on rsync (remote source only)
36 # bandwidthlimit = set a badnwidth limit in kbps (remote source only)
37 # remote_rsync = remove rsync program (remote source only)
40 # initscripts = absolute path where scripts are located
41 # service = script name to be stoped at the begining of the backup and started at its end
43 # You can also specify some system comands if you don't want the default system values:
48 # touch = touch command
52 # You dont need to manually specify vservers using "include = /vservers".
53 # They are automatically backuped if vserver is set to "yes" on you backupninja.conf.
56 # config file evaluation
66 getconf log /var/log/backupninja-rub.log
76 getconf enable_mv_timestamp_bug no
81 getconf testconnect no
83 getconf rsync_options "-av --delete"
89 getconf exclude_vserver
92 getconf bandwidthlimit
93 getconf remote_rsync rsync
99 # function definitions
103 if [[ "$2" < 4 ]]; then
104 error "Rotate: minimum of 4 rotations"
108 if [ -d $1.$2 ]; then
109 $nice $mv /$1.$2 /$1.tmp
112 for ((n=`echo "$2 - 1" | bc`; n >= 0; n--)); do
113 if [ -d $1.$n ]; then
114 dest=`echo "$n + 1" | bc`
115 $nice $mv /$1.$n /$1.$dest
120 if [ -d $1.tmp ]; then
121 $nice $mv /$1.tmp /$1.0
125 $nice $cp -alf /$1.1/. /$1.0
130 function move_files {
132 ref=$tmp/makesnapshot-mymv-$$;
140 backupdir="$mountpoint/$backupdir"
142 # does $backupdir exists?
144 if [ ! -d "$backupdir" ]; then
145 error "Backupdir $backupdir does not exist"
149 # setup number of increments
151 if [ -z "$days" ]; then
154 keep="`echo $days - 1 | bc -l`"
159 if [ ! -z "$lockfile" ]; then
160 $touch $lockfile || warning "Could not create lockfile $lockfile"
165 if [ ! -z "$nicelevel" ]; then
166 nice="nice -n $nicelevel"
173 if [ "$from" == "remote" ] && [ "$testconnect" == "yes" ]; then
174 debug "$ssh -o PasswordAuthentication=no $user@$host 'echo -n 1'"
175 result=`ssh -o PasswordAuthentication=no $user@$host 'echo -n 1'`
176 if [ "$result" != "1" ]; then
177 fatal "Can't connect to $host as $user."
179 debug "Connected to $srchost successfully"
183 # rsync options for local sources
185 if [ "$from" == "local" ]; then
187 rsync_local_options="$rsync_options"
189 if [ ! -z "$numericids" ]; then
190 rsync_local_options="$rsync_local_options --numeric-ids "
195 # rsync options for remote sources
197 if [ "$from" == "remote" ]; then
199 rsync_remote_options="$rsync_options --rsync-path=$remote_rsync"
201 if [ "$compress" == "1" ]; then
202 rsync_remote_options="$rsync_remote_options --compress"
205 if [ ! -z "$bandwidthlimit" ]; then
206 rsync_remote_options="$rsync_remote_options --bwlimit=$bandwidthlimit"
209 if [ ! -z "$numericids" ]; then
210 rsync_remote_options="$rsync_remote_options --numeric-ids"
217 if [ $enable_mv_timestamp_bug == "yes" ]; then
223 for path in $exclude; do
224 EXCLUDES="$EXCLUDES --exclude=$path"
229 if [ ! -z "$service" ]; then
230 for daemon in $service; do
231 info "Stopping service $daemon..."
232 $initscripts/$daemon stop
236 echo "Starting backup at `date`" >> $log
238 # mount backup destination folder as read-write
240 if [ "$read_only" == "1" ] || [ "$read_only" == "yes" ]; then
241 if [ -d "$mountpoint" ]; then
242 mount -o remount,rw $mountpoint
244 error "Could not mount $mountpoint"
250 # add vservers to included folders
252 if [ "$vservers_are_available" == "yes" ]; then
254 # sane permission on backup
255 mkdir -p $backupdir/$VROOTDIR
256 chmod 000 $backupdir/$VROOTDIR
258 for candidate in $found_vservers; do
259 candidate="`basename $candidate`"
260 found_excluded_vserver="0"
261 for excluded_vserver in $exclude_vserver; do
262 if [ "$excluded_vserver" == "$candidate" ]; then
263 found_excluded_vserver="1"
267 if [ "$found_excluded_vserver" == "0" ]; then
268 include="$include $VROOTDIR/$candidate"
273 # the backup procedure
275 for SECTION in $include; do
277 section="`basename $SECTION`"
279 if [ ! -d "$backupdir/$SECTION/$section.0" ]; then
280 mkdir -p $backupdir/$SECTION/$section.0
283 info "Rotating $backupdir/$SECTION/$section..."
284 echo "Rotating $backupdir/$SECTION/$section..." >> $log
285 rotate $backupdir/$SECTION/$section $keep
286 info "Syncing $SECTION on $backupdir/$SECTION/$section.0..."
288 if [ "$from" == "local" ]; then
289 debug $rsync $rsync_local_options $EXCLUDES /$SECTION/ $backupdir/$SECTION/$section.0/
290 $nice $rsync $rsync_local_options $EXCLUDES /$SECTION/ $backupdir/$SECTION/$section.0/ >> $log
291 if [ "$?" != "0" ]; then
292 warning "Rsync error when trying to transfer $SECTION"
294 elif [ "$from" == "remote" ]; then
295 if [ -z "$user" ] || [ -z "$host" ]; then
296 error "Config file error: either user or host was not specified"
299 debug $nice $rsync $rsync_remote_options $EXCLUDES -e "$ssh" $user@$host:/$SECTION/ $backupdir/$SECTION/$section.0
300 $nice $rsync $rsync_remote_options $EXCLUDES -e "$ssh" $user@$host:/$SECTION/ $backupdir/$SECTION/$section.0 >> $log
301 if [ "$?" != "0" ]; then
302 warning "Rsync error when trying to transfer $SECTION"
306 error "Invalid source $from"
310 $touch $backupdir/$SECTION/$section.0
314 # remount backup destination as read-only
316 if [ "$read_only" == "1" ] || [ "$read_only" == "yes" ]; then
317 mount -o remount,ro $mountpoint
320 # check partition for errors
322 if [ "$fscheck" == "1" ] || [ "$fscheck" == "yes" ]; then
325 warning "Could not umount $mountpoint to run fsck"
327 $nice $fsck -v -y $partition >> $log
334 if [ ! -z "$service" ]; then
335 for daemon in $service; do
336 info "Starting service $daemon..."
337 $initscripts/$daemon start
341 # removes the lockfile
343 if [ ! -z "$lockfile" ]; then
344 $rm $lockfile || warning "Could not remove lockfile $lockfile"
347 echo "Finnishing backup at `date`" >> $log