forked from mrgrshift/shift-snapshot
-
Notifications
You must be signed in to change notification settings - Fork 1
/
snap.sh
executable file
·469 lines (388 loc) · 13.2 KB
/
snap.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
#!/bin/bash
VERSION="1.2"
# CONFIG
SHIFT_DIRECTORY=~/shift-lisk
TRUSTED_NODE="https://wallet.shiftnrg.org"
NETWORK="mainnet"
HTTP="http"
PORT="9305"
# EXPORT
export LC_ALL=en_US.UTF-8
export LANG=en_US.UTF-8
export LANGUAGE=en_US.UTF-8
#============================================================
#= snapshot.sh v0.2 created by mrgr =
#= Please consider voting for delegate mrgr =
#============================================================
#============================================================
#= snap.sh v1.2 created by Mx =
#= Please consider voting for delegate 'mx' =
#============================================================
# markdown
redTextOpen="\e[31m"
greenTextOpen="\e[1;32m"
boldTextOpen="\e[1m"
highlitedTextOpen="\e[44m"
colorTextClose="\e[0m"
echo " "
if [ ! -f ${SHIFT_DIRECTORY}/app.js ]; then
echo -e "${redTextOpen}Error: No shift-lisk installation detected in the directory ${SHIFT_DIRECTORY}${colorTextClose}"
echo -e "Please, change config: ${boldTextOpen}nano snap.sh${colorTextClose}"
echo "or install: https://github.com/ShiftNrg/shift-lisk"
exit 1
fi
if [ "\$USER" == "root" ]; then
echo -e "${redTextOpen}Error: shift-lisk should not be run be as root. Exiting.${colorTextClose}"
exit 1
fi
SHIFT_CONFIG=${SHIFT_DIRECTORY}/config.json
DB_NAME="$(grep "database" $SHIFT_CONFIG | cut -f 4 -d '"')"
DB_USER="$(grep "user" $SHIFT_CONFIG | cut -f 4 -d '"')"
DB_PASS="$(grep "password" $SHIFT_CONFIG | cut -f 4 -d '"' | head -1)"
SNAPSHOT_COUNTER=snapshot/counter.json
SNAPSHOT_LOG=snapshot/snapshot.log
if [ ! -f "snapshot/counter.json" ]; then
mkdir -p snapshot
sudo chmod +x snap.sh
echo "0" > $SNAPSHOT_COUNTER
sudo chown postgres:${USER:=$(/usr/bin/id -run)} snapshot
sudo chmod -R 777 snapshot
fi
SNAPSHOT_DIRECTORY=snapshot/
SHIFT_SNAPSHOT_NAME="blockchain.db.gz"
IP="127.0.0.1"
blockHeight="0"
NOW=$(date +"%d-%m-%Y - %T")
################################################################################
# intercept user input function
ctrlc_count=0
no_ctrlc()
{
let ctrlc_count++
echo
if [[ $ctrlc_count == 1 ]]; then
echo -e "${redTextOpen}!Warning. At shutdown, errors in the database are possible.${colorTextClose}"
# echo "If you really want to exit, press Ctrl+C again."
else
echo -e "${redTextOpen}Exit.${colorTextClose}"
exit
fi
}
# progress bar
sp=",.·•oὀ0*' "
sp1="/-\|+"
i_pb=1
progress_bar() {
while ps -p $2 >/dev/null 2>&1
do
printf "\b${1:i_pb++%${#1}:1}"
sleep 0.1
done
}
stopNode() {
bash ${SHIFT_DIRECTORY}/shift_manager.bash stop
}
startNode() {
bash ${SHIFT_DIRECTORY}/shift_manager.bash start
}
rebuildNode() {
echo "n" | bash ${SHIFT_DIRECTORY}/shift_manager.bash rebuild
}
getNodeStatus() {
response=$(curl --connect-timeout 2 --fail -s $HTTP://$IP:$PORT/api/loader/status/sync)
height=$(echo $response | jq '.height')
syncing=$(echo $response | jq '.syncing')
consensus=$(echo $response | jq '.consensus')
tResponse=$(curl --connect-timeout 2 --fail -s $TRUSTED_NODE/api/loader/status/sync)
tHeight=$(echo $tResponse | jq '.height')
printf "\r${boldTextOpen}BLOCKCHAIN:${colorTextClose} $tHeight ${boldTextOpen}HEIGHT:${colorTextClose} $height ${boldTextOpen}CONSENSUS:${colorTextClose} ${consensus}%% ${boldTextOpen}SYNCING:${colorTextClose} ${syncing} "
sleep 1
}
snapshotStatusCheck() {
echo -e "\n${boldTextOpen}Please wait for blockchain synchronization:${colorTextClose}"
syncing="true"
height="0"
tHeight="2"
while [[ "$syncing" = "true" ]] || (( "$height"+2 < "$tHeight" ))
do
getNodeStatus
# check $syncing after 5 min, if "false" -> break the loop
# synced
if (( "$height"+2 >= "$tHeight" )) ; then
nodeIsSynced="true"
# to be sure
for (( a = 0; a < 5; a++ ))
do
getNodeStatus
done
if [[ $startVerified = "true" ]]; then
echo -e "\n\n${boldTextOpen}$NOW -- $SHIFT_DIRECTORY"/$SHIFT_SNAPSHOT_NAME" - verified.${colorTextClose}" | tee -a $SNAPSHOT_LOG
echo -e "${greenTextOpen}+ Snapshot verified! Height:$blockHeight Size: $myFileSizeCheck ${colorTextClose}" | tee -a $SNAPSHOT_LOG
fi
break
fi
done
}
nodeStatusCheck() {
nodeOkay="false"
getNodeStatus
if (( "$height"+2 >= "$tHeight" )) ; then
nodeOkay="true"
echo -e "\n${greenTextOpen}+ Node is fine${colorTextClose}\n"
else
echo -e "\n${redTextOpen}Node is not synchronized with the blockchain. Can't create a good snapshot.${colorTextClose}"
echo "Trying to wait for synchronization for 60 seconds"
for (( a = 0; a < 60; a++ ))
do
getNodeStatus
if (( "$height"+2 >= "$tHeight" )) ; then
nodeOkay="true"
echo -e "\n${greenTextOpen}+ Node is fine${colorTextClose}\n"
break
fi
done
if (( "$height"+2 <= "$tHeight" )) ; then
echo -e "\n${redTextOpen}Node is not synchronized with the blockchain. Try again later or rebuild your node.${colorTextClose}"
exit
fi
fi
}
blockHeightPrettify() {
for (( i=0; $i<${#1}; i++ )) # ${#1} is first parameter
do
prettyBlockHeight=${prettyBlockHeight}${1:$i:1} # ${1 is first parameter
# add commas
if (( $i == 0 )) || (( $i == 3 )) ; then
prettyBlockHeight=${prettyBlockHeight}","
fi
done
echo "$prettyBlockHeight"
}
uploadToGitHub() {
fileLocation=$SHIFT_DIRECTORY"/$SHIFT_SNAPSHOT_NAME"
todayDate=$(date +"%d-%m-%Y")
tag="$NETWORK"
blockHeight=$(blockHeightPrettify $blockHeight)
titleLine="$todayDate verified shift-lisk snapshot up to block $blockHeight"
githubLink=$(git config --get remote.origin.url | cut -d '.' -f 1,2)
textLine1="<p>You can use this blockchain snapshot to rebuild your node up to block <b>$blockHeight</b></p>"
textLine2="<pre><code>cd shift-lisk</code><br/>"
textLine3="<code>sudo rm -f blockchain.db.gz</code><br/>"
textLine4="<code>wget $githubLink/releases/download/$tag/$SHIFT_SNAPSHOT_NAME</code><br/>"
textLine5="<code>echo "n" | ./shift_manager.bash rebuild</code></pre><br/>"
textLine6="<blockquote>Thanks to <a href="https://github.com/Bx64">BFX</a> for a first version of this instruction.</blockquote>"
# delete old today's release if exists
gh release delete $tag --yes
# create a GitHub release and upload a verified snapshot
gh release create $tag "$fileLocation" -n "${textLine1}${textLine2}${textLine3}${textLine4}${textLine5}${textLine6}" -t "$titleLine"
}
start_test() {
echo -e "Test started\n"
}
create_snapshot() {
# retrieve parameter of compression and other values
case $1 in
"1")
dbComp="1"
;;
"2")
dbComp="2"
;;
"3")
dbComp="3"
;;
"4")
dbComp="4"
;;
"5")
dbComp="5"
;;
"6")
dbComp="6"
;;
"7")
dbComp="7"
;;
"8")
dbComp="8"
;;
"9")
dbComp="9"
;;
"--best")
dbComp="9"
;;
"--fast")
dbComp="1"
;;
"-v")
dbComp="9"
startVerified="true"
if [[ "$2" = "-y" ]] || [[ "$2" = "--yes" ]]; then
yeees="true"
fi
;;
"--verified")
dbComp="9"
startVerified="true"
if [[ "$2" = "-y" ]] || [[ "$2" = "--yes" ]]; then
yeees="true"
fi
;;
*)
# default
dbComp="1"
;;
esac
# height check
nodeStatusCheck
export PGPASSWORD=$DB_PASS
echo -e " ${boldTextOpen}+ Creating snapshot with compression: ${dbComp}${colorTextClose}"
echo "--------------------------------------------------"
snapshotName="shift_db$NOW.snapshot.sql.gz"
snapshotLocation="$SNAPSHOT_DIRECTORY'$snapshotName'"
trap no_ctrlc SIGINT # intercept user input
(sudo su postgres -c "pg_dump -Fp -Z ${dbComp} $DB_NAME > $snapshotLocation") & # to start progress bar
app_pid=$! # progress bar
progress_bar "$sp" "$app_pid" # progress bar
blockHeight=`psql -d $DB_NAME -U $DB_USER -h localhost -p 5432 -t -c "select height from blocks order by height desc limit 1;"`
dbSize=`psql -d $DB_NAME -U $DB_USER -h localhost -p 5432 -t -c "select pg_size_pretty(pg_database_size('$DB_NAME'));"`
trap -- SIGINT # release interception user input
if [ $? != 0 ] || (( ctrlc_count > "0" )); then
echo -e "\n${redTextOpen}X Failed to create compressed snapshot.${colorTextClose}" | tee -a $SNAPSHOT_LOG
startVerified="false"
sudo rm -f "$SNAPSHOT_DIRECTORY$snapshotName"
exit 1
else
myFileSizeCheck=$(du -h "$SNAPSHOT_DIRECTORY$snapshotName" | cut -f1)
echo -e "\n$NOW -- ${greenTextOpen}OK compressed snapshot created successfully${colorTextClose} at block$blockHeight ($myFileSizeCheck)." | tee -a $SNAPSHOT_LOG
fi
if [[ $startVerified = "true" ]]; then
echo -e "\n ${boldTextOpen}+ Verifying snapshot${colorTextClose}"
echo "--------------------------------------------------"
echo -e "${highlitedTextOpen}shift-lisk node will be stopped for rebuild${colorTextClose}"
echo -e "press ${boldTextOpen}Ctrl+C${colorTextClose} to cancel"
(sleep 5) & # to start progress bar
app_pid=$! # progress bar
progress_bar "$sp1" "$app_pid" # progress bar
# rename
sudo mv $SNAPSHOT_DIRECTORY"$snapshotName" $SNAPSHOT_DIRECTORY"$SHIFT_SNAPSHOT_NAME"
# delete old
sudo rm -f $SHIFT_DIRECTORY"/$SHIFT_SNAPSHOT_NAME"
# move new
sudo mv $SNAPSHOT_DIRECTORY"$SHIFT_SNAPSHOT_NAME" ${SHIFT_DIRECTORY}"/"
rebuildNode
# pause to start node synchronization
(sleep 5) & # to start progress bar
app_pid=$! # progress bar
progress_bar "$sp1" "$app_pid" # progress bar
snapshotStatusCheck
# upload to GitHub
if [[ "$nodeIsSynced" = "true" ]] ; then
if [[ "$yeees" != "true" ]]; then
read -p "$(echo -e ${highlitedTextOpen}"Upload it to your GitHub repository (y/n)?"${colorTextClose}) " -r
if [[ ! $REPLY =~ ^[Yyнд]$ ]]; then
echo "Exit."
exit 1
fi
fi
uploadToGitHub
fi
fi
}
restore_snapshot(){
echo -e " ${boldTextOpen}+ Restoring snapshot${colorTextClose}"
echo "--------------------------------------------------"
SNAPSHOT_FILE=`ls -t snapshot/shift_db* | head -1`
if [ -z "$SNAPSHOT_FILE" ]; then
echo -e "${redTextOpen}!No snapshot to restore, please consider create it first${colorTextClose}"
echo -e "Using: bash shift-snapshot.sh create"
echo " "
exit 1
fi
echo -e "Snapshot to restore = $SNAPSHOT_FILE"
read -p "$(echo -e ${highlitedTextOpen}"shift-lisk node will be stopped, are you ready (y/n)?"${colorTextClose}) " -r
if [[ ! $REPLY =~ ^[Yyнд]$ ]]
then
echo -e "${redTextOpen}!Please stop app.js first. Then execute restore again${colorTextClose}"
echo " "
exit 1
fi
stopNode
trap no_ctrlc SIGINT # intercept user input
# snapshot restoring
export PGPASSWORD=$DB_PASS
# drop db
resp=$(sudo -u postgres dropdb --if-exists "$DB_NAME" 2> /dev/null)
resp=$(sudo -u postgres createdb -O "$DB_USER" "$DB_NAME" 2> /dev/null)
resp=$(sudo -u postgres psql -t -c "SELECT count(*) FROM pg_database where datname='$DB_NAME'" 2> /dev/null)
if [[ $resp -eq 1 ]]; then
echo "√ Database reset successfully."
else
echo "X Failed to create Postgresql database."
exit 1
fi
echo -e "\n${boldTextOpen}Snapshot restoring started${colorTextClose}"
echo "Please keep calm and don't push the button :)"
# restore dump
(gunzip -fcq "$SNAPSHOT_FILE" | psql -d $DB_NAME -U $DB_USER -h localhost -q &> /dev/null) & # to start progress bar
app_pid=$! # progress bar
progress_bar "$sp1" "$app_pid" # progress bar
trap -- SIGINT # release interception user input
if [ $? != 0 ] || (( ctrlc_count > "0" )); then
echo -e "${redTextOpen}X Failed to restore. Please rebuild your shift-lisk node.${colorTextClose}"
startNode
exit 1
else
startNode
snapshotStatusCheck
if [[ "$nodeIsSynced" = "true" ]] ; then
echo -e "\n${greenTextOpen}OK snapshot restored successfully.${colorTextClose}"
else
echo -e "${redTextOpen}X Snapshot restored, but failed to sync with the blockchain.${colorTextClose}"
exit 1
fi
fi
}
show_log(){
echo " + Snapshot Log"
echo "--------------------------------------------------"
cat snapshot/snapshot.log
echo "--------------------------------------------------END"
}
################################################################################
case $1 in
"create")
create_snapshot $2 $3
;;
"restore")
restore_snapshot
;;
"log")
show_log
;;
"hello")
echo "Hello my friend - $NOW"
;;
"test")
start_test $2 $3
;;
"help")
echo "Available commands are: "
echo " create Create a new snapshot with compression level of 1"
echo " create [1-9] Create a new snapshot with level of compression from 1 to 9"
echo " create --best Create a new snapshot with high level of compression (9)"
echo " create -v"
echo " create --verified Create a new snapshot with high level of compression then verify it"
echo " create -v -y"
echo " create -v --yes Create a verified snapshot and upload it to GitHub repo release section"
echo " restore Restore the last snapshot available in folder snapshot/"
echo " log Display log"
;;
*)
echo "Error: Unrecognized command."
echo ""
echo "Available commands are: create [1-9] -v -y, restore, log, help"
echo "Try: bash snap.sh help"
;;
esac