mirror of
https://github.com/rvalitov/zabbix-php-fpm.git
synced 2023-11-05 03:30:27 +01:00
Fix ShellCheck errors and warnings (#25)
* [fix] comparison operators * [fix] indents, globbing issue, and other shellcheck related fixes
This commit is contained in:
parent
b4e4e172f3
commit
ccf91cbf17
@ -3,78 +3,78 @@
|
||||
#https://github.com/rvalitov/zabbix-php-fpm
|
||||
#This script scans local machine for active PHP-FPM pools and returns them as a list in JSON format
|
||||
|
||||
S_PS=`type -P ps`
|
||||
S_GREP=`type -P grep`
|
||||
S_AWK=`type -P awk`
|
||||
S_SORT=`type -P sort`
|
||||
S_HEAD=`type -P head`
|
||||
S_LSOF=`type -P lsof`
|
||||
S_JQ=`type -P jq`
|
||||
S_DIRNAME=`type -P dirname`
|
||||
S_CAT=`type -P cat`
|
||||
S_BASH=`type -P bash`
|
||||
S_ECHO=`type -P echo`
|
||||
S_PRINTF=`type -P printf`
|
||||
S_WHOAMI=`type -P whoami`
|
||||
S_PS=$(type -P ps)
|
||||
S_GREP=$(type -P grep)
|
||||
S_AWK=$(type -P awk)
|
||||
S_SORT=$(type -P sort)
|
||||
S_HEAD=$(type -P head)
|
||||
S_LSOF=$(type -P lsof)
|
||||
S_JQ=$(type -P jq)
|
||||
S_DIRNAME=$(type -P dirname)
|
||||
S_CAT=$(type -P cat)
|
||||
S_BASH=$(type -P bash)
|
||||
S_ECHO=$(type -P echo)
|
||||
S_PRINTF=$(type -P printf)
|
||||
S_WHOAMI=$(type -P whoami)
|
||||
|
||||
if [[ ! -f $S_PS ]]; then
|
||||
${S_ECHO} "Utility 'ps' not found. Please, install it first."
|
||||
exit 1
|
||||
${S_ECHO} "Utility 'ps' not found. Please, install it first."
|
||||
exit 1
|
||||
fi
|
||||
if [[ ! -f $S_GREP ]]; then
|
||||
${S_ECHO} "Utility 'grep' not found. Please, install it first."
|
||||
exit 1
|
||||
${S_ECHO} "Utility 'grep' not found. Please, install it first."
|
||||
exit 1
|
||||
fi
|
||||
if [[ ! -f $S_AWK ]]; then
|
||||
${S_ECHO} "Utility 'awk' not found. Please, install it first."
|
||||
exit 1
|
||||
${S_ECHO} "Utility 'awk' not found. Please, install it first."
|
||||
exit 1
|
||||
fi
|
||||
if [[ ! -f $S_SORT ]]; then
|
||||
${S_ECHO} "Utility 'sort' not found. Please, install it first."
|
||||
exit 1
|
||||
${S_ECHO} "Utility 'sort' not found. Please, install it first."
|
||||
exit 1
|
||||
fi
|
||||
if [[ ! -f $S_HEAD ]]; then
|
||||
${S_ECHO} "Utility 'head' not found. Please, install it first."
|
||||
exit 1
|
||||
${S_ECHO} "Utility 'head' not found. Please, install it first."
|
||||
exit 1
|
||||
fi
|
||||
if [[ ! -f $S_LSOF ]]; then
|
||||
${S_ECHO} "Utility 'lsof' not found. Please, install it first."
|
||||
exit 1
|
||||
${S_ECHO} "Utility 'lsof' not found. Please, install it first."
|
||||
exit 1
|
||||
fi
|
||||
if [[ ! -f $S_JQ ]]; then
|
||||
${S_ECHO} "Utility 'jq' not found. Please, install it first."
|
||||
exit 1
|
||||
${S_ECHO} "Utility 'jq' not found. Please, install it first."
|
||||
exit 1
|
||||
fi
|
||||
if [[ ! -f ${S_DIRNAME} ]]; then
|
||||
${S_ECHO} "Utility 'dirname' not found. Please, install it first."
|
||||
exit 1
|
||||
${S_ECHO} "Utility 'dirname' not found. Please, install it first."
|
||||
exit 1
|
||||
fi
|
||||
if [[ ! -f ${S_CAT} ]]; then
|
||||
${S_ECHO} "Utility 'cat' not found. Please, install it first."
|
||||
exit 1
|
||||
${S_ECHO} "Utility 'cat' not found. Please, install it first."
|
||||
exit 1
|
||||
fi
|
||||
if [[ ! -f ${S_BASH} ]]; then
|
||||
${S_ECHO} "Utility 'bash' not found. Please, install it first."
|
||||
exit 1
|
||||
${S_ECHO} "Utility 'bash' not found. Please, install it first."
|
||||
exit 1
|
||||
fi
|
||||
if [[ ! -f ${S_PRINTF} ]]; then
|
||||
${S_ECHO} "Utility 'printf' not found. Please, install it first."
|
||||
exit 1
|
||||
${S_ECHO} "Utility 'printf' not found. Please, install it first."
|
||||
exit 1
|
||||
fi
|
||||
if [[ ! -f ${S_WHOAMI} ]]; then
|
||||
${S_ECHO} "Utility 'whoami' not found. Please, install it first."
|
||||
exit 1
|
||||
${S_ECHO} "Utility 'whoami' not found. Please, install it first."
|
||||
exit 1
|
||||
fi
|
||||
|
||||
STATUS_PATH="/php-fpm-status"
|
||||
DEBUG_MODE=""
|
||||
ACTIVE_USER=`${S_WHOAMI}`
|
||||
ACTIVE_USER=$(${S_WHOAMI})
|
||||
|
||||
# Prints a string on screen. Works only if debug mode is enabled.
|
||||
function PrintDebug(){
|
||||
if [[ ! -z $DEBUG_MODE ]] && [[ ! -z $1 ]]; then
|
||||
${S_ECHO} $1
|
||||
fi
|
||||
function PrintDebug() {
|
||||
if [[ -n $DEBUG_MODE ]] && [[ -n $1 ]]; then
|
||||
${S_ECHO} "$1"
|
||||
fi
|
||||
}
|
||||
|
||||
# Encodes input data to JSON and saves it to result string
|
||||
@ -82,21 +82,21 @@ function PrintDebug(){
|
||||
# - pool name
|
||||
# - pool socket
|
||||
# Function returns 1 if all OK, and 0 otherwise.
|
||||
function EncodeToJson(){
|
||||
POOL_NAME=$1
|
||||
POOL_SOCKET=$2
|
||||
if [[ -z ${POOL_NAME} ]] || [[ -z ${POOL_SOCKET} ]]; then
|
||||
return 0
|
||||
fi
|
||||
function EncodeToJson() {
|
||||
POOL_NAME=$1
|
||||
POOL_SOCKET=$2
|
||||
if [[ -z ${POOL_NAME} ]] || [[ -z ${POOL_SOCKET} ]]; then
|
||||
return 0
|
||||
fi
|
||||
|
||||
JSON_POOL=`${S_ECHO} -n "$POOL_NAME" | ${S_JQ} -aR .`
|
||||
JSON_SOCKET=`${S_ECHO} -n "$POOL_SOCKET" | ${S_JQ} -aR .`
|
||||
if [[ ${POOL_FIRST} == 1 ]]; then
|
||||
RESULT_DATA="$RESULT_DATA,"
|
||||
fi
|
||||
RESULT_DATA="$RESULT_DATA{\"{#POOLNAME}\":$JSON_POOL,\"{#POOLSOCKET}\":$JSON_SOCKET}"
|
||||
POOL_FIRST=1
|
||||
return 1
|
||||
JSON_POOL=$(${S_ECHO} -n "$POOL_NAME" | ${S_JQ} -aR .)
|
||||
JSON_SOCKET=$(${S_ECHO} -n "$POOL_SOCKET" | ${S_JQ} -aR .)
|
||||
if [[ ${POOL_FIRST} == 1 ]]; then
|
||||
RESULT_DATA="$RESULT_DATA,"
|
||||
fi
|
||||
RESULT_DATA="$RESULT_DATA{\"{#POOLNAME}\":$JSON_POOL,\"{#POOLSOCKET}\":$JSON_SOCKET}"
|
||||
POOL_FIRST=1
|
||||
return 1
|
||||
}
|
||||
|
||||
# Checks if selected pool is in cache.
|
||||
@ -104,21 +104,22 @@ function EncodeToJson(){
|
||||
# - pool name
|
||||
# - pool socket
|
||||
# Function returns 1 if the pool is in cache, and 0 otherwise.
|
||||
function IsInCache(){
|
||||
SEARCH_NAME=$1
|
||||
SEARCH_SOCKET=$2
|
||||
if [[ -z ${SEARCH_NAME} ]] || [[ -z ${SEARCH_SOCKET} ]]; then
|
||||
return 0
|
||||
fi
|
||||
for CACHE_ITEM in "${NEW_CACHE[@]}"
|
||||
do
|
||||
ITEM_NAME=`${S_ECHO} "$CACHE_ITEM" | ${S_AWK} '{print $1}'`
|
||||
ITEM_SOCKET=`${S_ECHO} "$CACHE_ITEM" | ${S_AWK} '{print $2}'`
|
||||
if [[ ${ITEM_NAME} == ${SEARCH_NAME} ]] && [[ ${ITEM_SOCKET} == ${SEARCH_SOCKET} ]]; then
|
||||
return 1
|
||||
fi
|
||||
done
|
||||
function IsInCache() {
|
||||
SEARCH_NAME=$1
|
||||
SEARCH_SOCKET=$2
|
||||
if [[ -z ${SEARCH_NAME} ]] || [[ -z ${SEARCH_SOCKET} ]]; then
|
||||
return 0
|
||||
fi
|
||||
for CACHE_ITEM in "${NEW_CACHE[@]}"; do
|
||||
# shellcheck disable=SC2016
|
||||
ITEM_NAME=$(${S_ECHO} "$CACHE_ITEM" | ${S_AWK} '{print $1}')
|
||||
# shellcheck disable=SC2016
|
||||
ITEM_SOCKET=$(${S_ECHO} "$CACHE_ITEM" | ${S_AWK} '{print $2}')
|
||||
if [[ ${ITEM_NAME} == "${SEARCH_NAME}" ]] && [[ ${ITEM_SOCKET} == "${SEARCH_SOCKET}" ]]; then
|
||||
return 1
|
||||
fi
|
||||
done
|
||||
return 0
|
||||
}
|
||||
|
||||
# Validates the specified pool by getting its status and working with cache.
|
||||
@ -128,76 +129,76 @@ function IsInCache(){
|
||||
# 1 if the pool is OK and is ondemand and is not in cache
|
||||
# 2 if the pool is OK and is ondemand and is in cache
|
||||
# 3 if the pool is OK and is not ondemand and is not in cache
|
||||
function ProcessPool(){
|
||||
POOL_NAME=$1
|
||||
POOL_SOCKET=$2
|
||||
if [[ -z ${POOL_NAME} ]] || [[ -z ${POOL_SOCKET} ]]; then
|
||||
return 0
|
||||
function ProcessPool() {
|
||||
POOL_NAME=$1
|
||||
POOL_SOCKET=$2
|
||||
if [[ -z ${POOL_NAME} ]] || [[ -z ${POOL_SOCKET} ]]; then
|
||||
return 0
|
||||
fi
|
||||
|
||||
IsInCache "${POOL_NAME}" "${POOL_SOCKET}"
|
||||
FOUND=$?
|
||||
if [[ ${FOUND} == 1 ]]; then
|
||||
return 2
|
||||
fi
|
||||
|
||||
STATUS_JSON=$(${S_BASH} "${STATUS_SCRIPT}" "${POOL_SOCKET}" ${STATUS_PATH})
|
||||
EXIT_CODE=$?
|
||||
if [[ ${EXIT_CODE} == 0 ]]; then
|
||||
# The exit code is OK, let's check the JSON data
|
||||
# JSON data example:
|
||||
# {"pool":"www2","process manager":"ondemand","start time":1578181845,"start since":117,"accepted conn":3,"listen queue":0,"max listen queue":0,"listen queue len":0,"idle processes":0,"active processes":1,"total processes":1,"max active processes":1,"max children reached":0,"slow requests":0}
|
||||
# We use basic regular expression here, i.e. we need to use \+ and not escape { and }
|
||||
if [[ -n $(${S_ECHO} "${STATUS_JSON}" | ${S_GREP} -G '^{.*\"pool\":\".\+\".*,\"process manager\":\".\+\".*}$') ]]; then
|
||||
PrintDebug "Status data for pool $POOL_NAME, socket $POOL_SOCKET, status path $STATUS_PATH is valid"
|
||||
# Checking if we have ondemand pool
|
||||
if [[ -n $(${S_ECHO} "${STATUS_JSON}" | ${S_GREP} -F '"process manager":"ondemand"') ]]; then
|
||||
PrintDebug "Detected pool's process manager is ondemand, it needs to be cached"
|
||||
NEW_CACHE+=("$POOL_NAME $POOL_SOCKET")
|
||||
return 1
|
||||
fi
|
||||
PrintDebug "Detected pool's process manager is NOT ondemand, it will not be cached"
|
||||
return 3
|
||||
fi
|
||||
|
||||
IsInCache ${POOL_NAME} ${POOL_SOCKET}
|
||||
FOUND=$?
|
||||
if [[ ${FOUND} == 1 ]]; then
|
||||
return 2
|
||||
fi
|
||||
|
||||
STATUS_JSON=`${S_BASH} ${STATUS_SCRIPT} ${POOL_SOCKET} ${STATUS_PATH}`
|
||||
EXIT_CODE=$?
|
||||
if [[ ${EXIT_CODE} == 0 ]]; then
|
||||
# The exit code is OK, let's check the JSON data
|
||||
# JSON data example:
|
||||
# {"pool":"www2","process manager":"ondemand","start time":1578181845,"start since":117,"accepted conn":3,"listen queue":0,"max listen queue":0,"listen queue len":0,"idle processes":0,"active processes":1,"total processes":1,"max active processes":1,"max children reached":0,"slow requests":0}
|
||||
# We use basic regular expression here, i.e. we need to use \+ and not escape { and }
|
||||
if [[ ! -z `${S_ECHO} ${STATUS_JSON} | ${S_GREP} -G '^{.*\"pool\":\".\+\".*,\"process manager\":\".\+\".*}$'` ]]; then
|
||||
PrintDebug "Status data for pool $POOL_NAME, socket $POOL_SOCKET, status path $STATUS_PATH is valid"
|
||||
# Checking if we have ondemand pool
|
||||
if [[ ! -z `${S_ECHO} ${STATUS_JSON} | ${S_GREP} -F '"process manager":"ondemand"'` ]]; then
|
||||
PrintDebug "Detected pool's process manager is ondemand, it needs to be cached"
|
||||
NEW_CACHE+=("$POOL_NAME $POOL_SOCKET")
|
||||
return 1
|
||||
fi
|
||||
PrintDebug "Detected pool's process manager is NOT ondemand, it will not be cached"
|
||||
return 3
|
||||
fi
|
||||
|
||||
PrintDebug "Failed to validate status data for pool $POOL_NAME, socket $POOL_SOCKET, status path $STATUS_PATH"
|
||||
if [[ ! -z ${STATUS_JSON} ]]; then
|
||||
PrintDebug "Status script returned: $STATUS_JSON"
|
||||
fi
|
||||
return 0
|
||||
fi
|
||||
PrintDebug "Failed to get status for pool $POOL_NAME, socket $POOL_SOCKET, status path $STATUS_PATH"
|
||||
if [[ ! -z ${STATUS_JSON} ]]; then
|
||||
PrintDebug "Status script returned: $STATUS_JSON"
|
||||
PrintDebug "Failed to validate status data for pool $POOL_NAME, socket $POOL_SOCKET, status path $STATUS_PATH"
|
||||
if [[ -n ${STATUS_JSON} ]]; then
|
||||
PrintDebug "Status script returned: $STATUS_JSON"
|
||||
fi
|
||||
return 0
|
||||
fi
|
||||
PrintDebug "Failed to get status for pool $POOL_NAME, socket $POOL_SOCKET, status path $STATUS_PATH"
|
||||
if [[ -n ${STATUS_JSON} ]]; then
|
||||
PrintDebug "Status script returned: $STATUS_JSON"
|
||||
fi
|
||||
return 0
|
||||
}
|
||||
|
||||
for ARG in "$@"; do
|
||||
if [[ ${ARG} == "debug" ]]; then
|
||||
DEBUG_MODE="1"
|
||||
${S_ECHO} "Debug mode enabled"
|
||||
elif [[ ${ARG} == /* ]]; then
|
||||
STATUS_PATH=${ARG}
|
||||
PrintDebug "Argument $ARG is interpreted as status path"
|
||||
else
|
||||
PrintDebug "Argument $ARG is unknown and skipped"
|
||||
fi
|
||||
if [[ ${ARG} == "debug" ]]; then
|
||||
DEBUG_MODE="1"
|
||||
${S_ECHO} "Debug mode enabled"
|
||||
elif [[ ${ARG} == /* ]]; then
|
||||
STATUS_PATH=${ARG}
|
||||
PrintDebug "Argument $ARG is interpreted as status path"
|
||||
else
|
||||
PrintDebug "Argument $ARG is unknown and skipped"
|
||||
fi
|
||||
done
|
||||
PrintDebug "Current user is $ACTIVE_USER"
|
||||
PrintDebug "Status path to be used: $STATUS_PATH"
|
||||
|
||||
LOCAL_DIR=`${S_DIRNAME} $0`
|
||||
LOCAL_DIR=$(${S_DIRNAME} "$0")
|
||||
CACHE_FILE="$LOCAL_DIR/php_fpm.cache"
|
||||
STATUS_SCRIPT="$LOCAL_DIR/zabbix_php_fpm_status.sh"
|
||||
PrintDebug "Local directory is $LOCAL_DIR"
|
||||
if [[ ! -f ${STATUS_SCRIPT} ]]; then
|
||||
${S_ECHO} "Helper script $STATUS_SCRIPT not found"
|
||||
exit 1
|
||||
${S_ECHO} "Helper script $STATUS_SCRIPT not found"
|
||||
exit 1
|
||||
fi
|
||||
if [[ ! -r ${STATUS_SCRIPT} ]]; then
|
||||
${S_ECHO} "Helper script $STATUS_SCRIPT is not readable"
|
||||
exit 1
|
||||
${S_ECHO} "Helper script $STATUS_SCRIPT is not readable"
|
||||
exit 1
|
||||
fi
|
||||
PrintDebug "Helper script $STATUS_SCRIPT is reachable"
|
||||
|
||||
@ -206,130 +207,134 @@ PrintDebug "Helper script $STATUS_SCRIPT is reachable"
|
||||
CACHE=()
|
||||
NEW_CACHE=()
|
||||
if [[ -r ${CACHE_FILE} ]]; then
|
||||
PrintDebug "Reading cache file $CACHE_FILE..."
|
||||
mapfile -t CACHE < <( ${S_CAT} ${CACHE_FILE} )
|
||||
PrintDebug "Reading cache file $CACHE_FILE..."
|
||||
mapfile -t CACHE < <(${S_CAT} "${CACHE_FILE}")
|
||||
else
|
||||
PrintDebug "Cache file $CACHE_FILE not found, skipping..."
|
||||
PrintDebug "Cache file $CACHE_FILE not found, skipping..."
|
||||
fi
|
||||
|
||||
mapfile -t PS_LIST < <( $S_PS ax | $S_GREP -F "php-fpm: pool " | $S_GREP -F -v "grep" )
|
||||
POOL_LIST=`${S_PRINTF} '%s\n' "${PS_LIST[@]}" | $S_AWK '{print $NF}' | $S_SORT -u`
|
||||
mapfile -t PS_LIST < <($S_PS ax | $S_GREP -F "php-fpm: pool " | $S_GREP -F -v "grep")
|
||||
# shellcheck disable=SC2016
|
||||
POOL_LIST=$(${S_PRINTF} '%s\n' "${PS_LIST[@]}" | $S_AWK '{print $NF}' | $S_SORT -u)
|
||||
POOL_FIRST=0
|
||||
#We store the resulting JSON data for Zabbix in the following var:
|
||||
RESULT_DATA="{\"data\":["
|
||||
while IFS= read -r line
|
||||
do
|
||||
POOL_PID=`${S_PRINTF} '%s\n' "${PS_LIST[@]}" | $S_GREP -F -w "php-fpm: pool $line" | $S_HEAD -1 | $S_AWK '{print $1}'`
|
||||
if [[ ! -z $POOL_PID ]]; then
|
||||
#We search for socket or IP address and port
|
||||
#Socket example:
|
||||
#php-fpm7. 25897 root 9u unix 0x000000006509e31f 0t0 58381847 /run/php/php7.3-fpm.sock type=STREAM
|
||||
#IP example:
|
||||
#php-fpm7. 1110 default 0u IPv4 15760 0t0 TCP localhost:8002 (LISTEN)
|
||||
while IFS= read -r line; do
|
||||
# shellcheck disable=SC2016
|
||||
POOL_PID=$(${S_PRINTF} '%s\n' "${PS_LIST[@]}" | $S_GREP -F -w "php-fpm: pool $line" | $S_HEAD -1 | $S_AWK '{print $1}')
|
||||
if [[ -n $POOL_PID ]]; then
|
||||
#We search for socket or IP address and port
|
||||
#Socket example:
|
||||
#php-fpm7. 25897 root 9u unix 0x000000006509e31f 0t0 58381847 /run/php/php7.3-fpm.sock type=STREAM
|
||||
#IP example:
|
||||
#php-fpm7. 1110 default 0u IPv4 15760 0t0 TCP localhost:8002 (LISTEN)
|
||||
|
||||
#Check all matching processes, because we may face a redirect (or a symlink?), examples:
|
||||
#php-fpm7. 1203 www-data 5u unix 0x000000006509e31f 0t0 15068771 type=STREAM
|
||||
#php-fpm7. 6086 www-data 11u IPv6 21771 0t0 TCP *:9000 (LISTEN)
|
||||
#php-fpm7. 1203 www-data 8u IPv4 15070917 0t0 TCP localhost.localdomain:23054->localhost.localdomain:postgresql (ESTABLISHED)
|
||||
#More info at https://github.com/rvalitov/zabbix-php-fpm/issues/12
|
||||
#Check all matching processes, because we may face a redirect (or a symlink?), examples:
|
||||
#php-fpm7. 1203 www-data 5u unix 0x000000006509e31f 0t0 15068771 type=STREAM
|
||||
#php-fpm7. 6086 www-data 11u IPv6 21771 0t0 TCP *:9000 (LISTEN)
|
||||
#php-fpm7. 1203 www-data 8u IPv4 15070917 0t0 TCP localhost.localdomain:23054->localhost.localdomain:postgresql (ESTABLISHED)
|
||||
#More info at https://github.com/rvalitov/zabbix-php-fpm/issues/12
|
||||
|
||||
PrintDebug "Started analysis of pool $line, PID $POOL_PID"
|
||||
#Extract only important information:
|
||||
POOL_PARAMS_LIST=`$S_LSOF -p $POOL_PID 2>/dev/null | $S_GREP -w -e "unix" -e "TCP"`
|
||||
FOUND_POOL=""
|
||||
while IFS= read -r pool
|
||||
do
|
||||
if [[ ! -z $pool ]]; then
|
||||
if [[ -z $FOUND_POOL ]]; then
|
||||
PrintDebug "Checking process: $pool"
|
||||
POOL_TYPE=`${S_ECHO} "${pool}" | $S_AWK '{print $5}'`
|
||||
POOL_SOCKET=`${S_ECHO} "${pool}" | $S_AWK '{print $9}'`
|
||||
if [[ ! -z $POOL_TYPE ]] && [[ ! -z $POOL_SOCKET ]]; then
|
||||
if [[ $POOL_TYPE == "unix" ]]; then
|
||||
#We have a socket here, test if it's actually a socket:
|
||||
if [[ -S $POOL_SOCKET ]]; then
|
||||
PrintDebug "Found socket $POOL_SOCKET"
|
||||
ProcessPool ${line} ${POOL_SOCKET}
|
||||
POOL_STATUS=$?
|
||||
if [[ ${POOL_STATUS} > 0 ]]; then
|
||||
FOUND_POOL="1"
|
||||
PrintDebug "Success: socket $POOL_SOCKET returned valid status data"
|
||||
else
|
||||
PrintDebug "Error: socket $POOL_SOCKET didn't return valid data"
|
||||
fi
|
||||
else
|
||||
PrintDebug "Error: specified socket $POOL_SOCKET is not valid"
|
||||
fi
|
||||
elif [[ $POOL_TYPE == "IPv4" ]] || [[ $POOL_TYPE == "IPv6" ]]; then
|
||||
#We have a TCP connection here, check it:
|
||||
CONNECTION_TYPE=`${S_ECHO} "${pool}" | $S_AWK '{print $8}'`
|
||||
if [[ $CONNECTION_TYPE == "TCP" ]]; then
|
||||
#The connection must have state LISTEN:
|
||||
LISTEN=`${S_ECHO} ${pool} | $S_GREP -F -w "(LISTEN)"`
|
||||
if [[ ! -z $LISTEN ]]; then
|
||||
#Check and replace * to localhost if it's found. Asterisk means that the PHP listens on
|
||||
#all interfaces.
|
||||
POOL_SOCKET=`${S_ECHO} -n ${POOL_SOCKET/*:/localhost:}`
|
||||
PrintDebug "Found TCP connection $POOL_SOCKET"
|
||||
ProcessPool ${line} ${POOL_SOCKET}
|
||||
POOL_STATUS=$?
|
||||
if [[ ${POOL_STATUS} > 0 ]]; then
|
||||
FOUND_POOL="1"
|
||||
PrintDebug "Success: TCP connection $POOL_SOCKET returned valid status data"
|
||||
else
|
||||
PrintDebug "Error: TCP connection $POOL_SOCKET didn't return valid data"
|
||||
fi
|
||||
else
|
||||
PrintDebug "Warning: expected connection state must be LISTEN, but it was not detected"
|
||||
fi
|
||||
else
|
||||
PrintDebug "Warning: expected connection type is TCP, but found $CONNECTION_TYPE"
|
||||
fi
|
||||
else
|
||||
PrintDebug "Unsupported type $POOL_TYPE, skipping"
|
||||
fi
|
||||
else
|
||||
PrintDebug "Warning: pool type or socket is empty"
|
||||
fi
|
||||
PrintDebug "Started analysis of pool $line, PID $POOL_PID"
|
||||
#Extract only important information:
|
||||
POOL_PARAMS_LIST=$($S_LSOF -p "$POOL_PID" 2>/dev/null | $S_GREP -w -e "unix" -e "TCP")
|
||||
FOUND_POOL=""
|
||||
while IFS= read -r pool; do
|
||||
if [[ -n $pool ]]; then
|
||||
if [[ -z $FOUND_POOL ]]; then
|
||||
PrintDebug "Checking process: $pool"
|
||||
# shellcheck disable=SC2016
|
||||
POOL_TYPE=$(${S_ECHO} "${pool}" | $S_AWK '{print $5}')
|
||||
# shellcheck disable=SC2016
|
||||
POOL_SOCKET=$(${S_ECHO} "${pool}" | $S_AWK '{print $9}')
|
||||
if [[ -n $POOL_TYPE ]] && [[ -n $POOL_SOCKET ]]; then
|
||||
if [[ $POOL_TYPE == "unix" ]]; then
|
||||
#We have a socket here, test if it's actually a socket:
|
||||
if [[ -S $POOL_SOCKET ]]; then
|
||||
PrintDebug "Found socket $POOL_SOCKET"
|
||||
ProcessPool "${line}" "${POOL_SOCKET}"
|
||||
POOL_STATUS=$?
|
||||
if [[ ${POOL_STATUS} -gt 0 ]]; then
|
||||
FOUND_POOL="1"
|
||||
PrintDebug "Success: socket $POOL_SOCKET returned valid status data"
|
||||
else
|
||||
PrintDebug "Pool already found, skipping process: $pool"
|
||||
PrintDebug "Error: socket $POOL_SOCKET didn't return valid data"
|
||||
fi
|
||||
else
|
||||
PrintDebug "Error: specified socket $POOL_SOCKET is not valid"
|
||||
fi
|
||||
elif [[ $POOL_TYPE == "IPv4" ]] || [[ $POOL_TYPE == "IPv6" ]]; then
|
||||
#We have a TCP connection here, check it:
|
||||
# shellcheck disable=SC2016
|
||||
CONNECTION_TYPE=$(${S_ECHO} "${pool}" | $S_AWK '{print $8}')
|
||||
if [[ $CONNECTION_TYPE == "TCP" ]]; then
|
||||
#The connection must have state LISTEN:
|
||||
LISTEN=$(${S_ECHO} "${pool}" | $S_GREP -F -w "(LISTEN)")
|
||||
if [[ -n $LISTEN ]]; then
|
||||
#Check and replace * to localhost if it's found. Asterisk means that the PHP listens on
|
||||
#all interfaces.
|
||||
POOL_SOCKET=$(${S_ECHO} -n "${POOL_SOCKET/*:/localhost:}")
|
||||
PrintDebug "Found TCP connection $POOL_SOCKET"
|
||||
ProcessPool "${line}" "${POOL_SOCKET}"
|
||||
POOL_STATUS=$?
|
||||
if [[ ${POOL_STATUS} -gt 0 ]]; then
|
||||
FOUND_POOL="1"
|
||||
PrintDebug "Success: TCP connection $POOL_SOCKET returned valid status data"
|
||||
else
|
||||
PrintDebug "Error: TCP connection $POOL_SOCKET didn't return valid data"
|
||||
fi
|
||||
else
|
||||
PrintDebug "Warning: expected connection state must be LISTEN, but it was not detected"
|
||||
fi
|
||||
else
|
||||
PrintDebug "Warning: expected connection type is TCP, but found $CONNECTION_TYPE"
|
||||
fi
|
||||
else
|
||||
PrintDebug "Error: failed to get process information. Probably insufficient privileges. Use sudo or run this script under root."
|
||||
PrintDebug "Unsupported type $POOL_TYPE, skipping"
|
||||
fi
|
||||
done <<< "$POOL_PARAMS_LIST"
|
||||
|
||||
if [[ ! -z ${FOUND_POOL} ]]; then
|
||||
EncodeToJson ${line} ${POOL_SOCKET}
|
||||
else
|
||||
PrintDebug "Warning: pool type or socket is empty"
|
||||
fi
|
||||
else
|
||||
PrintDebug "Error: failed to discover information for pool $line"
|
||||
PrintDebug "Pool already found, skipping process: $pool"
|
||||
fi
|
||||
else
|
||||
PrintDebug "Error: failed to get process information. Probably insufficient privileges. Use sudo or run this script under root."
|
||||
fi
|
||||
done <<<"$POOL_PARAMS_LIST"
|
||||
|
||||
if [[ -n ${FOUND_POOL} ]]; then
|
||||
EncodeToJson "${line}" "${POOL_SOCKET}"
|
||||
else
|
||||
PrintDebug "Error: failed to find PID for pool $line"
|
||||
PrintDebug "Error: failed to discover information for pool $line"
|
||||
fi
|
||||
done <<< "$POOL_LIST"
|
||||
else
|
||||
PrintDebug "Error: failed to find PID for pool $line"
|
||||
fi
|
||||
done <<<"$POOL_LIST"
|
||||
|
||||
PrintDebug "Processing pools from old cache..."
|
||||
for CACHE_ITEM in "${CACHE[@]}"
|
||||
do
|
||||
ITEM_NAME=`${S_ECHO} "$CACHE_ITEM" | ${S_AWK} '{print $1}'`
|
||||
ITEM_SOCKET=`${S_ECHO} "$CACHE_ITEM" | ${S_AWK} '{print $2}'`
|
||||
ProcessPool ${ITEM_NAME} ${ITEM_SOCKET}
|
||||
POOL_STATUS=$?
|
||||
if [[ ${POOL_STATUS} == "1" ]]; then
|
||||
# This is a new pool and we must add it
|
||||
EncodeToJson ${ITEM_NAME} ${ITEM_SOCKET}
|
||||
fi
|
||||
for CACHE_ITEM in "${CACHE[@]}"; do
|
||||
# shellcheck disable=SC2016
|
||||
ITEM_NAME=$(${S_ECHO} "$CACHE_ITEM" | ${S_AWK} '{print $1}')
|
||||
# shellcheck disable=SC2016
|
||||
ITEM_SOCKET=$(${S_ECHO} "$CACHE_ITEM" | ${S_AWK} '{print $2}')
|
||||
ProcessPool "${ITEM_NAME}" "${ITEM_SOCKET}"
|
||||
POOL_STATUS=$?
|
||||
if [[ ${POOL_STATUS} == "1" ]]; then
|
||||
# This is a new pool and we must add it
|
||||
EncodeToJson "${ITEM_NAME}" "${ITEM_SOCKET}"
|
||||
fi
|
||||
done
|
||||
|
||||
if [[ -f ${CACHE_FILE} ]] && [[ ! -w ${CACHE_FILE} ]]; then
|
||||
${S_ECHO} "Error: write permission is not granted to user $ACTIVE_USER for cache file $CACHE_FILE"
|
||||
exit 1
|
||||
${S_ECHO} "Error: write permission is not granted to user $ACTIVE_USER for cache file $CACHE_FILE"
|
||||
exit 1
|
||||
fi
|
||||
|
||||
PrintDebug "Saving new cache file $CACHE_FILE..."
|
||||
${S_PRINTF} "%s\n" "${NEW_CACHE[@]}" > ${CACHE_FILE}
|
||||
${S_PRINTF} "%s\n" "${NEW_CACHE[@]}" >"${CACHE_FILE}"
|
||||
|
||||
RESULT_DATA="$RESULT_DATA]}"
|
||||
PrintDebug "Resulting JSON data for Zabbix:"
|
||||
${S_ECHO} -n $RESULT_DATA
|
||||
${S_ECHO} -n "$RESULT_DATA"
|
||||
|
@ -3,37 +3,39 @@
|
||||
#https://github.com/rvalitov/zabbix-php-fpm
|
||||
#Script gets status of PHP-FPM pool
|
||||
|
||||
S_FCGI=`type -P cgi-fcgi`
|
||||
S_GREP=`type -P grep`
|
||||
S_ECHO=`type -P echo`
|
||||
S_FCGI=$(type -P cgi-fcgi)
|
||||
S_GREP=$(type -P grep)
|
||||
S_ECHO=$(type -P echo)
|
||||
|
||||
if [[ ! -f $S_FCGI ]]; then
|
||||
echo "Utility 'cgi-fcgi' not found. Please, install it first."
|
||||
exit 1
|
||||
echo "Utility 'cgi-fcgi' not found. Please, install it first."
|
||||
exit 1
|
||||
fi
|
||||
|
||||
if [[ ! -f $S_GREP ]]; then
|
||||
echo "Utility 'grep' not found. Please, install it first."
|
||||
exit 1
|
||||
echo "Utility 'grep' not found. Please, install it first."
|
||||
exit 1
|
||||
fi
|
||||
|
||||
if [[ -z $1 ]] || [[ -z $2 ]]; then
|
||||
echo "No input data specified"
|
||||
echo "Usage: $0 php-path status"
|
||||
echo "where:"
|
||||
echo "php-path - path to socket file, for example, /var/lib/php7.3-fpm/web1.sock"
|
||||
echo "or IP and port of the PHP-FPM, for example, 127.0.0.1:9000"
|
||||
echo "status - path configured in pm.status of PHP-FPM"
|
||||
exit 1
|
||||
echo "No input data specified"
|
||||
echo "Usage: $0 php-path status"
|
||||
echo "where:"
|
||||
echo "php-path - path to socket file, for example, /var/lib/php7.3-fpm/web1.sock"
|
||||
echo "or IP and port of the PHP-FPM, for example, 127.0.0.1:9000"
|
||||
echo "status - path configured in pm.status of PHP-FPM"
|
||||
exit 1
|
||||
fi
|
||||
|
||||
POOL_URL=$1
|
||||
POOL_PATH=$2
|
||||
#connecting to socket or address, https://easyengine.io/tutorials/php/directly-connect-php-fpm/
|
||||
PHP_STATUS=`SCRIPT_NAME=$POOL_PATH \
|
||||
SCRIPT_FILENAME=$POOL_PATH \
|
||||
QUERY_STRING=json \
|
||||
REQUEST_METHOD=GET \
|
||||
$S_FCGI -bind -connect $POOL_URL 2>/dev/null`
|
||||
#connecting to socket or address, https://easyengine.io/tutorials/php/directly-connect-php-fpm/
|
||||
PHP_STATUS=$(
|
||||
SCRIPT_NAME=$POOL_PATH \
|
||||
SCRIPT_FILENAME=$POOL_PATH \
|
||||
QUERY_STRING=json \
|
||||
REQUEST_METHOD=GET \
|
||||
$S_FCGI -bind -connect "$POOL_URL" 2>/dev/null
|
||||
)
|
||||
$S_ECHO "$PHP_STATUS" | $S_GREP "{"
|
||||
exit 0
|
||||
|
Loading…
Reference in New Issue
Block a user