3 # backup-script system for cloning systems using rsync
4 # Copyright (c)2008-2016 Alexander Barton, alex@barton.de
6 # This program is free software; you can redistribute it and/or modify
7 # it under the terms of the GNU General Public License as published by
8 # the Free Software Foundation; either version 2 of the License, or
9 # (at your option) any later version.
10 # Please read the file COPYING, README and AUTHORS for more information.
14 PIDFILE="/var/run/backup-script.pid"
22 declare -i snapshots=0
24 # Default settings, can be overwritten in backup-script.conf:
25 [ -d "/usr/local/etc/backup-script.d" ] \
26 && conf_d="/usr/local/etc/backup-script.d" \
27 || conf_d="/etc/backup-script.d"
28 default_backup_type="rsync"
30 default_target="/var/backups"
36 # Search configuration file (last one is used as default!)
38 "/usr/local/etc/backup-script.conf" \
39 "/etc/backup-script.conf" \
40 "${conf_d}/backup-script.conf" \
41 "/usr/local/etc/backup-script.conf" \
43 if [ -r "$conf" ]; then
44 # shellcheck source=/dev/null
51 echo "Usage: $NAME [--errors|--latest] [--quick] [<job> [<job> [...]]]"
52 echo " $NAME --running"
54 echo " -e, --errors Only show current backups with errors (implies \"--latest\")."
55 echo " -l, --latest Only show latest backup generations."
56 echo " -q, --quick Don't calculate backup sizes."
57 echo " -r, --running Check if an \"backup-script\" task is currently running."
59 echo "When no <job> is given, all defined jobs are listed."
68 if [ "$QUICK" = "0" ]; then
69 size=$(du -Hhs "$1" | cut -f1)
70 # shellcheck disable=SC2086
71 echo "$2 - Size:" $size
85 declare -i duration_t=-1
87 # Read in "stamp file"
88 # shellcheck source=/dev/null
91 if [ $start_t -gt 0 ] && [ $end_t -gt 0 ]; then
92 if [ "$(uname)" = "Linux" ]; then
93 start=$(date -d @"$start_t")
94 end=$(date -d @"$end_t")
96 start=$(date -r "$start_t")
97 end=$(date -r "$end_t")
99 duration_t=$end_t-$start_t
101 if [ "$(uname)" = "Linux" ]; then
102 end=$(LC_ALL=C stat "$1" | grep "^Modify: " \
103 | cut -d':' -f2- | cut -d. -f1)
105 end=$(LC_ALL=C stat -f "%Sc" "$1")
108 # shellcheck disable=SC2086
109 [ -n "$start" ] && echo "$2 - Start date:" $start
110 # shellcheck disable=SC2086
111 [ -n "$end" ] && echo "$2 - End date:" $end
112 if [ $duration_t -gt -1 ]; then
113 declare -i s=$duration_t
114 if [ $s -ge 60 ]; then
115 declare -i m=$((s / 60))
116 declare -i s=$((s % 60))
117 if [ $m -ge 60 ]; then
118 declare -i h=$((m / 60))
119 declare -i m=$((m % 60))
120 if [ $h -ge 24 ]; then
121 declare -i d=$((h / 24))
122 declare -i h=$((h % 24))
123 duration="${d}d${h}h${m}m${s}s"
125 duration="${h}h${m}m${s}s"
128 duration="${m}m${s}s"
133 echo "$2 - Duration:" $duration
138 24) txt=", WARNING (some files vanished during backup)"; ;;
141 [ $code -ge 0 ] && echo "$2 - Result code: ${code}${txt}"
143 echo "$2 - No timestamp recorded! Backup currently running or aborted?"
148 echo " - Snapshot: $1"
150 Check_Stamp "$1/.stamp" " "
155 # shellcheck source=/dev/null
156 [ -r "$1" ] && source "$1"
157 [ -z "$code" ] && code=1
161 if [[ "$1" == "-r" || "$1" == "--running" ]]; then
162 pid="$(cat "$PIDFILE" 2>/dev/null)"
163 if [ -n "$pid" ]; then
164 if kill -0 "$pid" >/dev/null 2>&1; then
165 echo "Backup job running with PID $pid."
167 pstree -ap "$pid" 2>/dev/null
170 echo "No backup running (invalid PID $pid in \"$PIDFILE\")."
174 echo "No backup running (no PID file \"$PIDFILE\" found)."
178 while [ $# -gt 0 ]; do
199 if [ $# -ge 1 ]; then
201 if [ ! -r "${conf_d}/$s" ]; then
202 echo "$NAME: Can' read \"${conf_d}/$s\"!"
205 sys+=("${conf_d}/$s")
211 for f in "${sys[@]}"; do
212 [[ -r "$f" && -f "$f" ]] || continue
214 fname=$(basename "$f")
216 "backup-script.conf"|*.sh)
221 # Set global defaults
223 target="$default_target"
224 generations="$default_generations"
225 backup_type="$default_backup_type"
227 # Read in system configuration file
228 # shellcheck source=/dev/null
231 target="$target/$(basename "$f")"
233 [ -d "$target" ] || continue
235 if [ "$ONLY_ERRORS" != "0" ]; then
236 [[ "$backup_type" = "disabled" ]] && continue
237 [ $generations -gt 0 ] \
238 && result=$(Get_Result_Code "$target/latest/.stamp") \
239 || result=$(Get_Result_Code "$target/.stamp")
240 [[ $result -eq 0 || $result -eq 24 ]] && continue
244 [ "$system" = "$fname" ] && echo "$fname" || echo "$fname [$system]"
246 # System target directory
247 echo "- Target: $target"
249 if [ $generations -gt 0 ]; then
250 if [ "$ONLY_LATEST" = "0" ]; then
251 for s in $target/[0-9]*-[0-9]* $target/current; do
252 [ -e "$s" ] || continue
254 snapshots=$snapshots+1
256 elif [ -e "$target/latest" ]; then
257 Snapshot_Info "$target/latest"
258 snapshots=$snapshots+1
261 # Timestamp and result code
263 Check_Stamp "$target/.stamp"
264 snapshots=$snapshots+1
271 if [ "$ONLY_ERRORS" != "0" ]; then
272 status="failed "; p0="."; pN="!"
274 status=""; p0="!"; pN="."
276 if [ $count -lt 1 ]; then
277 echo "No ${status}backups found${p0}"
280 [ $count -eq 1 ] && sc="" || sc="s"
281 [ $snapshots -eq 1 ] && ss="" || ss="s"
282 echo "$count ${status}system backup$sc found, $snapshots snapshot$ss${pN}"