gameservergame-servergame-servershacktoberfestdedicated-game-serversgamelinuxgsmserverbashgaminglinuxmultiplayer-game-servershell
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
250 lines
7.7 KiB
250 lines
7.7 KiB
#!/bin/bash
|
|
# LinuxGSM command_monitor.sh function
|
|
# Author: Daniel Gibbs
|
|
# Contributor: UltimateByte
|
|
# Website: https://linuxgsm.com
|
|
# Description: Monitors server by checking for running processes
|
|
# then passes to gamedig and gsquery.
|
|
|
|
local modulename="MONITOR"
|
|
local commandaction="Monitor"
|
|
local function_selfname=$(basename "$(readlink -f "${BASH_SOURCE[0]}")")
|
|
|
|
fn_monitor_check_lockfile(){
|
|
# Monitor does not run it lockfile is not found.
|
|
if [ ! -f "${rootdir}/${lockselfname}" ]; then
|
|
fn_print_dots "Checking lockfile: "
|
|
fn_print_checking_eol
|
|
fn_script_log_info "Checking lockfile: CHECKING"
|
|
fn_sleep_time
|
|
fn_print_error "Checking lockfile: No lockfile found: "
|
|
fn_print_error_eol_nl
|
|
fn_script_log_error "Checking lockfile: No lockfile found: ERROR"
|
|
fn_sleep_time
|
|
echo -e " * Start ${selfname} to run monitor."
|
|
core_exit.sh
|
|
fi
|
|
|
|
# Fix if lockfile is not unix time or contains letters
|
|
if [[ "$(cat "${rootdir}/${lockselfname}")" =~ [A-Za-z] ]]; then
|
|
date '+%s' > "${rootdir}/${lockselfname}"
|
|
fi
|
|
}
|
|
|
|
fn_monitor_check_update(){
|
|
# Monitor will check if update is already running.
|
|
if [ "$(pgrep "${selfname} update" | wc -l)" != "0" ]; then
|
|
fn_print_dots "Checking active updates: "
|
|
fn_print_checking_eol
|
|
fn_script_log_info "Checking active updates: CHECKING"
|
|
fn_sleep_time
|
|
fn_print_error_nl "Checking active updates: SteamCMD is currently checking for updates: "
|
|
fn_print_error_eol
|
|
fn_script_log_error "Checking active updates: SteamCMD is currently checking for updates: ERROR"
|
|
fn_sleep_time
|
|
core_exit.sh
|
|
fi
|
|
}
|
|
|
|
fn_monitor_check_session(){
|
|
fn_print_dots "Checking session: "
|
|
fn_print_checking_eol
|
|
fn_script_log_info "Checking session: CHECKING"
|
|
fn_sleep_time
|
|
# uses status var from check_status.sh
|
|
if [ "${status}" != "0" ]; then
|
|
fn_print_ok "Checking session: "
|
|
fn_print_ok_eol_nl
|
|
fn_script_log_pass "Checking session: OK"
|
|
fn_sleep_time
|
|
else
|
|
fn_print_error "Checking session: "
|
|
fn_print_fail_eol_nl
|
|
fn_script_log_fatal "Checking session: FAIL"
|
|
fn_sleep_time
|
|
alert="restart"
|
|
alert.sh
|
|
fn_script_log_info "Checking session: Monitor is restarting ${selfname}"
|
|
command_restart.sh
|
|
core_exit.sh
|
|
fi
|
|
}
|
|
|
|
fn_monitor_check_queryport(){
|
|
# Monitor will check queryport is set before continuing.
|
|
if [ -z "${queryport}" ]||[ "${queryport}" == "0" ]; then
|
|
fn_print_dots "Checking port: "
|
|
fn_print_checking_eol
|
|
fn_script_log_info "Checking session: CHECKING"
|
|
fn_sleep_time
|
|
fn_print_error "Checking port: Unable to query as queryport is not set: "
|
|
fn_print_error_eol_nl
|
|
fn_script_log_error "Checking port: Unable to query as queryport is not set: ERROR"
|
|
fn_sleep_time
|
|
core_exit.sh
|
|
fi
|
|
}
|
|
|
|
fn_query_gsquery(){
|
|
if [ ! -f "${functionsdir}/query_gsquery.py" ]; then
|
|
fn_fetch_file_github "lgsm/functions" "query_gsquery.py" "${functionsdir}" "chmodx" "norun" "noforce" "nomd5"
|
|
fi
|
|
"${functionsdir}"/query_gsquery.py -a "${ip}" -p "${queryport}" -e "${querytype}" > /dev/null 2>&1
|
|
querystatus="$?"
|
|
}
|
|
|
|
fn_query_tcp(){
|
|
bash -c 'exec 3<> /dev/tcp/'${ip}'/'${queryport}'' > /dev/null 2>&1
|
|
querystatus="$?"
|
|
}
|
|
|
|
fn_monitor_query(){
|
|
# Will loop and query up to 5 times every 15 seconds.
|
|
# Query will wait up to 60 seconds to confirm server is down as server can become non-responsive during map changes.
|
|
totalseconds=0
|
|
for queryattempt in {1..5}; do
|
|
fn_print_dots "Querying port: ${querymethod}: ${ip}:${queryport} : ${totalseconds}/${queryattempt}: "
|
|
fn_print_querying_eol
|
|
fn_script_log_info "Querying port: ${querymethod}: ${ip}:${queryport} : ${queryattempt} : QUERYING"
|
|
fn_sleep_time
|
|
# querydelay
|
|
if [ "$(cat "${rootdir}/${lockselfname}")" -gt "$(date "+%s" -d "${querydelay} mins ago")" ]; then
|
|
fn_print_ok "Querying port: ${querymethod}: ${ip}:${queryport} : ${totalseconds}/${queryattempt}: "
|
|
fn_print_delay_eol_nl
|
|
fn_script_log_info "Querying port: ${querymethod}: ${ip}:${queryport} : ${queryattempt} : DELAY"
|
|
fn_script_log_info "Query bypassed: ${gameservername} started less than ${querydelay} minutes ago"
|
|
fn_sleep_time
|
|
monitorpass=1
|
|
core_exit.sh
|
|
# will use query method selected in fn_monitor_loop
|
|
# gamedig
|
|
elif [ "${querymethod}" == "gamedig" ]; then
|
|
query_gamedig.sh
|
|
# gsquery
|
|
elif [ "${querymethod}" == "gsquery" ]; then
|
|
fn_query_gsquery
|
|
#tcp query
|
|
elif [ "${querymethod}" == "tcp" ]; then
|
|
fn_query_tcp
|
|
fi
|
|
|
|
if [ "${querystatus}" == "0" ]; then
|
|
# Server query OK.
|
|
fn_print_ok "Querying port: ${querymethod}: ${ip}:${queryport} : ${totalseconds}/${queryattempt}: "
|
|
fn_print_ok_eol_nl
|
|
fn_script_log_pass "Querying port: ${querymethod}: ${ip}:${queryport} : ${queryattempt}: OK"
|
|
fn_sleep_time
|
|
monitorpass=1
|
|
# send LinuxGSM stats if monitor is OK.
|
|
if [ "${stats}" == "on" ]||[ "${stats}" == "y" ]; then
|
|
info_stats.sh
|
|
fi
|
|
if [ "${querystatus}" == "0" ]; then
|
|
# Add query data to log.
|
|
if [ -n "${gdname}" ]; then
|
|
fn_script_log_info "Server name: ${gdname}"
|
|
fi
|
|
if [ -n "${gdplayers}" ]; then
|
|
fn_script_log_info "Players: ${gdplayers}/${gdmaxplayers}"
|
|
fi
|
|
if [ -n "${gdmap}" ]; then
|
|
fn_script_log_info "Map: ${gdmap}"
|
|
fi
|
|
if [ -n "${gdgamemode}" ]; then
|
|
fn_script_log_info "Game Mode: ${gdgamemode}"
|
|
fi
|
|
fi
|
|
core_exit.sh
|
|
else
|
|
# Server query FAIL.
|
|
fn_print_fail "Querying port: ${querymethod}: ${ip}:${queryport} : ${totalseconds}/${queryattempt}: "
|
|
fn_print_fail_eol
|
|
fn_script_log_warn "Querying port: ${querymethod}: ${ip}:${queryport} : ${queryattempt}: FAIL"
|
|
fn_sleep_time
|
|
# Monitor will try gamedig (if supported) for first 30s then gsquery before restarting.
|
|
if [ "${querymethod}" == "gsquery" ]||[ "${querymethod}" == "tcp" ]; then
|
|
# gsquery will fail if longer than 60s
|
|
if [ "${totalseconds}" -ge "59" ]; then
|
|
# Monitor will FAIL if over 60s and trigger gane server reboot.
|
|
fn_print_fail "Querying port: ${querymethod}: ${ip}:${queryport} : ${totalseconds}/${queryattempt}: "
|
|
fn_print_fail_eol_nl
|
|
fn_script_log_warn "Querying port: ${querymethod}: ${ip}:${queryport} : ${queryattempt}: FAIL"
|
|
fn_sleep_time
|
|
# Send alert if enabled.
|
|
alert="restartquery"
|
|
alert.sh
|
|
command_restart.sh
|
|
core_exit.sh
|
|
fi
|
|
elif [ "${querymethod}" == "gamedig" ]; then
|
|
# gamedig will fail and try gsquery if longer than 30s
|
|
if [ "${totalseconds}" -ge "29" ]; then
|
|
break
|
|
fi
|
|
fi
|
|
|
|
# Second counter will wait for 15s before breaking loop.
|
|
for seconds in {1..15}; do
|
|
fn_print_fail "Querying port: ${querymethod}: ${ip}:${queryport} : ${totalseconds}/${queryattempt}: WAIT"
|
|
totalseconds=$((totalseconds + 1))
|
|
if [ "${seconds}" == "15" ]; then
|
|
break
|
|
fi
|
|
done
|
|
fi
|
|
done
|
|
}
|
|
|
|
fn_monitor_loop(){
|
|
# loop though query methods selected by querymode.
|
|
totalseconds=0
|
|
if [ "${querymode}" == "2" ]; then
|
|
local query_methods_array=( gamedig gsquery )
|
|
elif [ "${querymode}" == "3" ]; then
|
|
local query_methods_array=( gamedig )
|
|
elif [ "${querymode}" == "4" ]; then
|
|
local query_methods_array=( gsquery )
|
|
elif [ "${querymode}" == "5" ]; then
|
|
local query_methods_array=( tcp )
|
|
fi
|
|
for querymethod in "${query_methods_array[@]}"
|
|
do
|
|
# Will check if gamedig is installed and bypass if not.
|
|
if [ "${querymethod}" == "gamedig" ]; then
|
|
if [ "$(command -v gamedig 2>/dev/null)" ]&&[ "$(command -v jq 2>/dev/null)" ]; then
|
|
if [ -z "${monitorpass}" ]; then
|
|
fn_monitor_query
|
|
fi
|
|
else
|
|
fn_script_log_info "gamedig is not installed"
|
|
fn_script_log_info "https://docs.linuxgsm.com/requirements/gamedig"
|
|
fi
|
|
else
|
|
# will not query if query already passed.
|
|
if [ -z "${monitorpass}" ]; then
|
|
fn_monitor_query
|
|
fi
|
|
fi
|
|
done
|
|
}
|
|
|
|
monitorflag=1
|
|
check.sh
|
|
logs.sh
|
|
info_config.sh
|
|
info_parms.sh
|
|
|
|
# query pre-checks
|
|
fn_monitor_check_lockfile
|
|
fn_monitor_check_update
|
|
fn_monitor_check_session
|
|
fn_monitor_check_queryport
|
|
|
|
# Add a querydelay of 1 min if var missing.
|
|
if [ -z "${querydelay}" ]; then
|
|
querydelay="1"
|
|
fi
|
|
|
|
fn_monitor_loop
|
|
|
|
core_exit.sh
|
|
|