blob: 760d78eff9168e1d7ac1cfd0edecb997d54fd712 (
plain)
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
|
#!/bin/bash
#
# This is a simple backup script using duplicity. It's supposed to serve as a
# starting point and to be adjusted to your system.
#
# Important steps:
# - define a host "backup" in root's .ssh/config
# - read the script and adjust to your needs
set -e
main() {
if [[ $UID != 0 ]]; then
exec sudo "$0" "$@"
fi
TMPDIR="$(mktemp -d "/tmp/${0##*/}.XXXXXX")"
trap "rm -rf '${TMPDIR}'" EXIT TERM
# ensure duplicity keeps its cache at a central location
export HOME=/root
# if you want to encrypt the backups remove --no-encryption in the duplicity call
# and uncomment the lines that contain PASSPHRASE
#PASSPHRASE="randomstringhere"
# these mountpoints will be excluded
excludeMountpoints=(
/tmp/
/sys/
/dev/
/proc/
/run/
/mnt/levant/nfs/
/media/
)
# mountpoints of these types do not need to be excluded
fsWhitelist=(ext4 btrfs)
# first line that matches wins
IFS='' read -r -d '' excludeList <<EOF || true
+ /home/flo/.local/share/Steam/steamapps/common/Counter-Strike Global Offensive/csgo/cfg
- /home/*/.local/share/Steam/steamapps/common/*/*
- /home/*/.cache/*
- /home/*/.claws-mail/imapcache
- /root/.cache/*
- /var/cache/pacman/pkg/*
EOF
exclude_mountpoints
echo "$excludeList" > "$TMPDIR/exclude-list"
# save some data that's useful for restores
local backupDataDir=/root/backup-data/
mkdir -p "$backupDataDir"
fdisk -l > "$backupDataDir/fdisk"
vgdisplay > "$backupDataDir/vgdisplay"
pvdisplay > "$backupDataDir/pvdisplay"
lvdisplay > "$backupDataDir/lvdisplay"
lvdisplay > "$backupDataDir/lvdisplay"
df -a > "$backupDataDir/df"
findmnt -l > "$backupDataDir/findmnt"
# this does not ignore /proc and network mounts so it's not that useful :(
#find / | gzip > /root/full-file-list.txt.gz
local backupdir="$HOSTNAME-backup/full-backup"
backup / "sftp://backup/$backupdir/" --exclude-filelist "$TMPDIR/exclude-list"
ssh backup "touch $backupdir/last-backup-timestamp"
}
backup() {
local src=$1
local dest=$2
shift 2
local -a options=()
if [[ $(date +%u ) == '1' ]]; then
# try to only run full backups on monday (1)
options+=(--full-if-older-than 2D)
else
# force a full backup once in a while
options+=(--full-if-older-than 20D)
fi
#export PASSPHRASE
duplicity \
-v5 \
--numeric-owner \
--volsize 250 \
--allow-source-mismatch \
--asynchronous-upload \
--no-encryption \
"${options[@]}" "$@" "$src" "$dest"
duplicity --force remove-older-than 120D "$dest"
#export PASSPHRASE=""
}
### support functions below ###
##
# usage : in_array( $needle, $haystack )
# return : 0 - found
# 1 - not found
##
in_array() {
local needle=$1; shift
local item
for item in "$@"; do
[[ $item = "$needle" ]] && return 0 # Found
done
return 1 # Not Found
}
# same as in_array except 0 is returned if any item in haystack starts with needle
in_array_startswith() {
local needle=$1; shift
local item
for item in "$@"; do
[[ "$needle" == "$item"* ]] && return 0 # Found
done
return 1 # Not Found
}
exclude_mountpoints() {
local error=0
for fs in "${excludeMountpoints[@]}"; do
if [[ $fs != */ ]]; then
error=1
echo "Error: excludeMountpoints entry doesn't end with /: $fs" >&2
fi
excludeList+="- $fs*"$'\n'
done
while read line; do
local mountpoint=$(echo "$line" | cut -d\ -f2 | sed 's#\040# #g;')
local type=$(echo "$line" | cut -d\ -f3)
if ! in_array $type "${fsWhitelist[@]}"; then
if ! in_array_startswith "$mountpoint/" "${excludeMountpoints[@]}"; then
error=1
echo "Warning: mountpoint not excluded: $mountpoint" >&2
fi
fi
done </etc/mtab
if ((error)); then
exit 1
fi
}
main "$@"
|