-
Notifications
You must be signed in to change notification settings - Fork 0
/
CuboxBackup.sh
485 lines (404 loc) · 13.3 KB
/
CuboxBackup.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
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
#!/bin/bash
#
# Utility script to backup SD Card to a sparse image file
# mounted as a filesystem in a file, allowing for efficient incremental
# backups using rsync
#
# The backup is taken while the system is up, so it's a good idea to stop
# programs and services which modifies the filesystem and needed a consistant state
# of their file.
# Especially applications which use databases needs to be stopped (and the database systems too).
#
# So it's a smart idea to put all these stop commands in a script and perfom it before
# starting the backup. After the backup terminates normally you may restart all stopped
# applications or just reboot the system.
#
SDCARD=/dev/mmcblk1
setup() {
#
# Define some fancy colors only if connected to a terminal.
# Thus output to file is no more cluttered
#
[ -t 1 ] && {
RED=$(tput setaf 1)
GREEN=$(tput setaf 2)
YELLOW=$(tput setaf 3)
BLUE=$(tput setaf 4)
MAGENTA=$(tput setaf 5)
CYAN=$(tput setaf 6)
WHITE=$(tput setaf 7)
RESET=$(tput setaf 9)
BOLD=$(tput bold)
NOATT=$(tput sgr0)
} || {
RED=""
GREEN=""
YELLOW=""
BLUE=""
MAGENTA=""
CYAN=""
WHITE=""
RESET=""
BOLD=""
NOATT=""
}
MYNAME=$(basename $0)
MYDIR=$(dirname $0)
}
# Echos traces with yellow text to distinguish from other output
trace() {
echo -e "${YELLOW}${1}${NOATT}"
}
# Echos en error string in red text and exit
error() {
echo -e "${RED}${1}${NOATT}" >&2
exit 1
}
version() {
trace "$Date$"
trace "$Revision$"
trace " "
}
# Creates a sparse ${IMAGE} clone of ${SDCARD} and attaches to ${LOOPBACK}
do_create() {
local ubootdir
SIZE=${SIZE:-6000}
#
# /usr/lib/linux-u-boot-current-cubox-i_23.08.0-trunk_armhf/
#
#
local ubootinst="/usr/lib/u-boot/platform_install.sh"
[ -f ${ubootinst} ] || error "${ubootinst} not found!"
# this sets DIR to the directory containing the spl uns u-boot of the current system
. /usr/lib/u-boot/platform_install.sh
local SPL=${DIR}/SPL.sdhc
local UBOOT=${DIR}/u-boot.img.sdhc
[ -f ${SPL} ] || error "SPL not found."
[ -f ${UBOOT} ] || error "UBOOT not found."
trace "Creating sparse ${IMAGE}, the apparent size of $SDCARD"
rm ${IMAGE} >/dev/null 2>&1
dd if=/dev/zero of=${IMAGE} bs=${BLOCKSIZE} count=0 seek=${SIZE}
[ -s ${IMAGE} ] || error "${IMAGE} has not been created or has zero size"
trace "Attaching ${IMAGE} to ${LOOPBACK}"
losetup ${LOOPBACK} ${IMAGE}
LOOP=$(losetup -f)
losetup ${LOOP} ${IMAGE}
trace "Creating partitions on ${LOOPBACK}"
parted -s ${LOOPBACK} mktable msdos
parted -s ${LOOPBACK} mkpart primary ext4 4MiB 100%
trace "Formatting partitions"
partx --add ${LOOPBACK}
mkfs.ext4 ${LOOPBACK}p1
trace "writing u-boot to ${LOOP}"
dd if=${SPL} of=${LOOP} bs=1k seek=1 oflag=sync status=noxfer
dd if=${UBOOT} of=${LOOP} bs=1k seek=69 oflag=sync status=noxfer
losetup -d ${LOOP}
}
# Mounts the ${IMAGE} to ${LOOPBACK} (if needed) and ${MOUNTDIR}
do_mount() {
# Check if do_create already attached the SD Image
[ $(losetup -f) = ${LOOPBACK} ] && {
trace "Attaching ${IMAGE} to ${LOOPBACK}"
losetup ${LOOPBACK} ${IMAGE}
partx --add ${LOOPBACK}
}
trace "Mounting ${LOOPBACK}p1 to ${MOUNTDIR}"
[ ! -n "${opt_mountdir}" ] && mkdir ${MOUNTDIR}
mount ${LOOPBACK}p1 ${MOUNTDIR}
}
do_check() {
do_umount
# Check if do_create already attached the SD Image
[ $(losetup -f) = ${LOOPBACK} ] && {
msg "Attaching ${IMAGE} to ${LOOPBACK}"
losetup ${LOOPBACK} "${IMAGE}"
partx --add ${LOOPBACK}
}
fsck -y ${LOOPBACK}p1 || {
msgwarn "Checking ${LOOPBACK}p1 returned_:${err}"
return 1
}
msg "Detaching ${IMAGE} from ${LOOPBACK}"
partx --delete ${LOOPBACK}
losetup -d ${LOOPBACK}
}
# Rsyncs content of ${SDCARD} to ${IMAGE} if properly mounted
do_backup() {
local rsyncopt
rsyncopt="-aEvx --del --stats"
[ -n "${opt_log}" ] && rsyncopt="$rsyncopt --log-file ${LOG}"
if mountpoint -q ${MOUNTDIR}; then
trace "Starting rsync backup of / to ${MOUNTDIR}"
rsync $rsyncopt --exclude='.gvfs/**' \
--exclude='tmp/**' \
--exclude='proc/**' \
--exclude='run/**' \
--exclude='sys/**' \
--exclude='mnt/**' \
--exclude='lost+found/**' \
--exclude='var/swap ' \
--exclude='home/*/.cache/**' \
--exclude='var/cache/apt/archives/**' \
--exclude='var/lib/docker/' \
--exclude='var/lib/containerd/' \
/ ${MOUNTDIR}/
#
# Set the UUID in armbianEnv.txt and fstab
#
UUIDorg=$(blkid -s UUID -o value ${SDCARD}p1)
UUIDloop=$(blkid -s UUID -o value ${LOOPBACK}p1)
trace "changeing ${MOUNTDIR}/boot/armbianEnv.txt from ${UUIDorg} to ${UUIDloop}"
sed -i s/${UUIDorg}/${UUIDloop}/ ${MOUNTDIR}/boot/armbianEnv.txt
trace "changeing ${MOUNTDIR}/etc/fstab from ${UUIDorg} to ${UUIDloop}"
sed -i s/${UUIDorg}/${UUIDloop}/ ${MOUNTDIR}/etc/fstab
trace "\n${MOUNTDIR}/boot/armbianEnv.txt:"
cat ${MOUNTDIR}/boot/armbianEnv.txt
trace "\n${MOUNTDIR}/etc/fstab:"
cat ${MOUNTDIR}/etc/fstab
trace "\n"
else
trace "Skipping rsync since ${MOUNTDIR} is not a mount point"
fi
}
do_showdf() {
echo -n "${GREEN}"
df -m ${LOOPBACK}p1
echo -n "$NOATT"
}
# Unmounts the ${IMAGE} from ${MOUNTDIR} and ${LOOPBACK}
do_umount() {
trace "Flushing to disk"
sync
sync
trace "Unmounting ${LOOPBACK}1 from ${MOUNTDIR}"
umount ${MOUNTDIR}
[ ! -n "${opt_mountdir}" ] && rmdir ${MOUNTDIR}
trace "Detaching ${IMAGE} from ${LOOPBACK}"
partx --delete ${LOOPBACK}
losetup -d ${LOOPBACK}
}
#
# resize image
#
do_resize() {
do_umount >/dev/null 2>&1
truncate --size=+1G "${IMAGE}"
losetup ${LOOPBACK} "${IMAGE}"
parted -s ${LOOPBACK} resizepart 1 100%
partx --add ${LOOPBACK}
e2fsck -f ${LOOPBACK}p1
resize2fs ${LOOPBACK}p1
}
# Compresses ${IMAGE} to ${IMAGE}.gz using a temp file during compression
do_compress() {
trace "Compressing ${IMAGE} to ${IMAGE}.gz"
pv -tpreb ${IMAGE} | gzip >${IMAGE}.gz.tmp
[ -s ${IMAGE}.gz.tmp ] && {
mv -f ${IMAGE}.gz.tmp ${IMAGE}.gz
[ -n "${opt_delete}" ] && rm -f ${IMAGE}
}
}
# Tries to cleanup after Ctrl-C interrupt
ctrl_c() {
trace "Ctrl-C detected."
if [ -s ${IMAGE}.gz.tmp ]; then
rm ${IMAGE}.gz.tmp
else
do_umount
fi
[ -n "${opt_log}" ] && trace "See rsync log in ${LOG}"
error "SD Image backup process interrupted"
}
# Prints usage information
usage() {
cat <<EOF
${MYNAME}
Usage:
${MYNAME} ${BOLD}start${NOATT} [-clzdf] [-L logfile] [-i sdcard] sdimage
${MYNAME} ${BOLD}mount${NOATT} [-c] sdimage [mountdir]
${MYNAME} ${BOLD}umount${NOATT} sdimage [mountdir]
${MYNAME} ${BOLD}gzip${NOATT} [-df] sdimage
Commands:
${BOLD}start${NOATT} starts complete backup of the SD Card to 'sdimage'
${BOLD}mount${NOATT} mounts the 'sdimage' to 'mountdir' (default: /mnt/'sdimage'/)
${BOLD}umount${NOATT} unmounts the 'sdimage' from 'mountdir'
${BOLD}check${NOATT} filesystemcheck on 'sdimage'
${BOLD}gzip${NOATT} compresses the 'sdimage' to 'sdimage'.gz
${BOLD}showdf${NOATT} shows allocation of the image
${BOLD}version${NOATT} show script version
Options:
${BOLD}-c${NOATT} creates the SD Image if it does not exist
${BOLD}-l${NOATT} writes rsync log to 'sdimage'-YYYYmmddHHMMSS.log
${BOLD}-z${NOATT} compresses the SD Image (after backup) to 'sdimage'.gz
${BOLD}-d${NOATT} deletes the SD Image after successful compression
${BOLD}-f${NOATT} forces overwrite of 'sdimage'.gz if it exists
${BOLD}-L logfile${NOATT} writes rsync log to 'logfile'
${BOLD}-i sdcard${NOATT} specifies the SD Card location (default: $SDCARD)
${BOLD}-s Mb${NOATT} specifies the size of image in MB (default: Size of $SDCARD)
Examples:
${MYNAME} start -c /path/to/imx6_backup.img
starts backup to 'imx6_backup.img', creating it if it does not exist
${MYNAME} start -c -s 8000 /path/to/imx6_backup.img
starts backup to 'imx6_backup.img', creating it with a size of 8000mb.
You are responsible for defining a size sufficiant to hold all data.
${MYNAME} start /path/to/\$(uname -n).img
uses the hostname as the SD Image filename
${MYNAME} start -cz /path/to/\$(uname -n)-\$(date +%Y-%m-%d).img
uses the hostname and today's date as the SD Image filename,
creating it if it does not exist, and compressing it after backup
${MYNAME} mount /path/to/\$(uname -n).img /mnt/cubox_image
mounts the SD Image in /mnt/cubox_image
${MYNAME} umount /path/to/cubox-$(date +%Y-%m-%d).img
unmounts the SD Image from default mountdir (/mnt/cubox-$(date +%Y-%m-%d).img/)
EOF
}
setup
# Read the command from command line
case ${1} in
start | mount | umount | gzip | cloneid | showdf | resize | version)
opt_command=${1}
;;
-h | --help)
usage
exit 0
;;
*)
error "Invalid command or option: ${1}\nSee '${MYNAME} --help' for usage"
;;
esac
shift 1
# Make sure we have root rights
[ $(id -u) -ne 0 ] && error "Please run as root. Try sudo."
# Default size, can be overwritten by the -s option
SIZE=$(blockdev --getsz $SDCARD)
BLOCKSIZE=$(blockdev --getss $SDCARD)
# Read the options from command line
while getopts ":czdflL:i:s:" opt; do
case ${opt} in
c) opt_create=1 ;;
z) opt_compress=1 ;;
d) opt_delete=1 ;;
f) opt_force=1 ;;
l) opt_log=1 ;;
L)
opt_log=1
LOG=${OPTARG}
;;
i) SDCARD=${OPTARG} ;;
s)
SIZE=${OPTARG}
BLOCKSIZE=1M
;;
\?) error "Invalid option: -$OPTARG\nSee '${MYNAME} --help' for usage" ;;
:) error "Option -${OPTARG} requires an argument\nSee '${MYNAME} --help' for usage" ;;
esac
done
shift $((OPTIND - 1))
# Read the sdimage path from command line
IMAGE=${1}
[ -z ${IMAGE} ] && error "No sdimage specified"
# Check if sdimage exists
if [ ${opt_command} = umount ] || [ ${opt_command} = gzip ]; then
[ ! -f ${IMAGE} ] && error "${IMAGE} does not exist"
else
if [ ! -f ${IMAGE} ] && [ ! -n "${opt_create}" ]; then
error "${IMAGE} does not exist\nUse -c to allow creation"
fi
fi
# Check if we should compress and sdimage.gz exists
if [ -n "${opt_compress}" ] || [ ${opt_command} = gzip ]; then
if [ -s ${IMAGE}.gz ] && [ ! -n "${opt_force}" ]; then
error "${IMAGE}.gz already exists\nUse -f to force overwriting"
fi
fi
# Define default rsync logfile if not defined
[ -z ${LOG} ] && LOG=${IMAGE}-$(date +%Y%m%d%H%M%S).log
# Identify which loopback device to use
LOOPBACK=$(losetup -j ${IMAGE} | grep -o ^[^:]*)
if [ ${opt_command} = umount ]; then
if [ -z ${LOOPBACK} ]; then
error "No /dev/loop<X> attached to ${IMAGE}"
fi
elif [ ! -z ${LOOPBACK} ]; then
error "${IMAGE} already attached to ${LOOPBACK} mounted on $(grep ${LOOPBACK}p1 /etc/mtab | cut -d ' ' -f 2)/"
else
LOOPBACK=$(losetup -f)
fi
# Read the optional mountdir from command line
MOUNTDIR=${2}
if [ -z ${MOUNTDIR} ]; then
MOUNTDIR=/mnt/$(basename ${IMAGE})/
else
opt_mountdir=1
[ ! -d ${MOUNTDIR} ] && error "Mount point ${MOUNTDIR} does not exist"
fi
# Check if default mount point exists
if [ ${opt_command} = umount ]; then
[ ! -d ${MOUNTDIR} ] && error "Default mount point ${MOUNTDIR} does not exist"
else
if [ ! -n "${opt_mountdir}" ] && [ -d ${MOUNTDIR} ]; then
error "Default mount point ${MOUNTDIR} already exists"
fi
fi
# Trap keyboard interrupt (ctrl-c)
trap ctrl_c SIGINT SIGTERM
# Check for dependencies
for c in dd losetup parted sfdisk partx mkfs.vfat mkfs.ext4 mountpoint rsync; do
command -v ${c} >/dev/null 2>&1 || error "Required program ${c} is not installed"
done
if [ -n "${opt_compress}" ] || [ ${opt_command} = gzip ]; then
for c in pv gzip; do
command -v ${c} >/dev/null 2>&1 || error "Required program ${c} is not installed"
done
fi
# Do the requested functionality
case ${opt_command} in
start)
trace "Starting SD Image backup process"
if [ ! -f ${IMAGE} ] && [ -n "$opt_create" ]; then
do_create
fi
do_mount
do_backup
do_showdf
do_umount
if [ -n "${opt_compress}" ]; then
do_compress
fi
trace "SD Image backup process completed."
if [ -n "$opt_log" ]; then
trace "See rsync log in $LOG"
fi
;;
mount)
if [ ! -f ${IMAGE} ] && [ -n "$opt_create" ]; then
do_create
fi
do_mount
trace "SD Image has been mounted and can be accessed at:\n ${MOUNTDIR}"
;;
umount)
do_umount
;;
check)
do_check
;;
gzip)
do_compress
;;
resize)
do_resize
;;
showdf)
do_mount
do_showdf
do_umount
;;
version)
version
;;
*)
error "Unknown command: ${opt_command}"
;;
esac
exit 0