From eb393896ac2ec59987bdd2c09f91d614819a9479 Mon Sep 17 00:00:00 2001 From: Price Hiller Date: Sun, 26 Dec 2021 02:05:44 -0600 Subject: [PATCH] Added support for xmllint on start to avoid issues with silent errors --- Scripts/7D2D-Manage.bash | 1024 +++++++++++++++++++------------------- 1 file changed, 516 insertions(+), 508 deletions(-) diff --git a/Scripts/7D2D-Manage.bash b/Scripts/7D2D-Manage.bash index 425ee4d..91617f3 100644 --- a/Scripts/7D2D-Manage.bash +++ b/Scripts/7D2D-Manage.bash @@ -1,145 +1,144 @@ #!/bin/bash - echo_rgb() { - # Echo a colored string to the terminal based on rgb values - # - # Positional Arguments: - # - # message - # - The message to be printed to stdout - # red - # - The red value from 0 to 255 - # green - # - The green value from 0 to 255 - # blue - # - The blue value from 0 to 255 - # - # Usage: - # echo_rgb "Yep" 10 8 30 - # - # POSIX Compliant: - # N/A - # + # Echo a colored string to the terminal based on rgb values + # + # Positional Arguments: + # + # message + # - The message to be printed to stdout + # red + # - The red value from 0 to 255 + # green + # - The green value from 0 to 255 + # blue + # - The blue value from 0 to 255 + # + # Usage: + # echo_rgb "Yep" 10 8 30 + # + # POSIX Compliant: + # N/A + # - local red - local green - local blue - local input + local red + local green + local blue + local input - input="${1}" - red="${2}" - green="${3}" - blue="${4}" + input="${1}" + red="${2}" + green="${3}" + blue="${4}" - printf "\e[0;38;2;%s;%s;%sm%s\e[m\n" "${red}" "${green}" "${blue}" "${input}" + printf "\e[0;38;2;%s;%s;%sm%s\e[m\n" "${red}" "${green}" "${blue}" "${input}" } log() { - # Print a message and send it to stdout or stderr depending upon log level, also configurable with debug etc. - # - # Arguments: - # level - # - The log level, defined within a case check in this function - # message - # - The info message - # line_number - # - The line number of the calling function (${LINNO}) - # - # Usage: - # log "info" "Could not find that directory" - # - # POSIX Compliant: - # Yes - # + # Print a message and send it to stdout or stderr depending upon log level, also configurable with debug etc. + # + # Arguments: + # level + # - The log level, defined within a case check in this function + # message + # - The info message + # line_number + # - The line number of the calling function (${LINNO}) + # + # Usage: + # log "info" "Could not find that directory" + # + # POSIX Compliant: + # Yes + # - # Set debug status depending if a global debug variable has been set to either 1 or 0 - local debug - if [ ${DEBUG} ]; then - debug=${DEBUG} - else - debug=0 - fi + # Set debug status depending if a global debug variable has been set to either 1 or 0 + local debug + if [ ${DEBUG} ]; then + debug=${DEBUG} + else + debug=0 + fi - local FORMAT - FORMAT="[$(echo_rgb "$(date +%Y-%m-%dT%H:%M:%S)" 180 140 255)]" + local FORMAT + FORMAT="[$(echo_rgb "$(date +%Y-%m-%dT%H:%M:%S)" 180 140 255)]" - # Convert the level to uppercase - local level - level=$(echo "${1}" | tr '[:lower:]' '[:upper:]') + # Convert the level to uppercase + local level + level=$(echo "${1}" | tr '[:lower:]' '[:upper:]') - local message - message="${2}" + local message + message="${2}" - case "${level}" in + case "${level}" in INFO) - # Output all info log levels to stdout - printf "${FORMAT}[$(echo_rgb "INFO" 0 140 255)] %s\n" "${message}" >&1 - return 0 - ;; + # Output all info log levels to stdout + printf "${FORMAT}[$(echo_rgb "INFO" 0 140 255)] %s\n" "${message}" >&1 + return 0 + ;; WARN | WARNING) - # Output all warning log levels to stdout - printf "${FORMAT}[$(echo_rgb "WARNING" 255 255 0)] %s\n" "${message}" >&1 - return 0 - ;; + # Output all warning log levels to stdout + printf "${FORMAT}[$(echo_rgb "WARNING" 255 255 0)] %s\n" "${message}" >&1 + return 0 + ;; DEBUG) - # Output all debug log levels to stdout - if [ "${DEBUG}" ]; then - printf "${FORMAT}[$(echo_rgb "DEBUG" 0 160 110)] %s\n" "${message}" >&1 - fi - return 0 - ;; + # Output all debug log levels to stdout + if [ "${DEBUG}" ]; then + printf "${FORMAT}[$(echo_rgb "DEBUG" 0 160 110)] %s\n" "${message}" >&1 + fi + return 0 + ;; ERROR) - # Output all error log levels to stderr - printf "${FORMAT}[$(echo_rgb "ERROR" 255 0 0)] %s\n" "${message}" >&2 - return 0 - ;; + # Output all error log levels to stderr + printf "${FORMAT}[$(echo_rgb "ERROR" 255 0 0)] %s\n" "${message}" >&2 + return 0 + ;; # Further log levels can be added by extending this switch statement with more comparisons *) # Default case, no matches - # Returns non-zero code as an improper log option was passed, this helps with using `set -e` - printf "${FORMAT}[ERROR] %s\n" "Invalid log level passed, received level \"${level}\" with message \"${message}\"" >&2 - return 1 - ;; - esac + # Returns non-zero code as an improper log option was passed, this helps with using `set -e` + printf "${FORMAT}[ERROR] %s\n" "Invalid log level passed, received level \"${level}\" with message \"${message}\"" >&2 + return 1 + ;; + esac } confirmation() { - # Receive confirmation from user as y, Y, n, or N - # returns 0 when answer is yes and 1 when answer is no - # - # Arguments: - # message - # - The confirmation prompt sent to the user, for example: - # Would you like to overwrite foobar.txt (y/N)? - # - # Usage: - # confirmation "Some prompt" - # - Sends "Some prompt" to the user and gets their input - # - # POSIX Compliant: - # Yes - # + # Receive confirmation from user as y, Y, n, or N + # returns 0 when answer is yes and 1 when answer is no + # + # Arguments: + # message + # - The confirmation prompt sent to the user, for example: + # Would you like to overwrite foobar.txt (y/N)? + # + # Usage: + # confirmation "Some prompt" + # - Sends "Some prompt" to the user and gets their input + # + # POSIX Compliant: + # Yes + # - local message - message="${1}" + local message + message="${1}" - local choice + local choice - while true; do - read -p "${message} " -n 1 -r choice - case "$choice" in - y | Y) - echo "" - return 0 - ;; - n | N) - echo "" - return 1 - ;; - *) echo -e "\nInput must be either y, Y, n, or N" ;; - esac - done + while true; do + read -p "${message} " -n 1 -r choice + case "$choice" in + y | Y) + echo "" + return 0 + ;; + n | N) + echo "" + return 1 + ;; + *) echo -e "\nInput must be either y, Y, n, or N" ;; + esac + done } ### IMPORTS ### @@ -150,21 +149,21 @@ BASE_DIR="${HOME}/7-Days-To-Die" ### CONSTANTS ### important() { - echo_rgb "${1}" 135 195 255 + echo_rgb "${1}" 135 195 255 } start_server() { - local server_id - local can_kill + local server_id + local can_kill - server_id="" - can_kill=1 + server_id="" + can_kill=1 - while :; do - case ${1} in - -h | -\? | --help) - printf "Usage: %s\n" \ - "start [options] + while :; do + case ${1} in + -h | -\? | --help) + printf "Usage: %s\n" \ + "start [options] --server | -s Starts the given server id @@ -176,284 +175,295 @@ start_server() { Example: --can-kill" - exit - ;; - --) # End of all options. - break - ;; - --server | -s) - shift - server_id="${1}" - [[ -z "${server_id}" ]] && log "error" "No server id passed" && exit 1 - ;; - --can-kill | -c) - can_kill=0 - ;; - -?*) - printf 'Unknown option: %s\n' "$1" >&2 - ;; - *) # Default case: No more options, so break out of the loop. - break ;; - esac - shift - done + exit + ;; + --) # End of all options. + break + ;; + --server | -s) + shift + server_id="${1}" + [[ -z "${server_id}" ]] && log "error" "No server id passed" && exit 1 + ;; + --can-kill | -c) + can_kill=0 + ;; + -?*) + printf 'Unknown option: %s\n' "$1" >&2 + ;; + *) # Default case: No more options, so break out of the loop. + break ;; + esac + shift + done - [[ -z "${server_id}" ]] && log "error" "No server id passed" && exit 1 + [[ -z "${server_id}" ]] && log "error" "No server id passed" && exit 1 - local server_directory - local server_config - local server_save_dir - local server_name - local prefix - local server_session_name + local server_directory + local server_config + local server_save_dir + local server_name + local prefix + local server_session_name - server_name="Server-${server_id}" - server_directory="${BASE_DIR}/${server_name}" - server_config="${server_directory}/serverconfig.xml" - server_save_dir="${server_directory}/Saves/" - server_userdata_dir="${server_directory}/UserData" - prefix="7D2D" - server_session_name="${prefix}-${server_name}" + server_name="Server-${server_id}" + server_directory="${BASE_DIR}/${server_name}" + server_config="${server_directory}/serverconfig.xml" + server_save_dir="${server_directory}/Saves/" + server_userdata_dir="${server_directory}/UserData" + prefix="7D2D" + server_session_name="${prefix}-${server_name}" - [[ ! -d "${server_directory}" ]] && - log "error" "Unable to find the server directory for $(important "${server_name}")" && - return 1 + [[ ! -d "${server_directory}" ]] && + log "error" "Unable to find the server directory for $(important "${server_name}")" && + return 1 - backup_configs "${server_directory}" + backup_configs "${server_directory}" + local server_port + server_port="$((START_PORT_RANGE + server_id))" - local server_port - server_port="$(( START_PORT_RANGE + server_id ))" + log "info" "Generating a few required directories in $(important "${server_directory}")" + mkdir -p "${server_save_dir}" + mkdir -p "${server_userdata_dir}" - log "info" "Generating a few required directories in $(important "${server_directory}")" - mkdir -p "${server_save_dir}" - mkdir -p "${server_userdata_dir}" + log "info" "Configuring $(important "serverconfig.xml")" - log "info" "Configuring $(important "serverconfig.xml")" + # Overwrite values that we want to control, e.g. server port + while IFS='' read -r; do + if [[ "${REPLY}" = *"property name=\"ServerPort\""* ]]; then + printf "\t%s\n" "" - # Overwrite values that we want to control, e.g. server port - while IFS='' read -r; do - if [[ "${REPLY}" = *"property name=\"ServerPort\""* ]]; then - printf "\t%s\n" "" + # Override SaveGameFolder, opiniated in that it should exist in the Server Directory + elif [[ "${REPLY}" = *"property name=\"SaveGameFolder\""* ]]; then + printf "\t%s\n" "" + # Override UserDataFolder, opiniated in that it should exist in the Server Directory + elif [[ "${REPLY}" = *"property name=\"UserDataFolder\""* ]]; then + printf "\t%s\n" "" + # This is a path RELATIVE to the save game folder, hence we overwrite ../ so it's in the + # base of the server directory + elif [[ "${REPLY}" = *"property name=\"AdminFileName\""* ]]; then + printf "\t%s\n" "" + else + printf "%s\n" "${REPLY}" + fi + done <"${server_config}" >"temp-serverconfig.xml" - # Override SaveGameFolder, opiniated in that it should exist in the Server Directory - elif [[ "${REPLY}" = *"property name=\"SaveGameFolder\""* ]]; then - printf "\t%s\n" "" - # Override UserDataFolder, opiniated in that it should exist in the Server Directory - elif [[ "${REPLY}" = *"property name=\"UserDataFolder\""* ]]; then - printf "\t%s\n" "" - # This is a path RELATIVE to the save game folder, hence we overwrite ../ so it's in the - # base of the server directory - elif [[ "${REPLY}" = *"property name=\"AdminFileName\""* ]]; then - printf "\t%s\n" "" - else - printf "%s\n" "${REPLY}" + # Occasionally the closing tag in serverconfig.xml will be missing, this ensures it is there in that scenario + if [[ ! "$(tail "temp-serverconfig.xml" -n 1)" = *"ServerSettings"* ]]; then + printf "\n" >>"temp-serverconfig.xml" fi - done < "${server_config}" > "temp-serverconfig.xml" - # Occasionally the closing tag in serverconfig.xml will be missing, this ensures it is there in that scenario - if [[ ! "$(tail "temp-serverconfig.xml" -n 1 )" = *"ServerSettings"* ]]; then - printf "\n" >> "temp-serverconfig.xml" - fi + mv "temp-serverconfig.xml" "${server_config}" - mv "temp-serverconfig.xml" "${server_config}" - - log "info" "Starting $(important "${server_name}") located at $(important "${server_directory}") on port $(important "${server_port}")" - - if tmux has-session -t "${server_session_name}" >/dev/null 2>&1; then - log "warning" "$(important "${server_name}") is currently running" - - if (( can_kill == 0 )); then - log "info" "Been explicitly permitted to kill, bypassing confirmation" - kill_server -s "${server_id}" + # xmllint, helps avoiding the random server launch failures for apparently no reason + if which xmllint >/dev/null 2>&1; then + for xml_file in "${server_directory}"/*.xml; do + log "info" "Linting $(important "${xml_file}")..." + if ! xmllint "${xml_file}" > /dev/null; then + log "error" "Xml parsing error in ${xml_file}, resolve the error and attempt to start again" + fi + done else - if confirmation "Would you like to kill $(important "${server_name}")? (Y/n)" ; then - log "info" "Given answer $(important "yes") to kill the server, killing $(important "${server_name}")" - kill_server -s "${server_id}" - else - log "info" "Given answer $(important "no") to kill the server, exiting" - exit 0 - fi + log "warning" "$(important "xmllint") not installed or not in PATH, skipping lint check" fi - fi - log "info" "Creating new session for $(important "${server_name}") as session $(important "${server_session_name}")" - tmux new-session -d -s "${server_session_name}" - tmux send-keys \ -"until ${server_directory}/startserver.sh -configfile=${server_directory}/serverconfig.xml; do + log "info" "Starting $(important "${server_name}") located at $(important "${server_directory}") on port $(important "${server_port}")" + + if tmux has-session -t "${server_session_name}" >/dev/null 2>&1; then + log "warning" "$(important "${server_name}") is currently running" + + if ((can_kill == 0)); then + log "info" "Been explicitly permitted to kill, bypassing confirmation" + kill_server -s "${server_id}" + else + if confirmation "Would you like to kill $(important "${server_name}")? (Y/n)"; then + log "info" "Given answer $(important "yes") to kill the server, killing $(important "${server_name}")" + kill_server -s "${server_id}" + else + log "info" "Given answer $(important "no") to kill the server, exiting" + exit 0 + fi + fi + fi + + log "info" "Creating new session for $(important "${server_name}") as session $(important "${server_session_name}")" + tmux new-session -d -s "${server_session_name}" + tmux send-keys \ + "until ${server_directory}/startserver.sh -configfile=${server_directory}/serverconfig.xml; do echo 'Server died with code $?, restarting in 60 seconds...' >&2 sleep 60; done" C-m - log "info" "Finished starting $(important "${server_name}") on port $(important "${server_port}") as tmux session $(important "${server_session_name}")" + log "info" "Finished starting $(important "${server_name}") on port $(important "${server_port}") as tmux session $(important "${server_session_name}")" } kill_server() { - local prefix - local server_id + local prefix + local server_id - server_id="" - prefix="7D2D" + server_id="" + prefix="7D2D" - while :; do - case ${1} in - -h | -\? | --help) - printf "Usage: %s\n" \ - "kill [options] + while :; do + case ${1} in + -h | -\? | --help) + printf "Usage: %s\n" \ + "kill [options] --server | -s Forcefully kills the server with the given id Example: --server 3" - exit - ;; - --) # End of all options. - break - ;; - --server | -s) - shift - server_id="${1}" - [[ -z "${server_id}" ]] && log "error" "No server id passed" && exit 1 - ;; - -?*) - printf 'Unknown option: %s\n' "$1" >&2 - ;; - *) # Default case: No more options, so break out of the loop. - break ;; - esac - shift - done + exit + ;; + --) # End of all options. + break + ;; + --server | -s) + shift + server_id="${1}" + [[ -z "${server_id}" ]] && log "error" "No server id passed" && exit 1 + ;; + -?*) + printf 'Unknown option: %s\n' "$1" >&2 + ;; + *) # Default case: No more options, so break out of the loop. + break ;; + esac + shift + done - [[ -z "${server_id}" ]] && log "error" "No server id passed" && exit 1 + [[ -z "${server_id}" ]] && log "error" "No server id passed" && exit 1 - if tmux kill-session -t "${prefix}-Server-${server_id}" >/dev/null 2>&1; then - log "info" "Stopped $(important "${prefix}-Server-${server_id}")" - return 0 - else - log "error" "$(important "${prefix}-Server-${server_id}") is not running" - return 1 - fi + if tmux kill-session -t "${prefix}-Server-${server_id}" >/dev/null 2>&1; then + log "info" "Stopped $(important "${prefix}-Server-${server_id}")" + return 0 + else + log "error" "$(important "${prefix}-Server-${server_id}") is not running" + return 1 + fi } install() { - local server_id - server_id="" + local server_id + server_id="" - while :; do - case ${1} in - -h | -\? | --help) - printf "Usage: %s\n" \ - "install [OPTIONS] + while :; do + case ${1} in + -h | -\? | --help) + printf "Usage: %s\n" \ + "install [OPTIONS] --server | -s Installs the server to the given id if it doesn't exist Example: --server 3" - exit - ;; - --) # End of all options. - break - ;; - --server | -s) - shift - server_id="${1}" - [[ -z "${server_id}" ]] && log "error" "No server id passed" && exit 1 - ;; - -?*) - printf 'Unknown option: %s\n' "$1" >&2 - ;; - *) # Default case: No more options, so break out of the loop. - break ;; - esac - shift - done + exit + ;; + --) # End of all options. + break + ;; + --server | -s) + shift + server_id="${1}" + [[ -z "${server_id}" ]] && log "error" "No server id passed" && exit 1 + ;; + -?*) + printf 'Unknown option: %s\n' "$1" >&2 + ;; + *) # Default case: No more options, so break out of the loop. + break ;; + esac + shift + done - [[ -z "${server_id}" ]] && log "error" "No server id passed" && exit 1 + [[ -z "${server_id}" ]] && log "error" "No server id passed" && exit 1 - local prefix + local prefix - prefix="7D2D" + prefix="7D2D" - local server_name - local server_directory - local server_admin_xml - server_name="Server-${server_id}" - server_directory="${BASE_DIR}/${server_name}" - server_admin_xml="${HOME}/.local/share/7DaysToDie/Saves/serveradmin.xml" + local server_name + local server_directory + local server_admin_xml + server_name="Server-${server_id}" + server_directory="${BASE_DIR}/${server_name}" + server_admin_xml="${HOME}/.local/share/7DaysToDie/Saves/serveradmin.xml" - # Ensure that the global mods directory exists - [[ -d "${server_directory}" ]] && - log "error" "A server already exists at $(important "${server_directory}"), delete it and try again" && - exit 1 + # Ensure that the global mods directory exists + [[ -d "${server_directory}" ]] && + log "error" "A server already exists at $(important "${server_directory}"), delete it and try again" && + exit 1 - log "info" "Installing $(important "${server_name}") to $(important "${server_directory}")" - steamcmd +force_install_dir "${server_directory}" +login anonymous +app_update 294420 validate +quit - mkdir -p "${server_directory}/Mods" + log "info" "Installing $(important "${server_name}") to $(important "${server_directory}")" + steamcmd +force_install_dir "${server_directory}" +login anonymous +app_update 294420 validate +quit + mkdir -p "${server_directory}/Mods" - log "info" "Running $(important "${server_name}") to get a valid $(important "serveradmin.xml")" - start_server -s "${server_id}" -c - log "info" "Waiting to kill $(important "${server_name}")" - sleep 5 - kill_server -s "${server_id}" + log "info" "Running $(important "${server_name}") to get a valid $(important "serveradmin.xml")" + start_server -s "${server_id}" -c + log "info" "Waiting to kill $(important "${server_name}")" + sleep 5 + kill_server -s "${server_id}" - cp "${server_admin_xml}" "${server_directory}/" - log "info" "Successfully installed $(important "${server_name}") to $(important "${server_directory}")" + cp "${server_admin_xml}" "${server_directory}/" + log "info" "Successfully installed $(important "${server_name}") to $(important "${server_directory}")" } update() { - local server_id - server_id="" + local server_id + server_id="" - while :; do - case ${1} in - -h | -\? | --help) - printf "Usage: %s\n" \ -"update [OPTIONS] + while :; do + case ${1} in + -h | -\? | --help) + printf "Usage: %s\n" \ + "update [OPTIONS] --server | -s Updates the given server id Example: --server 3" - exit - ;; - --) # End of all options. - break - ;; - --server | -s) - shift - server_id="${1}" - [[ -z "${server_id}" ]] && log "error" "No server id passed" && exit 1 - ;; - -?*) - printf 'Unknown option: %s\n' "$1" >&2 - ;; - *) # Default case: No more options, so break out of the loop. - break ;; - esac - shift - done + exit + ;; + --) # End of all options. + break + ;; + --server | -s) + shift + server_id="${1}" + [[ -z "${server_id}" ]] && log "error" "No server id passed" && exit 1 + ;; + -?*) + printf 'Unknown option: %s\n' "$1" >&2 + ;; + *) # Default case: No more options, so break out of the loop. + break ;; + esac + shift + done - [[ -z "${server_id}" ]] && log "error" "No server id passed" && exit 1 + [[ -z "${server_id}" ]] && log "error" "No server id passed" && exit 1 - # Kill the server to ensure a smooth update - kill_server -s "${server_id}" >/dev/null 2>&1 + # Kill the server to ensure a smooth update + kill_server -s "${server_id}" >/dev/null 2>&1 - local server_directory - local server_name - local wait_time - server_name="Server-${server_id}" - server_directory="${BASE_DIR}/${server_name}" - wait_time="5" + local server_directory + local server_name + local wait_time + server_name="Server-${server_id}" + server_directory="${BASE_DIR}/${server_name}" + wait_time="5" - log "info" "Recommend running a $(important "backup") operation before running this, waiting $(important "${wait_time}") seconds before continuing..." - sleep "${wait_time}" + log "info" "Recommend running a $(important "backup") operation before running this, waiting $(important "${wait_time}") seconds before continuing..." + sleep "${wait_time}" - backup_configs "${server_directory}" - log "info" "Updating server $(important "${server_name}") located at $(important "${server_directory}")..." - steamcmd +force_install_dir "${server_directory}" +login anonymous +app_update 294420 validate +quit + backup_configs "${server_directory}" + log "info" "Updating server $(important "${server_name}") located at $(important "${server_directory}")..." + steamcmd +force_install_dir "${server_directory}" +login anonymous +app_update 294420 validate +quit - log "info" "Finished updating $(important "${server_name}") located at $(important "${server_directory}")" + log "info" "Finished updating $(important "${server_name}") located at $(important "${server_directory}")" } @@ -471,76 +481,75 @@ backup_configs() { backup() { - local server_id - server_id="" + local server_id + server_id="" - while :; do - case ${1} in - -h | -\? | --help) - printf "Usage: %s\n" \ -"backup [OPTIONS] + while :; do + case ${1} in + -h | -\? | --help) + printf "Usage: %s\n" \ + "backup [OPTIONS] --server | -s Backups the given server id Example: --server 3" - exit - ;; - --) # End of all options. - break - ;; - --server | -s) - shift - server_id="${1}" - [[ -z "${server_id}" ]] && log "error" "No server id passed" && exit 1 - ;; - -?*) - printf 'Unknown option: %s\n' "$1" >&2 - ;; - *) # Default case: No more options, so break out of the loop. - break ;; - esac - shift - done + exit + ;; + --) # End of all options. + break + ;; + --server | -s) + shift + server_id="${1}" + [[ -z "${server_id}" ]] && log "error" "No server id passed" && exit 1 + ;; + -?*) + printf 'Unknown option: %s\n' "$1" >&2 + ;; + *) # Default case: No more options, so break out of the loop. + break ;; + esac + shift + done - [[ -z "${server_id}" ]] && log "error" "No server id passed" && exit 1 + [[ -z "${server_id}" ]] && log "error" "No server id passed" && exit 1 + # Kill the server to ensure a smooth backup + kill_server -s "${server_id}" >/dev/null 2>&1 - # Kill the server to ensure a smooth backup - kill_server -s "${server_id}" >/dev/null 2>&1 + local backup_dir + local backup_full_path + local server_directory + local server_name + server_name="Server-${server_id}" + server_directory="${BASE_DIR}/${server_name}" + backup_dir="${HOME}/7D2D-Server-Backups/${server_name}" + backup_full_path="${backup_dir}/$(date +%s).tar.gz" - local backup_dir - local backup_full_path - local server_directory - local server_name - server_name="Server-${server_id}" - server_directory="${BASE_DIR}/${server_name}" - backup_dir="${HOME}/7D2D-Server-Backups/${server_name}" - backup_full_path="${backup_dir}/$(date +%s).tar.gz" + [[ ! -d "${server_directory}" ]] && log "info" "The server $(important "${server_name}") had no directory located at $(important "${server_directory}")" && exit 1 - [[ ! -d "${server_directory}" ]] && log "info" "The server $(important "${server_name}") had no directory located at $(important "${server_directory}")" && exit 1 + log "info" "Backing up server $(important "${server_name}") to $(important "${backup_full_path}"), this may take a while" + mkdir -p "${backup_dir}" - log "info" "Backing up server $(important "${server_name}") to $(important "${backup_full_path}"), this may take a while" - mkdir -p "${backup_dir}" - - # Do a check if pv is there, pv is used for showing progress - if which pv > /dev/null 2>&1; then - tar cf - "${server_directory}" -P 2> /dev/null | pv -s "$(du -sb "${server_directory}" | awk '{print $1}')" | gzip > "${backup_full_path}" - else - log "info" "$(important "pv") not installed, not showing progress..." - tar czf "${server_directory}" "${backup_full_path}" 2> /dev/null - fi + # Do a check if pv is there, pv is used for showing progress + if which pv >/dev/null 2>&1; then + tar cf - "${server_directory}" -P 2>/dev/null | pv -s "$(du -sb "${server_directory}" | awk '{print $1}')" | gzip >"${backup_full_path}" + else + log "info" "$(important "pv") not installed, not showing progress..." + tar czf "${server_directory}" "${backup_full_path}" 2>/dev/null + fi } list_servers() { - local picked_option - picked_option="" - while :; do - case ${1} in - -h | -\? | --help) - printf "Usage: %s\n" \ -"list [OPTIONS] + local picked_option + picked_option="" + while :; do + case ${1} in + -h | -\? | --help) + printf "Usage: %s\n" \ + "list [OPTIONS] --running | -r Lists the currently running 7 Days To Die Servers @@ -551,67 +560,67 @@ list_servers() { Example: --installed" - exit - ;; - --) # End of all options. - break - ;; - --running | -r) - picked_option=0 - ;; - --installed | -i) - picked_option=1 - ;; - -?*) - printf 'Unknown option: %s\n' "$1" >&2 - return 1 - ;; - *) # Default case: No more options, so break out of the loop. - break ;; - esac - shift - done + exit + ;; + --) # End of all options. + break + ;; + --running | -r) + picked_option=0 + ;; + --installed | -i) + picked_option=1 + ;; + -?*) + printf 'Unknown option: %s\n' "$1" >&2 + return 1 + ;; + *) # Default case: No more options, so break out of the loop. + break ;; + esac + shift + done - [[ -z "${picked_option}" ]] && - log "error" "An option must be passed for list, check list -h" && - return 1 + [[ -z "${picked_option}" ]] && + log "error" "An option must be passed for list, check list -h" && + return 1 - if (( picked_option == 0 )); then - log "debug" "Listing running servers" - local tmux_sessions - tmux_sessions="$(tmux list-sessions)" >/dev/null 2>&1 - if [[ ! "${?}" -eq "0" ]]; then - important "No servers currently running." + if ((picked_option == 0)); then + log "debug" "Listing running servers" + local tmux_sessions + tmux_sessions="$(tmux list-sessions)" >/dev/null 2>&1 + if [[ ! "${?}" -eq "0" ]]; then + important "No servers currently running." + fi + while read -r; do + if [[ "${REPLY}" = *"-Server-"* ]]; then + local running_server + running_server="$(echo "${REPLY}" | cut -d ":" -f1)" + important "${running_server}" + fi + done <<<"${tmux_sessions}" + elif ((picked_option == 1)); then + log "debug" "Listing installed servers" + while read -r; do + important "${BASE_DIR}/${REPLY}" + done <<<"$(find "${BASE_DIR}" -name "startserver.sh" | cut -d "/" -f5)" fi - while read -r; do - if [[ "${REPLY}" = *"-Server-"* ]]; then - local running_server - running_server="$(echo "${REPLY}" | cut -d ":" -f1)" - important "${running_server}" - fi - done <<< "${tmux_sessions}" - elif (( picked_option == 1 )); then - log "debug" "Listing installed servers" - while read -r; do - important "${BASE_DIR}/${REPLY}" - done <<< "$(find "${BASE_DIR}" -name "startserver.sh" | cut -d "/" -f5)" - fi } usage() { - # Print out usage instructions for the local script - # - # Arguments: - # None - # - # Usage: - # usage - # - # POSIX Compliant: - # Yes - # - printf "Usage: %s\n" \ - "$(basename "${0}") -h + # Print out usage instructions for the local script + # + # Arguments: + # None + # + # Usage: + # usage + # + # POSIX Compliant: + # Yes + # + printf "Usage: %s\n" \ + "$(basename "${0}") -h start Exposes options to start 7 Days To Die Servers, pass -h to it for details kill @@ -626,70 +635,69 @@ usage() { Exposes options to backup 7 Days To Die Servers, pass -h to it for details" } - main() { - # Parse input arguments - # - # Arguments: - # Consult the `usage` function - # - # Usage: - # parse_args "$@" - # - All arguments should be ingested by parse_args first for variable setting - # - # POSIX Compliant: - # Yes - # + # Parse input arguments + # + # Arguments: + # Consult the `usage` function + # + # Usage: + # parse_args "$@" + # - All arguments should be ingested by parse_args first for variable setting + # + # POSIX Compliant: + # Yes + # - while :; do - case ${1} in - -h | -\? | --help) - usage # Display a usage synopsis. - exit - ;; - --) # End of all options. - break - ;; - start | s) - shift - start_server "$@" - break - ;; - kill | k) - shift - kill_server "$@" - break - ;; - install | i) - shift - install "$@" - break - ;; - update | u) - shift - update "$@" - break - ;; - list | l) - shift - list_servers "$@" - break - ;; - backup | b) - shift - backup "$@" - break - ;; - -?*) - printf "Unknown option: %s\n" "$1" >&2 - usage - exit 1 - ;; - *) # Default case: No more options, so break out of the loop. - break ;; - esac - shift - done + while :; do + case ${1} in + -h | -\? | --help) + usage # Display a usage synopsis. + exit + ;; + --) # End of all options. + break + ;; + start | s) + shift + start_server "$@" + break + ;; + kill | k) + shift + kill_server "$@" + break + ;; + install | i) + shift + install "$@" + break + ;; + update | u) + shift + update "$@" + break + ;; + list | l) + shift + list_servers "$@" + break + ;; + backup | b) + shift + backup "$@" + break + ;; + -?*) + printf "Unknown option: %s\n" "$1" >&2 + usage + exit 1 + ;; + *) # Default case: No more options, so break out of the loop. + break ;; + esac + shift + done } main "$@"