#!/bin/bash # LinuxGSM command_backup.sh function # Author: Daniel Gibbs # Contributor: UltimateByte # Website: https://gameservermanagers.com # Description: Creates a .tar.gz file in the backup directory. local commandname="BACKUP" local commandaction="Backup" local function_selfname="$(basename "$(readlink -f "${BASH_SOURCE[0]}")")" check.sh # Trap to remove lockfile on quit. fn_backup_trap(){ echo "" echo -ne "backup ${backupname}.tar.gz..." fn_print_canceled_eol_nl fn_script_log_info "Backup ${backupname}.tar.gz: CANCELED" sleep 1 rm -f "${backupdir}/${backupname}.tar.gz" | tee -a "${lgsmlog}" echo -ne "backup ${backupname}.tar.gz..." fn_print_removed_eol_nl fn_script_log_info "Backup ${backupname}.tar.gz: REMOVED" # Remove lock file rm -f "${tmpdir}/.backup.lock" core_exit.sh } # Check if a backup is pending or has been aborted using .backup.lock fn_backup_check_lockfile(){ if [ -f "${tmpdir}/.backup.lock" ]; then fn_print_info_nl "Lock file found: Backup is currently running" fn_script_log_error "Lock file found: Backup is currently running: ${tmpdir}/.backup.lock" core_exit.sh fi } # Initialisation fn_backup_init(){ # Backup file name with servicename and current date backupname="${servicename}-$(date '+%Y-%m-%d-%H%M%S')" info_distro.sh fn_print_dots "Backup starting" fn_script_log_info "Backup starting" sleep 1 fn_print_ok "Backup starting" sleep 1 echo -ne "\n" if [ ! -d "${backupdir}" ]||[ "${backupcount}" == "0" ]; then fn_print_info_nl "There are no previous backups" else if [ "${lastbackupdaysago}" == "0" ]; then daysago="less than 1 day ago" elif [ "${lastbackupdaysago}" == "1" ]; then daysago="1 day ago" else daysago="${lastbackupdaysago} days ago" fi echo " * Previous backup was created ${daysago}, total size ${lastbackupsize}" sleep 1 fi } # Check if server is started and wether to stop it fn_backup_stop_server(){ check_status.sh # Server is stopped if [ "${status}" == "0" ]; then serverstopped="no" # Server is running and stoponbackup=off elif [ "${stoponbackup}" == "off" ]; then serverstopped="no" fn_print_warn_nl "${servicename} is currently running" echo " * Although unlikely; creating a backup while ${servicename} is running might corrupt the backup." fn_script_log_warn "${servicename} is currently running" fn_script_log_warn "Although unlikely; creating a backup while ${servicename} is running might corrupt the backup" # Server is running and will be stopped if stoponbackup=on or unset else fn_print_warn_nl "${servicename} will be stopped during the backup" fn_script_log_warn "${servicename} will be stopped during the backup" sleep 4 serverstopped="yes" exitbypass=1 command_stop.sh fi } # Create required folders fn_backup_dir(){ # Create backupdir if it doesn't exist if [ ! -d "${backupdir}" ]; then mkdir -p "${backupdir}" fi } fn_backup_create_lockfile(){ # Create lockfile date > "${tmpdir}/.backup.lock" fn_script_log_info "Lockfile generated" fn_script_log_info "${tmpdir}/.backup.lock" # trap to remove lockfile on quit. trap fn_backup_trap INT } # Compressing files fn_backup_compression(){ # Tells how much will be compressed using rootdirduexbackup value from info_distro and prompt for continue fn_print_info "A total of ${rootdirduexbackup} will be compressed." fn_script_log_info "A total of ${rootdirduexbackup} will be compressed: ${backupdir}/${backupname}.tar.gz" sleep 2 fn_print_dots "Backup (${rootdirduexbackup}) ${backupname}.tar.gz, in progress..." fn_script_log_info "backup ${rootdirduexbackup} ${backupname}.tar.gz, in progress" excludedir=$(fn_backup_relpath) # CHECK THAT excludedir is a valid path. if [ ! -d "${excludedir}" ] ; then fn_print_info_nl "Problem identifying the previous backup directory for exclusion." fn_script_log_error "Problem identifying the previous backup directory for exclusion" core_exit.sh fi tar -czf "${backupdir}/${backupname}.tar.gz" -C "${rootdir}" --exclude "${excludedir}" --exclude "${tmpdir}/.backup.lock" ./* local exitcode=$? if [ ${exitcode} -ne 0 ]; then fn_print_fail_eol fn_script_log_fatal "Backup in progress: FAIL" echo "${tarcmd}" | tee -a "${lgsmlog}" fn_print_fail_nl "Starting backup" fn_script_log_fatal "Starting backup" else fn_print_ok_eol sleep 1 fn_print_ok_nl "Completed: ${backupname}.tar.gz, total size $(du -sh "${backupdir}/${backupname}.tar.gz" | awk '{print $1}')" fn_script_log_pass "Backup created: ${backupname}.tar.gz, total size $(du -sh "${backupdir}/${backupname}.tar.gz" | awk '{print $1}')" fi # Remove lock file rm -f "${tmpdir}/.backup.lock" } # Clear old backups according to maxbackups and maxbackupdays variables fn_backup_prune(){ # Clear if backup variables are set if [ -n "${maxbackups}" ]&&[ -n "${maxbackupdays}" ]; then # How many backups there are info_distro.sh # How many backups exceed maxbackups backupquotadiff=$((backupcount-maxbackups)) # How many backups exceed maxbackupdays backupsoudatedcount=$(find "${backupdir}"/ -type f -name "*.tar.gz" -mtime +"${maxbackupdays}"|wc -l) # If anything can be cleared if [ "${backupquotadiff}" -gt "0" ]||[ "${backupsoudatedcount}" -gt "0" ]; then fn_print_dots "Pruning" fn_script_log_info "Backup pruning activated" sleep 1 fn_print_ok_nl "Pruning" sleep 1 # If maxbackups greater or equal to backupsoutdatedcount, then it is over maxbackupdays if [ "${backupquotadiff}" -ge "${backupsoudatedcount}" ]; then # Display how many backups will be cleared echo " * Pruning: ${backupquotadiff} backup(s) has exceeded the ${maxbackups} backups limit" fn_script_log_info "Pruning: ${backupquotadiff} backup(s) has exceeded the ${maxbackups} backups limit" sleep 1 fn_print_dots "Pruning: Clearing ${backupquotadiff} backup(s)" fn_script_log_info "Pruning: Clearing ${backupquotadiff} backup(s)" sleep 1 # Clear backups over quota find "${backupdir}"/ -type f -name "*.tar.gz" -printf '%T@ %p\n' | sort -rn | tail -${backupquotadiff} | cut -f2- -d" " | xargs rm fn_print_ok_nl "Pruning: Clearing ${backupquotadiff} backup(s)" fn_script_log_pass "Pruning: Cleared ${backupquotadiff} backup(s)" # If maxbackupdays is used over maxbackups elif [ "${backupquotadiff}" -lt "${backupsoudatedcount}" ]; then # Display how many backups will be cleared echo " * Pruning: ${backupsoudatedcount} backup(s) are older than ${maxbackupdays} days." fn_script_log_info "Pruning: ${backupsoudatedcount} backup(s) older than ${maxbackupdays} days." sleep 1 fn_print_dots "Pruning: Clearing ${backupquotadiff} backup(s)." fn_script_log_info "Pruning: Clearing ${backupquotadiff} backup(s)" sleep 1 # Clear backups over quota find "${backupdir}"/ -type f -mtime +"${maxbackupdays}" -exec rm -f {} \; fn_print_ok_nl "Pruning: Clearing ${backupquotadiff} backup(s)" fn_script_log_pass "Pruning: Cleared ${backupquotadiff} backup(s)" fi sleep 1 fi fi } fn_backup_relpath() { # Written by CedarLUG as a "realpath --relative-to" alternative in bash # Populate an array of tokens initialized from the rootdir components: declare -a rdirtoks=($(readlink -f "${rootdir}" | sed "s/\// /g")) if [ ${#rdirtoks[@]} -eq 0 ]; then fn_print_info_nl "Problem assessing rootdir during relative path assessment" fn_script_log_error "Problem assessing rootdir during relative path assessment: ${rootdir}" core_exit.sh fi # Populate an array of tokens initialized from the backupdir components: declare -a bdirtoks=($(readlink -f "${backupdir}" | sed "s/\// /g")) if [ ${#bdirtoks[@]} -eq 0 ]; then fn_print_info_nl "Problem assessing backupdir during relative path assessment" fn_script_log_error "Problem assessing backupdir during relative path assessment: ${rootdir}" core_exit.sh fi # Compare the leading entries of each array. These common elements will be clipped off # for the relative path output. for ((base=0; base<${#rdirtoks[@]}; base++)) do [[ "${rdirtoks[$base]}" != "${bdirtoks[$base]}" ]] && break done # Next, climb out of the remaining rootdir location with updir references... for ((x=base;x<${#rdirtoks[@]};x++)) do echo -n "../" done # Climb down the remaining components of the backupdir location. for ((x=base;x<$(( ${#bdirtoks[@]} - 1 ));x++)) do echo -n "${bdirtoks[$x]}/" done # In the event there were no directories left in the backupdir above to # traverse down, just add a newline. Otherwise at this point, there is # one remaining directory component in the backupdir to navigate. if (( "$base" < "${#bdirtoks[@]}" )) ; then echo "${bdirtoks[ $(( ${#bdirtoks[@]} - 1)) ]}" else echo fi } # Restart the server if it was stopped for the backup fn_backup_start_server(){ if [ "${serverstopped}" == "yes" ]; then exitbypass=1 command_start.sh fi } # Run functions fn_backup_check_lockfile fn_backup_create_lockfile fn_backup_init fn_backup_stop_server fn_backup_dir fn_backup_compression fn_backup_prune fn_backup_start_server core_exit.sh