#!/bin/bash # Version: 0.4.0-daemon-runuser # Copyright (c) 2005-2022: # Darren 'Tadgy' Austin # Licensed under the terms of the GNU General Public License version 3. SERVICE_EXEC="%BUILD_PREFIX%/sbin/greylistd" SERVICE_ARGS=() SERVICE_RUNUSER="greylist" SERVICE_RUNDIR="/run/greylistd" SERVICE_LIBDIR="/var/lib/greylistd" DAEMON_EXEC="/usr/bin/daemon" DAEMON_ARGS=('-N' '-n' "${0##*rc.}" '-r' '-a' '60' '-A' '5' '-L' '3600' '-M' '3' '-l' 'daemon.err' '-b' 'daemon.debug' '-o' 'daemon.info') # Allow configuration in /etc/default to override. # Additional available variables: # SERVICE_ENVIRONMENT=() # Extra environment passed to $SERVICE_EXEC. Must be an array. # SERVICE_EXTRA_ARGS=() # Extra arguments passed to $SERVICE_EXEC. Must be an array. # SERVICE_RUNUSER="" # Username to run the service as. Default: User who starts the script. # DAEMON_ENVIRONMENT=() # Extra environment passed to $DAEMON_EXEC. Must be an array. # DAEMON_EXTRA_ARGS=() # Extra arguments passed to $DAEMON_EXEC. Must be an array. # SLAY_DELAY="" # Delay between the SIGTERM and SIGKILL on a 'stop'. Default: 2s. # RESTART_DELAY="" # Delay between stopping and starting on a 'restart'. Default: 2s. # shellcheck disable=SC1090 [[ -e "/etc/default/${0##*rc.}" ]] && { source "/etc/default/${0##*rc.}" || return 1 2>/dev/null || exit 1; } error() { printf "%s: %s\\n" "${BASH_SOURCE[0]##*/}" "$*" >&2 } checkconfigured() { # This function can be used to perform any pre-start tests; hopfully to insure the daemon # can start correctly, before actually trying to start it. A return value of 0 means the # tests were passed and the daemon should be started. Any other value prevents the # daemon from being started, and an error message will be emitted. [[ ! -e /etc/greylistd/config ]] && return 1 return 0 } checkstatus() { id "${SERVICE_RUNUSER:-$(whoami)}" >/dev/null 2>&1 || { error "invalid user: ${SERVICE_RUNUSER:-$(whoami)}" return 2 } if su - "${SERVICE_RUNUSER:-$(whoami)}" -c "\"$DAEMON_EXEC\" --running -n \"${0##*rc.}\""; then printf "%s: %s: %s\\n" "${BASH_SOURCE[0]##*/}" "${SERVICE_EXEC##*/}" "running" return 0 else printf "%s: %s: %s\\n" "${BASH_SOURCE[0]##*/}" "${SERVICE_EXEC##*/}" "stopped" return 1 fi } startdaemon() { local EXEC for EXEC in "$SERVICE_EXEC" "$DAEMON_EXEC"; do if [[ ! -e "$EXEC" ]]; then error "not found: $EXEC" return 2 elif [[ ! -x "$EXEC" ]]; then error "not executable: $EXEC" return 2 fi done id "${SERVICE_RUNUSER:-$(whoami)}" >/dev/null 2>&1 || { error "invalid user: ${SERVICE_RUNUSER:-$(whoami)}" return 2 } checkconfigured || { error "not started - pre-start checks failed" return 2 } [[ ! -e "$SERVICE_RUNDIR" ]] && { mkdir -p "$SERVICE_RUNDIR" 2>/dev/null || { error "failed to create directory: $SERVICE_RUNDIR"; return 2; }; } chown -R "$SERVICE_RUNUSER" "$SERVICE_RUNDIR" "$SERVICE_LIBDIR" >/dev/null || { error "failed to chmod {/run,/var/lib}/greylistd"; return 2; } su - "${SERVICE_RUNUSER:-$(whoami)}" -c "${DAEMON_ENVIRONMENT:+declare ${DAEMON_ENVIRONMENT[*]};} \"$DAEMON_EXEC\" ${DAEMON_ARGS[*]} \ ${DAEMON_EXTRA_ARGS[*]} $(printf -- "-e \"%s\" " "${SERVICE_ENVIRONMENT[@]}") -- \"$SERVICE_EXEC\" ${SERVICE_ARGS[*]} ${SERVICE_EXTRA_ARGS[*]}" # shellcheck disable=SC2181 if (( $? != 0 )); then error "error starting '${DAEMON_EXEC##*/}'" return 2 else return 0 fi } stopdaemon() { id "${SERVICE_RUNUSER:-$(whoami)}" >/dev/null 2>&1 || { error "invalid user: ${SERVICE_RUNUSER:-$(whoami)}" return 2 } su - "${SERVICE_RUNUSER:-$(whoami)}" -c "\"$DAEMON_EXEC\" -n \"${0##*rc.}\" --stop 2>/dev/null" sleep "${SLAY_DELAY:-2}" checkstatus >/dev/null && { error "failed to stop gracefully - slaying" su - "${SERVICE_RUNUSER:-$(whoami)}" -c "\"$DAEMON_EXEC\" -n \"${0##*rc.}\" --signal=kill 2>/dev/null" } return 0 } case "$1" in 'start') if checkstatus >/dev/null; then error "${SERVICE_EXEC##*/}: already running" printf " %s\\n" "Try: ${BASH_SOURCE[0]} status" >&2 RET=1 else startdaemon RET=$? fi ;; 'stop') if ! checkstatus >/dev/null; then error "${SERVICE_EXEC##*/}: not running" printf " %s\\n" "Try: ${BASH_SOURCE[0]} status" >&2 RET=1 else stopdaemon RET=$? fi ;; 'restart') checkstatus >/dev/null (( $? != 3 )) && { stopdaemon >/dev/null 2>&1 sleep "${RESTART_DELAY:-2}" } startdaemon RET=$? ;; 'status') checkstatus RET=$? ;; *) printf "%s\\n" "Usage: ${BASH_SOURCE[0]} " >&2 RET=1 ;; esac return $RET 2>/dev/null || exit $RET