-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathgit_backup
executable file
·210 lines (173 loc) · 3.48 KB
/
git_backup
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
#!/usr/bin/env bash
sync_base=$(cat $HOME/.sync 2> /dev/null)
tracker=$sync_base/.tracks
go_sync() {
cd "$sync_base"
}
go_back() {
cd -
}
check_install() {
if [ ! -d $sync_base/.git ]
then
go_sync
echo "Initializing global backup repo"
git init -q
go_back
fi
}
track_to_folder() {
grep "=$1$" $tracker | awk -F= '{ print $1 }'
}
track() {
if [ ! $# -eq 2 ]
then
echo "track <path> <track_name>"
exit 1
fi
local abs_folder=$(make_absolute "$1")
local name=$2
#check track name is not already in use
if [[ $name =~ "/" ]]
then
echo Track name cannot contains any slash
elif [[ -d $sync_base/$name ]]
then
echo "Have already a track name called : $name"
exit 1
else
#get absolute path before changing from working directory
go_sync
echo "$abs_folder=$name" >> $tracker
git add ".tracks"
mkdir "$name" && touch "$name/.gitkeep"
git add "$name/.gitkeep"
git commit -q -m "Track: $name"
#separate backup folder in branches, it should be easier to manipulate them
#individually after that.
git branch $name
go_back
fi
}
backup() {
if [ ! $# -eq 1 ]
then
echo "backup <track_name>"
exit 1
fi
local name=$1
local folder=$(track_to_folder $name)
if [[ -e $folder ]]
then
#a : archive
#r : recursive
#u : update
#c : skip on checksum and not mod time or size
#z : compress during transfer
#NOTE: need to experiment --delete option
local timestamp=$(date "+%Y-%m-%d %H:%M:%S")
go_sync
git checkout "$name"
rsync -arcuzv --delete "$folder/" "$sync_base/$name"
git add -A .
git commit -q -m "$timestamp"
git checkout master
go_back
else
echo "$folder does not exist on your system"
exit 1
fi
}
restore() {
if [ ! $# -eq 1 ]
then
echo "restore <track_name>"
exit 1
fi
local name=$1
local folder=$(track_to_folder $name)
[[ ! -e $folder ]] && mkdir -p "$folder"
#NOTE: fix this awful thing
go_sync
git checkout "$name"
go_back
rsync -arcuzv --delete "$sync_base/$name/" "$folder/"
go_sync
git checkout master
go_back
}
init() {
if [ $1 ]
then
sync_base=$1
else
sync_base=$PWD
fi
if [ ! -e $sync_base ]
then
mkdir -p $sync_base
fi
make_absolute $sync_base > $HOME/.sync
sync_base=$(cat $HOME/.sync)
check_install
}
list_tracks() {
while read entry
do
echo $entry | awk -F"=" '{ print $2 " \t=> " $1 }'
done < "$sync_base/.tracks"
}
# === utility functions ===
#As readlink is not available on all distro or OSX, provide a portable way to get an
#absolute path from a relative one.
make_absolute() {
local ret=false
local rel_path=$1
if [[ ! -z $rel_path ]]; then
#strip eventual last slash
rel_path=${rel_path%/}
if [ "$rel_path" = "" ]; then
rel_path="/"
fi
base_name=${rel_path##*/}
rel_dir=${rel_path%$base_name}
if [[ $rel_dir = "" ]]; then
rel_dir="."
fi
if cd -P "$rel_dir" 1>/dev/null; then
#we jump in the directory and use $PWD
if [[ $(pwd) = "/" ]]; then
echo "/$base_name"
else
echo "$(pwd)/$base_name"
fi
cd - 1>&2 > /dev/null #get back to OLD_PATH
ret=true
fi
fi
$ret
}
case $1 in
"init")
shift
init $@
;;
"backup")
shift
backup $@
;;
"restore")
shift
restore $@
;;
"track")
shift
track $@
;;
"tracks")
list_tracks
;;
*)
echo "Usage:"
;;
esac