chiark / gitweb /
67716bbce66671c7e50d022d8dfeeeb29778fc1f
[chiark-utils.git] / backup / snaprsync
1 #!/bin/bash
2 #
3 # usage: snaprsync <setting>... <positionals>
4 #  <setting> is ---<name>=<value>
5 #  <positionals> are assigned to unused mandatory values in order
6 # mandatory:
7 #   rhost device mountpoint localarea 
8 # optional:
9         localprevious=
10         snapkind=lvm
11         rsharedir=/usr/share/chiark-backup 
12         retcdir=/etc/chiark-backup
13         rvardir=/var/lib/chiark-backup
14         bwlimit=
15
16 set -e
17
18 badusage () { echo >&2 "snaprsync: bad usage: $1"; exit 12; }
19 x () { echo "+ $@"; "$@"; }
20 xspawned () { eval "${1}pid=$!; echo \"+[$!] ($1) &\";"; }
21 xwait () { eval "echo \"+[\$${1}pid] ($1)...\"; wait \$${1}pid;"; }
22
23 while true; do
24         case "$1" in
25         --?*=?*)
26                 name=${1#--}; name=${name%%=*}
27                 value=${1#--*=}
28                 case "$name" in
29                 rhost|device|mountpoint|localarea);;
30                 localprevious|rsharedir|retcdir|rvardir|bwlimit);;
31                 *) badusage "unknown setting $name";;
32                 esac
33                 eval "$name=\$value"
34                 ;;
35         --)     shift; break ;;
36         -*)     badusage "unknown option $1" ;;
37         *)      break ;;
38         esac
39         shift
40 done
41
42 for name in rhost device mountpoint localarea; do
43         eval "value=\$$name"
44         if [ "x$value" != x ]; then continue; fi
45         if [ $# = 0 ]; then badusage "no value for setting $name"; fi
46         eval "$name=$1"
47         shift
48 done
49
50 datefmt='%Y-%m-%d %H:%M:%S Z'
51 rsync="rsync ${bwlimit:+--bwlimit} $bwlimit"
52 export RSYNC_RSH='ssh -o compression=no'
53
54 ssh $rhost "date -u '+$rhost $datefmt start'"
55 ssh $rhost id
56 ssh $rhost ls -d $rsharedir
57 ssh $rhost ls -d $rvardir
58
59 test -d $localarea || x mkdir $localarea
60 ournode=`uname -n`
61 rsumsfile=for-$ournode.sums
62 summer='summer -ACDbtqf'
63
64 td=/dev/enoent
65 rc=12
66 trap 'rm -rf $td; exit $rc' 0
67 td=`mktemp -td`
68
69 mkfifo -m 600 $td/sentinel
70 exec 4<>$td/sentinel
71
72 x ssh $rhost "$rsharedir/snap-drop"
73 x ssh $rhost "$retcdir/snap/$snapkind snap $rvardir $device $mountpoint"
74 ssh $rhost <$td/sentinel 4<&- "
75   set -e
76   date -u '+$rhost $datefmt main'
77   exec 3<&0 0</dev/null
78   (set +e; read x <&3; kill 0) &
79   cd $rvardir
80   umask 077
81   exec 3>$rsumsfile
82   cd snap-mount
83   $summer . | sort -t'  ' -k2 >&3
84   test x\"\${PIPESTATUS[*]}\" = 'x0 0'
85   date -u '+$rhost $datefmt sumsdone'
86   cd ..
87 " &
88 xspawned rsum
89 x $rsync -aHSxz --numeric-ids --delete \
90         ${localprevious:+--link-dest} $localprevious \
91         $rhost:$rvardir/snap-mount/. $localarea/.
92 date -u "+ $datefmt rsyncdone"
93
94 exec 3>$localarea,lsums
95 (cd $localarea && \
96  $summer . | sort -t'   ' -k2 >&3 && \
97  test x"${PIPESTATUS[*]}" = 'x0 0') &
98 xspawned lsum
99 exec 3>&-
100
101 xwait rsum
102 exec 4<&-
103 date -u "+ $datefmt sumsdone"
104 x ssh $rhost "$rsharedir/snap-drop"
105
106 if [ "x${localprevious}" != x ]; then
107         cp "$localprevious,rsums" "$localarea,rsums"
108 fi
109 x $rsync -p \
110         $rhost:$rvardir/$rsumsfile \
111         "$localarea,rsums"
112
113 xwait $lsum
114 date -u "+ $datefmt checking"
115
116 set +e
117 x diff -u "$localarea,rsums" "$localarea,lsums" >"$localarea,sumsdiff"
118 diffrc=$?
119 set -e
120 test $diffrc = 0 || test $diffrc = 1
121
122 date -u "+ $datefmt checked $diffrc"
123 rc=$diffrc