-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathwsbkp.sh
More file actions
executable file
·308 lines (291 loc) · 9.12 KB
/
wsbkp.sh
File metadata and controls
executable file
·308 lines (291 loc) · 9.12 KB
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
#!/bin/sh
################################################################################
# Wake/Sleep Backup - 'wakes up' usb drive, rsyncs and powers off #
################################################################################
# Example:
# DST_UUID="abc12345-1234-1234-1234-aaaabbbbcccc" (see 'blkid' command)
# BKP_DIRS="/etc/ /root/ /home/ /opt/ /usr/local/bin/"
DST_UUID="" # uuid of backup device
DST_MNT="/mnt/bkp" # dst/target path
BKP_DIRS="/" # dirs to backup
EXCL_DIRS="
--exclude=${DST_MNT}/
--exclude=/proc/
--exclude=/sys/
--exclude=/lost+found/
--exclude=/mnt/
--exclude=/run/
--exclude=/tmp/
--exclude=/var/lib/docker/
"
RSYNC="rsync -aA --info=flist0,name0,progress2,stats2 --progress --stats"
LOG="/var/log/wsbkp.log"
OUT="/dev/null" # /dev/null or /dev/stdout (show rsync output)
FORCE=0 # force even if dev not found
# END OF CONFIG ################################################################
# Helper function to check if device exists before sleep/mount/list
func_dev() {
# DST_DEV="$( blkid | grep "$DST_UUID" | cut -d: -f1 )"
# if [ "$DST_DEV" = "" ] && [ "$FORCE" -eq 0 ]; then
# echo "! ERROR: Device \"$DST_UUID\" not found, exiting..."
# exit 1
# fi
DST_DEV="$(blkid -U "$DST_UUID")" || {
if [ "$FORCE" -eq 0 ]; then
printf "! ERROR: Device \"%s\" not found, exiting...\n" "$DST_UUID"
exit 1
fi
}
}
func_wake() {
hdparm -C "/dev/disk/by-uuid/$DST_UUID" || {
printf "! ERROR: Drive with UUID %s did not wake up, exiting...\n" "$DST_UUID"
exit 1
}
}
func_sleep() {
func_dev && if findmnt -n "$DST_DEV" | grep -q " $DST_DEV"; then
printf "> WARNING: \"%s\" is mounted, umount it first\n" "$DST_DEV"
else
hdparm -Y "$DST_DEV"
fi
}
# Verify drive and mount
func_mnt() {
# in case var DST_MNT is empty generate it using device name (/mnt/sdx)
if [ -z "$DST_MNT" ]; then
func_dev && {
TMP="$(basename "$DST_DEV")"
if echo "$TMP" | grep -q "^sd[a-z]"; then
DST_MNT="/mnt/${TMP}"
fi
}
fi
if [ ! -d "$DST_MNT" ]; then
mkdir "$DST_MNT"
fi
func_dev &&
mount "$DST_DEV" "$DST_MNT" &&
printf -- "- INFO: Mounted \"%s\" \"%s\"\n" "$DST_DEV" "$DST_MNT"
}
func_umnt() {
umount "$DST_MNT" ||
{
printf "! ERROR: Could not umount \"%s\", exiting...\n" "$DST_MNT"
exit 1
}
}
# Power on using usb host controller interface
# ohci-pci: usb 1.1 (uhci)
# ehci-pci: usb 1.1 2.0
# xhci_hcd: usb 1.0 2.0 3.0
func_power_on() {
#shopt -s nullglob
for i in /sys/bus/pci/drivers/?hci[_-][hp]c[di]; do
cd "$i" || {
printf "! ERROR: Failed to change host controller directory to %s\n" "$i"
exit 1
}
printf -- "- INFO: Resetting devices from %s...\n" "$i"
for j in ????:??:??.?; do
printf "%s" "$j" >unbind
printf "%s" "$j" >bind
sleep 1
done
if echo "$i" | grep -q "xhci_hcd"; then
sleep 20
fi
done
}
func_power_off() {
func_dev && {
command -v udisksctl >/dev/null 2>&1 && {
udisksctl power-off --no-user-interaction -b "$DST_DEV" || {
echo "> WARNING: udiskctl did not run successfully"
exit 1
}
} || {
echo "! ERROR: udisksctl not found"
exit 1
}
}
}
# Function to list devices using multiple methods, shows state of backup drive if it exists
func_list() {
FORCE=1
func_dev
printf "\nDST_UUID=\"%s\" DST_DEV=\"%s\"\n\n" "$DST_UUID" "$DST_DEV"
echo "fdisk :"
fdisk -l 2>/dev/null | grep "Disk /dev/sd[b-z]"
echo
echo "blkid :"
blkid | grep -Ev "/dev/(sda[0-9]|loop|mapper)|swap|LUKS"
echo
echo "findmnt :"
findmnt -nr |
grep -Ev '/(sda|pts)|^/(proc|run|sys|boot|jail)|tmpfs|overlay|fuseblk|mqueue|hugepages' |
grep -E "$DST_DEV|$DST_MNT"
echo
echo "/sys/devices :"
find /sys/devices -name block -regex '.*usb.*' -exec sh -c '
printf "%s: %s\n" "$1" "$(ls "$1")"
printf "%s: %s\n" "$(echo "$1"/../model|sed s@"$1"/../@\ @)" "$(cat "$1"/../model 2>/dev/null )"' sh "{}" \; |
sed -e 's|/sys/devices/pci|/pci|g' -e 's/0000:00[/:]\?//g'
echo
echo "/sys/block :"
for i in /sys/block/sd[b-z]; do
printf "%s device/state: %s power/control: %s\n" \
"$i" "$(cat "$i"/device/state 2>/dev/null)" "$(cat "$i"/power/control 2>/dev/null)"
done
echo
echo "/sys/bus/pci/drivers :"
find /sys/bus/pci/drivers/?hci[_-][hp]c[di]/????:??:??.? -printf "%f\n" | sed -e 's|0000:00:| |g'
echo
}
func_help() {
printf "\nWake/Sleep Backup\n"
printf -- "-----------------\n\n"
printf "SYNOPSIS: wake up usb drive, rsync, power off\n\n"
printf "USAGE: %s -[o|p][w|s][m|u][l][f] [dirs to backup]\n\n" "$(basename "$0")"
printf "OPTIONS: [-p]|[-o] power on | power off\n"
printf "\t [-w]|[-s] wakeup | sleep\n"
printf "\t [-m]|[-u] mount | umount\n"
printf "\t [-l] list drive info\n"
printf "\t [-f] force\n"
printf "\t [dirs to backup] overwrites setting in script\n\n"
}
# 'Shift' arg
func_args() {
ARGS="$(echo "$ARGS" | sed -r -- "s/ ?-${1} ?//g")"
}
# handle arguments
ARGS="$*"
if printf -- "%s" "$ARGS" | grep -q -- '\-f'; then
func_args "f"
FORCE=1
fi
if printf -- "%s" "$ARGS" | grep -q -- '\-n'; then
func_args "n"
RS_ARGS="$RS_ARGS -n"
fi
if printf -- "%s" "$ARGS" | grep -q -- '\-v'; then
func_args "v"
RS_ARGS="$RS_ARGS -v"
fi
#echo "DEBUG: ARGS=$ARGS"
#exit
if printf -- "%s" "$ARGS" | grep -q -- '\-h'; then
func_help
exit
elif printf -- "%s" "$ARGS" | grep -q -- '\-s'; then
func_sleep
exit
elif printf -- "%s" "$ARGS" | grep -q -- '\-w'; then
func_wake
exit
elif printf -- "%s" "$ARGS" | grep -q -- '\-m'; then
func_mnt
exit
elif printf -- "%s" "$ARGS" | grep -q -- '\-u'; then
func_umnt
exit
elif printf -- "%s" "$ARGS" | grep -q -- '\-o'; then
func_power_off
exit
elif printf -- "%s" "$ARGS" | grep -q -- '\-p'; then
func_power_on
exit
elif printf -- "%s" "$ARGS" | grep -q -- '\-l'; then
func_list
exit
elif printf -- "%s" "$ARGS" | grep -q -- '\-[^fnv]'; then
func_help
exit
elif [ "$ARGS" != "" ]; then
BKP_DIRS="$*"
fi
#
# NOTES:
#
# hdparm: -S 120 sets standby timeout (10min), to check: -C
# smart: 'smartctl -i -n standby'
# time: 'time rsync <...>' for summary (bashism)
# rsync: [-i] itemize changes [-v] erbose [-q] uiet [-n] dryrun
# info: udiskctl info -b /dev/sdX, udevadm info -p /sys/block/sdX
# rm, add: udiskctl poweroff, bind/unbind ?hci-pci
# echo 1 > /sys/block/sdX/device/delete, echo 1 > sdX/device/rescan
# see http://billauer.co.il/blog/2013/02/usb-reset-ehci-uhci-linux
# rescan: rescan-scsi-bus.sh
#
# Main
echo
blkid -U "$DST_UUID" >/dev/null || {
printf "[%s] Powering on drive (UUID=%s)...\n" "$(date +%F\ %T)" "$DST_UUID"
func_power_on
}
func_dev && { printf "[%s] Using backup device \"%s\" (UUID=%s)\n" "$(date +%F\ %T)" "$DST_DEV" "$DST_UUID"; }
if [ ! -d "$DST_MNT" ]; then
mkdir "$DST_MNT" || {
printf "! ERROR: Could not create %s\n" "$DST_MNT"
exit 1
}
fi
func_mnt || {
printf "\n! ERROR: Could not mount \"%s\" on \"%s\", exiting...\n" "$DST_DEV" "$DST_MNT"
exit 1
}
echo
if [ -d "$DST_MNT" ]; then
if findmnt -n "$DST_MNT" | grep -q " $DST_DEV"; then
status=-1
printf "* Command: \"%s\"\n" "$RSYNC $RS_ARGS"
printf "* Dir(s): \"%s\"\n" "$BKP_DIRS"
printf "* Target drive: \"%s\" mounted on \"%s\" (UUID=%s)\n" "$DST_DEV" "$DST_MNT" "$DST_UUID"
printf "* Exclude dir(s): %s\n\n" "$(echo "$EXCL_DIRS" | sed -- 's/--exclude=//g')"
# rsync dirs and poweroff if loop returns 0 (warn if it fails)
(
for i in $BKP_DIRS; do
if [ -d "$i" ]; then
MSG="sleeping 10s... CTRL-C TO ABORT"
if ! echo "$i" | grep -q '^/'; then
printf "> WARNING: \"$i\" has no leading slash, %s\n\n" "$MSG"
sleep 10
fi
if ! echo "$i" | grep -q '/$'; then
printf "> WARNING: \"$i\" has no trailing slash, %s\n\n" "$MSG"
sleep 10
fi
printf "[%s] Backup SOURCE DIR: \"%s\" to DESTINATION: \"%s\"\n" "$(date +%F\ %T)" "$i" "${DST_MNT}${i}" |
tee -a "$LOG"
if [ ! -d "${DST_MNT}${i}" ]; then
printf "+ Creating %s\n" "${DST_MNT}${i}"
mkdir -p "${DST_MNT}${i}" || printf "! ERROR: Could not create %s\n" "${DST_MNT}${i}"
fi
$RSYNC $RS_ARGS $EXCL_DIRS "$i" "${DST_MNT}${i}" >"$OUT" 2>&1
else
printf "! ERROR: \"%s\" does not exist\n" "$i"
fi
echo
done
) && status=0 || status=1
if [ "$status" -eq 0 ]; then
printf "* Syncing complete, listing \"%s\"\n%s\n\n" "$DST_MNT" "$(ls -la "$DST_MNT")"
printf "[%s] Done. Powering off...\n" "$(date +%F\ %T)" | tee -a "$LOG"
sleep 30 && func_umnt && func_power_off
else
printf"[%s] > WARNING: Check if drive is unmounted and powered down\n" "$(date +%F\ %T)" |
tee -a "$LOG"
fi
else
printf "! ERROR: mount \"%s\" not found, exiting\n" "$DST_MNT"
exit 1
fi
else
printf "ERROR: \"%s\" does not exists, exiting...\n" "$DST_MNT"
exit 1
fi
echo
if [ -d "$DST_MNT" ]; then
rmdir "$DST_MNT" || printf "NOTICE: could not remove dir \"%s\"\n" "$DST_DIR"
fi
exit 0