
This patchset introduces the framework by which all OSH-based database systems can use to backup and restore their databases. The framework is refactored from the Postgresql backup and restore logic. This will prevent alot of code duplication in the backup restore scripts across each cluster. In the process, some improvements needed to be made: 1) Removing the need for 2 separate containers to do the backup and restore work to a remote gateway. This simplifies the design and enables a higher level of robustness. 2) Adding separate "days to keep" config value for remote backup files, as there may be different requirements for the remote files than the local backup files. 3) Adding capability to send Storage_Policy when creating the remote RGW swift container. 4) Making coding style improvement for readability and maintainability. 5) Fixing a deployment bug that occurs when remote backup is disabled. Change-Id: I3a3482ad67320e89f04305b17da79abf7ad6eb45
376 lines
12 KiB
Smarty
Executable File
376 lines
12 KiB
Smarty
Executable File
{{- define "helm-toolkit.scripts.db-backup-restore.restore_main" }}
|
|
#!/bin/bash
|
|
|
|
# This file contains a database restore framework which database scripts
|
|
# can use to perform a backup. The idea here is that the database-specific
|
|
# functions will be implemented by the various databases using this script
|
|
# (like mariadb, postgresql or etcd for example). The database-specific
|
|
# script will need to first "source" this file like this:
|
|
# source /tmp/restore_main.sh
|
|
#
|
|
# Then the script should call the main CLI function (cli_main):
|
|
# cli_main <arg_list>
|
|
# where:
|
|
# <arg_list> is the list of arguments given by the user
|
|
#
|
|
# The framework will require the following variables to be exported:
|
|
#
|
|
# export DB_NAMESPACE Namespace where the database(s) reside
|
|
# export DB_NAME Name of the database system
|
|
# export ARCHIVE_DIR Location where the backup tarballs should
|
|
# be stored. (full directory path which
|
|
# should already exist)
|
|
# export CONTAINER_NAME Name of the container on the RGW where
|
|
# the backups are stored.
|
|
# RGW access variables:
|
|
# export OS_REGION_NAME Name of the region the RGW resides in
|
|
# export OS_AUTH_URL Keystone URL associated with the RGW
|
|
# export OS_PROJECT_NAME Name of the project associated with the
|
|
# keystone user
|
|
# export OS_USERNAME Name of the keystone user
|
|
# export OS_PASSWORD Password of the keystone user
|
|
# export OS_USER_DOMAIN_NAME Keystone domain the project belongs to
|
|
# export OS_PROJECT_DOMAIN_NAME Keystone domain the user belongs to
|
|
# export OS_IDENTITY_API_VERSION Keystone API version to use
|
|
#
|
|
# The database-specific functions that need to be implemented are:
|
|
# get_databases
|
|
# where:
|
|
# <tmp_dir> is the full directory path where the decompressed
|
|
# database files reside
|
|
# <db_file> is the full path of the file to write the database
|
|
# names into, one database per line
|
|
# returns: 0 if no errors; 1 if any errors occurred
|
|
#
|
|
# This function is expected to extract the database names from the
|
|
# uncompressed database files found in the given "tmp_dir", which is
|
|
# the staging directory for database restore. The database names
|
|
# should be written to the given "db_file", one database name per
|
|
# line.
|
|
#
|
|
# restore_single_db
|
|
# where:
|
|
# <db_name> is the name of the database to be restored
|
|
# <tmp_dir> is the full directory path where the decompressed
|
|
# database files reside
|
|
# returns: 0 if no errors; 1 if any errors occurred
|
|
#
|
|
# This function is expected to restore the database given as "db_name"
|
|
# using the database files located in the "tmp_dir". The framework
|
|
# will delete the "tmp_dir" and the files in it after the restore is
|
|
# complete.
|
|
#
|
|
# restore_all_dbs
|
|
# where:
|
|
# <tmp_dir> is the full directory path where the decompressed
|
|
# database files reside
|
|
# returns: 0 if no errors; 1 if any errors occurred
|
|
#
|
|
# This function is expected to restore all of the databases which
|
|
# are backed up in the database files located in the "tmp_dir". The
|
|
# framework will delete the "tmp_dir" and the files in it after the
|
|
# restore is complete.
|
|
#
|
|
# The functions in this file will take care of:
|
|
# 1) The CLI parameter parsing for the arguments passed in by the user.
|
|
# 2) The listing of either local or remote archive files at the request
|
|
# of the user.
|
|
# 3) The retrieval/download of an archive file located either in the local
|
|
# file system or remotely stored on an RGW.
|
|
# 4) Calling either "restore_single_db" or "restore_all_dbs" when the user
|
|
# chooses to restore a database or all databases.
|
|
# 5) The framework will call "get_databases" when it needs a list of
|
|
# databases when the user requests a database list or when the user
|
|
# requests to restore a single database (to ensure it exists in the
|
|
# archive).
|
|
#
|
|
|
|
export LOG_FILE=/tmp/dbrestore.log
|
|
|
|
usage() {
|
|
ret_val=$1
|
|
echo "Usage:"
|
|
echo "Restore command options"
|
|
echo "============================="
|
|
echo "help"
|
|
echo "list_archives [remote]"
|
|
echo "list_databases <archive_filename> [remote]"
|
|
echo "restore <archive_filename> <db_specifier> [remote]"
|
|
echo " where <db_specifier> = <dbname> | ALL"
|
|
clean_and_exit $ret_val ""
|
|
}
|
|
|
|
#Exit cleanly with some message and return code
|
|
clean_and_exit() {
|
|
RETCODE=$1
|
|
MSG=$2
|
|
|
|
# Clean/remove temporary directories/files
|
|
rm -rf $TMP_DIR
|
|
rm -f $DB_FILE
|
|
|
|
if [[ "x${MSG}" != "x" ]]; then
|
|
echo $MSG
|
|
fi
|
|
exit $RETCODE
|
|
}
|
|
|
|
# Retrieve a list of archives from the RGW.
|
|
retrieve_remote_listing() {
|
|
RESULT=$(openstack container show $CONTAINER_NAME 2>&1)
|
|
if [[ $? -eq 0 ]]; then
|
|
# Get the list, ensureing that we only pick up the right kind of backups from the
|
|
# requested namespace
|
|
openstack object list $CONTAINER_NAME | grep $DB_NAME | grep $DB_NAMESPACE | awk '{print $2}' > $TMP_DIR/archive_list
|
|
if [[ $? -ne 0 ]]; then
|
|
echo "Container object listing could not be obtained."
|
|
return 1
|
|
else
|
|
echo "Archive listing successfully retrieved."
|
|
fi
|
|
else
|
|
echo $RESULT | grep "HTTP 401"
|
|
if [[ $? -eq 0 ]]; then
|
|
echo "Could not access the container: ${RESULT}"
|
|
return 1
|
|
else
|
|
echo $RESULT | grep "ConnectionError"
|
|
if [[ $? -eq 0 ]]; then
|
|
echo "Could not reach the RGW: ${RESULT}"
|
|
# In this case, keystone or the site/node may be temporarily down.
|
|
# Return slightly different error code so the calling code can retry
|
|
return 2
|
|
else
|
|
echo "Container $CONTAINER_NAME does not exist: ${RESULT}"
|
|
return 1
|
|
fi
|
|
fi
|
|
fi
|
|
return 0
|
|
}
|
|
|
|
# Retrieve a single archive from the RGW.
|
|
retrieve_remote_archive() {
|
|
ARCHIVE=$1
|
|
|
|
RESULT=$(openstack object save --file $TMP_DIR/$ARCHIVE $CONTAINER_NAME $ARCHIVE 2>&1)
|
|
if [[ $? -ne 0 ]]; then
|
|
echo $RESULT | grep "HTTP 401"
|
|
if [[ $? -eq 0 ]]; then
|
|
echo "Could not access the archive: ${RESULT}"
|
|
return 1
|
|
else
|
|
echo $RESULT | grep "ConnectionError"
|
|
if [[ $? -eq 0 ]]; then
|
|
echo "Could not reach the RGW: ${RESULT}"
|
|
# In this case, keystone or the site/node may be temporarily down.
|
|
# Return slightly different error code so the calling code can retry
|
|
return 2
|
|
else
|
|
echo "Archive ${ARCHIVE} could not be retrieved: ${RESULT}"
|
|
return 1
|
|
fi
|
|
fi
|
|
else
|
|
echo "Archive $ARCHIVE successfully retrieved."
|
|
fi
|
|
return 0
|
|
}
|
|
|
|
# Display all archives
|
|
list_archives() {
|
|
REMOTE=$1
|
|
|
|
if [[ "x${REMOTE^^}" == "xREMOTE" ]]; then
|
|
retrieve_remote_listing
|
|
if [[ $? -eq 0 && -e $TMP_DIR/archive_list ]]; then
|
|
echo
|
|
echo "All Archives from RGW Data Store"
|
|
echo "=============================================="
|
|
cat $TMP_DIR/archive_list
|
|
clean_and_exit 0 ""
|
|
else
|
|
clean_and_exit 1 "ERROR: Archives could not be retrieved from the RGW."
|
|
fi
|
|
elif [[ "x${REMOTE}" == "x" ]]; then
|
|
if [[ -d $ARCHIVE_DIR ]]; then
|
|
archives=$(find $ARCHIVE_DIR/ -iname "*.gz" -print)
|
|
echo
|
|
echo "All Local Archives"
|
|
echo "=============================================="
|
|
for archive in $archives
|
|
do
|
|
echo $archive | cut -d '/' -f 8
|
|
done
|
|
clean_and_exit 0 ""
|
|
else
|
|
clean_and_exit 1 "ERROR: Local archive directory is not available."
|
|
fi
|
|
else
|
|
usage 1
|
|
fi
|
|
}
|
|
|
|
# Retrieve the archive from the desired location and decompress it into
|
|
# the restore directory
|
|
get_archive() {
|
|
ARCHIVE_FILE=$1
|
|
REMOTE=$2
|
|
|
|
if [[ "x$REMOTE" == "xremote" ]]; then
|
|
retrieve_remote_archive $ARCHIVE_FILE
|
|
if [[ $? -ne 0 ]]; then
|
|
clean_and_exit 1 "ERROR: Could not retrieve remote archive: $ARCHIVE_FILE"
|
|
fi
|
|
elif [[ "x$REMOTE" == "x" ]]; then
|
|
if [[ -e $ARCHIVE_DIR/$ARCHIVE_FILE ]]; then
|
|
cp $ARCHIVE_DIR/$ARCHIVE_FILE $TMP_DIR/$ARCHIVE_FILE
|
|
if [[ $? -ne 0 ]]; then
|
|
clean_and_exit 1 "ERROR: Could not copy local archive to restore directory."
|
|
fi
|
|
else
|
|
clean_and_exit 1 "ERROR: Local archive file could not be found."
|
|
fi
|
|
else
|
|
usage 1
|
|
fi
|
|
|
|
echo "Decompressing archive $ARCHIVE_FILE..."
|
|
cd $TMP_DIR
|
|
tar zxvf - < $TMP_DIR/$ARCHIVE_FILE 1>/dev/null
|
|
if [[ $? -ne 0 ]]; then
|
|
clean_and_exit 1 "ERROR: Archive decompression failed."
|
|
fi
|
|
}
|
|
|
|
# Display all databases from an archive
|
|
list_databases() {
|
|
ARCHIVE_FILE=$1
|
|
REMOTE=$2
|
|
WHERE="local"
|
|
|
|
if [[ "x${REMOTE}" != "x" ]]; then
|
|
WHERE="remote"
|
|
fi
|
|
|
|
# Get the archive from the source location (local/remote)
|
|
get_archive $ARCHIVE_FILE $REMOTE
|
|
|
|
# Expectation is that the database listing will be put into
|
|
# the given file one database per line
|
|
get_databases $TMP_DIR $DB_FILE
|
|
if [[ "$?" -ne 0 ]]; then
|
|
clean_and_exit 1 "ERROR: Could not list databases."
|
|
fi
|
|
|
|
if [[ -f "$DB_FILE" ]]; then
|
|
echo " "
|
|
echo "Databases in the $WHERE archive $ARCHIVE_FILE"
|
|
echo "================================================================================"
|
|
cat $DB_FILE
|
|
else
|
|
echo "There is no database in the archive."
|
|
fi
|
|
}
|
|
|
|
# Return 1 if the given database exists in the database file. 0 otherwise.
|
|
database_exists() {
|
|
DB=$1
|
|
|
|
grep "${DB}" ${DB_FILE}
|
|
if [[ $? -eq 0 ]]; then
|
|
return 1
|
|
fi
|
|
return 0
|
|
}
|
|
|
|
# This is the main CLI interpreter function
|
|
cli_main() {
|
|
ARGS=("$@")
|
|
|
|
# Create temp directory for a staging area to decompress files into
|
|
export TMP_DIR=$(mktemp -d)
|
|
|
|
# Create a temp file for storing list of databases (if needed)
|
|
export DB_FILE=$(mktemp -p /tmp)
|
|
|
|
if [[ ${#ARGS[@]} -gt 4 ]]; then
|
|
usage 1
|
|
elif [[ ${#ARGS[@]} -eq 1 ]]; then
|
|
if [[ "${ARGS[0]}" == "list_archives" ]]; then
|
|
list_archives
|
|
clean_and_exit 0 ""
|
|
elif [[ "${ARGS[0]}" == "help" ]]; then
|
|
usage 0
|
|
else
|
|
usage 1
|
|
fi
|
|
elif [[ ${#ARGS[@]} -eq 2 ]]; then
|
|
if [[ "${ARGS[0]}" == "list_databases" ]]; then
|
|
list_databases ${ARGS[1]}
|
|
clean_and_exit 0 ""
|
|
elif [[ "${ARGS[0]}" == "list_archives" ]]; then
|
|
list_archives ${ARGS[1]}
|
|
clean_and_exit 0 ""
|
|
else
|
|
usage 1
|
|
fi
|
|
elif [[ ${#ARGS[@]} -eq 3 || ${#ARGS[@]} -eq 4 ]]; then
|
|
if [[ "${ARGS[0]}" == "list_databases" ]]; then
|
|
list_databases ${ARGS[1]} ${ARGS[2]}
|
|
clean_and_exit 0 ""
|
|
elif [[ "${ARGS[0]}" != "restore" ]]; then
|
|
usage 1
|
|
else
|
|
ARCHIVE=${ARGS[1]}
|
|
DB_SPEC=${ARGS[2]}
|
|
REMOTE=""
|
|
if [[ ${#ARGS[@]} -eq 4 ]]; then
|
|
REMOTE=${ARGS[3]}
|
|
fi
|
|
|
|
#Get all the databases in that archive
|
|
get_archive $ARCHIVE $REMOTE
|
|
|
|
if [[ "$( echo $DB_SPEC | tr '[a-z]' '[A-Z]')" != "ALL" ]]; then
|
|
# Expectation is that the database listing will be put into
|
|
# the given file one database per line
|
|
get_databases $TMP_DIR $DB_FILE
|
|
if [[ "$?" -ne 0 ]]; then
|
|
clean_and_exit 1 "ERROR: Could not get the list of databases to restore."
|
|
fi
|
|
|
|
#check if the requested database is available in the archive
|
|
database_exists $DB_SPEC
|
|
if [[ $? -ne 1 ]]; then
|
|
clean_and_exit 1 "ERROR: Database ${DB_SPEC} does not exist."
|
|
fi
|
|
|
|
echo "Restoring Database $DB_SPEC And Grants"
|
|
restore_single_db $DB_SPEC $TMP_DIR
|
|
if [[ "$?" -eq 0 ]]; then
|
|
echo "Single database restored successfully."
|
|
else
|
|
clean_and_exit 1 "ERROR: Single database restore failed."
|
|
fi
|
|
echo "Tail ${LOG_FILE} for restore log."
|
|
clean_and_exit 0 ""
|
|
else
|
|
echo "Restoring All The Databases. This could take a few minutes..."
|
|
restore_all_dbs $TMP_DIR
|
|
if [[ "$?" -eq 0 ]]; then
|
|
echo "All databases restored successfully."
|
|
else
|
|
clean_and_exit 1 "ERROR: Database restore failed."
|
|
fi
|
|
clean_and_exit 0 "Tail ${LOG_FILE} for restore log."
|
|
fi
|
|
fi
|
|
else
|
|
usage 1
|
|
fi
|
|
|
|
clean_and_exit 0 "Done"
|
|
}
|
|
{{- end }}
|