-
Notifications
You must be signed in to change notification settings - Fork 119
/
Copy pathdatabase_backup.sh
executable file
·102 lines (91 loc) · 4 KB
/
database_backup.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
#!/bin/bash
# Redirect all output to syslog (but skip this if we are on Windows).
# https://www.urbanautomaton.com/blog/2014/09/09/redirecting-bash-script-output-to-syslog/
if uname -a | grep -v MINGW64 >/dev/null 2>&1; then
exec 1> >(logger -s -t $(basename $0)) 2>&1
fi
# Get the directory of this script:
# https://stackoverflow.com/questions/59895/getting-the-source-directory-of-a-bash-script-from-within
DIR="$( cd "$( dirname "${BASH_SOURCE[0]}" )" >/dev/null 2>&1 && pwd )"
# Get the name of the repository:
# https://stackoverflow.com/questions/23162299/how-to-get-the-last-part-of-dirname-in-bash/23162553
REPO_NAME="$(basename "$DIR")"
# Configuration
BACKUPS_DIR="$DIR/backups"
FILENAME=$REPO_NAME-$(date +%s).sql # "date +%s" returns the epoch timestamp.
# Import the database information.
ENV_PATH="$DIR/.env"
if [[ ! -f $ENV_PATH ]]; then
echo "Failed to find the \".env\" file at: $ENV_PATH"
exit 1
fi
source "$ENV_PATH"
if [[ -z ${DB_HOST-} ]]; then
DB_HOST=localhost
fi
if [[ -z ${DB_PORT-} ]]; then
DB_PORT=5432
fi
# Back up the database and gzip it.
mkdir -p "$BACKUPS_DIR"
echo "Dumping the database..."
PGPASSWORD="$DB_PASSWORD" pg_dump --host="$DB_HOST" --port="$DB_PORT" --username="$DB_USER" --dbname="$DB_NAME" > "$BACKUPS_DIR/$FILENAME"
if [[ $? -ne 0 ]]; then
exit 1
fi
echo "Zipping the backup..."
gzip "$BACKUPS_DIR/$FILENAME"
# Delete old backups if the hard drive is getting full (but skip this if we are on Windows).
function delete_file_if_near_full_local {
AMOUNT_FULL=$(df "$DIR" | tail -1 | awk '{print $5}' | rev | cut -c 2- | rev)
echo "Local hard drive amount full: $AMOUNT_FULL"
if [[ $AMOUNT_FULL -gt 75 ]]; then
# Delete the oldest file in the backups directory.
OLDEST_FILE=$(ls -t "$BACKUPS_DIR" | tail -1)
rm -f "$BACKUPS_DIR/$OLDEST_FILE"
echo "Hard drive over 80% full; deleted the oldest backup: $OLDEST_FILE"
delete_file_if_near_full_local
fi
}
if uname -a | grep -v MINGW64 >/dev/null 2>&1; then
delete_file_if_near_full_local
fi
# Detect to see if we have Google Drive backups configured (see "install.md").
if [[ -z ${GOOGLE_DRIVE_SERVICE_ACCOUNT_FILENAME-} ]]; then
echo "Skipping upload to Google Drive since \"GOOGLE_DRIVE_SERVICE_ACCOUNT_FILENAME\" is not set in the \".env\" file."
exit 0
fi
if [[ -z ${GOOGLE_DRIVE_PARENT_DIRECTORY_ID-} ]]; then
echo "Skipping upload to Google Drive since \"GOOGLE_DRIVE_PARENT_DIRECTORY_ID\" is not set in the \".env\" file."
exit 0
fi
if command -v "gdrive" > /dev/null; then
GDRIVE_PATH="gdrive"
else
GDRIVE_ROOT_PATH="/root/go/bin/gdrive"
if [[ -f $GDRIVE_ROOT_PATH ]]; then
GDRIVE_PATH=$GDRIVE_ROOT_PATH
fi
fi
if [[ -z ${GDRIVE_PATH-} ]]; then
echo "Skipping upload to Google Drive since the \"gdrive\" binary is not found."
exit 1
fi
# Check for free space; if we have less than a gig left, then start deleting old files until we have
# at least a gig of storage left.
function delete_file_if_near_full_gdrive {
AMOUNT_FULL=$($GDRIVE_PATH about --service-account "$GOOGLE_DRIVE_SERVICE_ACCOUNT_FILENAME" | grep Free)
echo "GDrive amount full: $AMOUNT_FULL"
if [[ $(echo $AMOUNT_FULL | grep GB) ]]; then
return
fi
# We need to filter out directories, so we make sure to include the file extension.
# https://developers.google.com/drive/api/guides/search-shareddrives
OLDEST_FILE_ID=$($GDRIVE_PATH list --service-account "$GOOGLE_DRIVE_SERVICE_ACCOUNT_FILENAME" --no-header --max 9999 --order "createdTime" --query "trashed = false and 'me' in owners and name contains '$REPO_NAME' and name contains '.sql.gz'" | head -n 1 | cut -f 1 -d " ")
$GDRIVE_PATH delete --service-account "$GOOGLE_DRIVE_SERVICE_ACCOUNT_FILENAME" "$OLDEST_FILE_ID"
echo "Google Drive account has under 1 gig of free space left; deleted the oldest backup: $OLDEST_FILE_ID"
delete_file_if_near_full_gdrive
}
delete_file_if_near_full_gdrive
# Upload the file to Google Drive.
$GDRIVE_PATH upload "$BACKUPS_DIR/$FILENAME.gz" --service-account "$GOOGLE_DRIVE_SERVICE_ACCOUNT_FILENAME" --parent "$GOOGLE_DRIVE_PARENT_DIRECTORY_ID"