-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathbackup.sh
executable file
·162 lines (136 loc) · 3.51 KB
/
backup.sh
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
# /bin/bash
# These are the directories that will be backed up:
SYNCTHESE="/boot /etc /home /opt /root /usr"
# This will be the name of the backup:
DIRNAME="$(date +%s)"
function usage {
cat <<\EOF
Arguments:
-c: Encrypted drive to open and use (Required if '-d' is not specified)
-d: Base directory where all backups are stored (Required if '-c' is not specified)
-h: This help screen
-v: Verbose mode
Return values:
0: Backup completed successfully
1: Error encountered during backup
2: Invalid arguments
EOF
}
# This puts the backups into a working state if any error is encountered.
function die {
if [ -d ${SYNCDIR}/${DIRNAME} ]; then
echo "Error encountered, rolling back progress..."
rm -rf ${SYNCDIR}/${DIRNAME}
else
echo "Error encountered"
fi
echo "Exiting"
exit 1
}
# This sets up the backup structure.
function init {
# Figure out where we're going to back up to.
if [ -n "${CRYPT}" ]; then
# We need to backup to an encrypted drive.
CRYPTNAME="backup-${RANDOM}"
cryptsetup open ${CRYPT} ${CRYPTNAME} || die
mount /dev/mapper/${CRYPTNAME} /mnt || die
SYNCTO="/mnt"
else
# Make sure we were passed a base directory.
if [ -z "${SYNCTO}" ] || [ ! -d "${SYNCTO}" ]; then
echo "Missing base directory ('-d')"
echo ""
usage
exit 2
fi
fi
# Go to the main directory where all the backups are stored.
if [ "$(basename ${SYNCTO})" == "backups" ]; then
SYNCDIR="${SYNCTO}"
else
SYNCDIR="${SYNCTO}/backups"
if [ ! -d ${SYNCDIR} ]; then
mkdir ${SYNCDIR} || die
fi
fi
cd ${SYNCDIR} || die
}
# This creates a duplicate of the latest backup by mirroring the directory structure and hard
# linking all files.
function mirror_backup {
# If we have a previous backup, then we'll make an identical mirror of it for later syncing.
if [ -L latest ]; then
echo "Mirroring latest backup..."
# Because we are preserving links with the -a switch, we have to follow the latest link
# first before creating the mirror.
cp -al ${VERBOSE} $(readlink latest) ${DIRNAME} || die
sync
echo "Mirror complete"
fi
}
# This syncs the chosen directories with the latest backup, creating a snapshot of the current
# system.
function perform_backup {
echo "Performing backup on these directories: ${SYNCTHESE}"
# If Verbose mode is enabled, then we'll show all of rsync's output. Otherwise, we'll only show
# the last 2 lines, which detail the size of the archive and how much was actually backed up.
if [ -n ${VERBOSE} ]; then
rsync --archive --hard-links --delete --verbose ${SYNCTHESE} ${DIRNAME} || die
else
rsync --archive --hard-links --delete --verbose ${SYNCTHESE} ${DIRNAME} | tail -n 2 || die
fi
sync
if [ -L latest ]; then
rm latest
fi
ln -s ${DIRNAME} latest
echo "Backup complete"
}
# Currently, this only closes out the encrypted drive, if one was used.
function deinit {
if [ -n "${CRYPTNAME}" ]; then
umount /mnt && cryptsetup close ${CRYPTNAME}
if [ "${?}" == "0" ]; then
echo "Crypt drive closed successfully"
else
echo "Error closing crypt drive"
fi
fi
}
while getopts "c:d:hv" opt; do
case ${opt} in
c)
if [ -n "${SYNCTO}" ]; then
echo "Cannot specify both '-c' and '-d' arguments"
usage
exit 2
fi
CRYPT=${OPTARG}
;;
d)
if [ -n "${CRYPT}" ]; then
echo "Cannot specify both '-c' and '-d' arguments"
usage
exit 2
fi
SYNCTO=${OPTARG}
;;
h)
usage
exit
;;
v)
VERBOSE="--verbose"
;;
*)
echo "Invalid argument: ${opt}"
usage
exit 2
;;
esac
done
init
mirror_backup
perform_backup
deinit