#!/usr/bin/env bash shopt -s nullglob # Utility functions ## Make setting default values a bit less awkward default() { declare -n _p=$1 if ! [[ "$_p" ]]; then _p=$2 fi } ## Die. Why not? die() { declare code=${1:-0} [[ "$2" ]] && printf '%s\n' "$2" exit "$code" } ## Run the command and wait for it to die svc() { declare job_pid svc::cleanup() { kill -n "$service_stop_signal" "$job_pid" pid_wait "$job_pid" && { rm -f $svc_pidfile $service_ready_flag } }; trap 'svc::cleanup' TERM "$@" & job_pid=$! printf '%s' "$job_pid" > "$svc_pidfile" wait "$job_pid" } ## Respawn respawn() { declare jobs job_pid respawn::cleanup() { jobs=( $(jobs -p) ) if [[ "$jobs" ]]; then kill -n 15 "${jobs[@]}" wait "${jobs[@]}" fi exit 0 }; trap 'respawn::cleanup' TERM respawn::sigpass() { declare sig=$1 pid=$2 kill -n "$sig" "$pid" } respawn::set_traps() { for s in "${service_signals[@]}"; do trap "respawn::sigpass $s \$job_pid" "$s" done }; respawn::set_traps while true; do exec "$@" & job_pid=$! while nullexec kill -n 0 "$job_pid"; do wait "$job_pid" done done } ## Run a command with its output discarded nullexec() { "$@" &>/dev/null } ## Wait for a pid to die pid_wait() { declare cnt=0 while nullexec kill -0 "$1"; do (( cnt >= (service_stop_timeout*10) )) && return 1 sleep 0.1 (( cnt++ )) done return 0 } ## See if NAME is a function is_function() { declare name=$1 name_tupe name_type=$( type -t "$name" ) if [[ $name_type == 'function' ]]; then return 0 fi return 1 } ## Simple timer timer() { declare cnt timeout=$1 shift while ! "$@"; do (( cnt >= (timeout*10) )) && return 1 sleep 0.1 (( cnt++ )) done return 0 } ## Is a service ready? is_ready() [[ -f "$service_ready_flag" ]] ## Wait for this service to get ready wait_ready() { timer "$service_ready_timeout" is_ready } ## Depend on other services to be started depend() { declare s for s in "$@"; do if ! "$0" "$s" qstatus; then nullexec "$0" "$s" start || { failed_deps+=( "$s" ) return 1 } fi done } ## Create tmpfiles mktmpfiles() { declare f for f in "${service_tmpfiles[@]}"; do IFS=':' read -r f_path f_type f_args <<< "$f" if ! [[ -e $f_path ]]; then case "$f_type" in symlink) ln -s "$f_args" "$f_path";; file|dir) IFS=':' read -r f_perms f_owner f_group <<< "$f_args" if [[ $f_type == 'file' ]]; then > "$f_path" chmod "${f_perms:-644}" "$f_path" elif [[ "$f_type" == 'dir' ]]; then mkdir -p -m "${f_perms:-755}" "$f_path" fi if [[ "$f_owner" || "$f_group" ]]; then chown "${f_owner:-root}:${f_group:-root}" "$f_path" fi;; esac fi done } ## Depend on other services to be ready depend_ready() { declare s depend "$@" || return 1 for s in "$@"; do "$0" "$s" wait_ready || { failed_deps+=( "$s" ) return 1 } done } # Super functions ## Start the service, write down the svc pid super_start() { (( service_running )) && return 3 [[ -f "${service_command[0]}" ]] || return 9 depend "${service_depends[@]}" || return 7 depend_ready "${service_depends_ready[@]}" || return 7 mktmpfiles || return 13 if (( service_managed )); then if (( service_respawn )); then svc respawn "${service_command[@]}" &>"$service_logfile" & else svc "${service_command[@]}" &>"$service_logfile" & fi if timer "$service_ready_timeout" ready; then printf '1' > "$service_ready_flag" else return 5 fi elif (( service_oneshot )); then "${service_command[@]}"; res=$? (( $res )) && return "$res" printf '1' > "$service_enabled_flag" else exec "${service_command[@]}" & fi return 0 } # A separate function for oneshot services super_oneshot() { (( service_enabled )) && return 3 } ## Reload the service ## Usually just sends HUP super_reload() { (( service_running )) || return 3 if (( service_managed )); then kill -n 1 "$service_pid" else kill -n "$service_reload_signal" "$service_pid" fi } ## Stop the service ## Returns: ## 3: Service is not running. super_stop() { if (( service_oneshot )); then (( service_enabled )) || return 3 rm -f "$service_enabled_flag" return 0 else (( service_running )) || return 3 nullexec kill -n "$service_stop_signal" "$service_pid" || return 1 pid_wait "$service_pid" || return 5 return 0 fi } # Overloadable functions start() { super_start; } stop() { super_stop; } reload() { super_reload; } restart() { "$0" "$service_name" stop "$0" "$service_name" start } logs() { cat "$service_logfile"; } ## Status is a bit of a special case. It's talkative. status() { (( service_running )) && return 0 (( service_enabled )) && return 0 return 1 } ## For use in scripts qstatus() { nullexec status; } ## By default there is no ready check ready() { :; } # Code main() { # Needs to be global declare -g service_pid # Let's set some defaults service_managed=1 if (( $UID )); then # XDG stuff default XDG_CONFIG_HOME "$HOME/.config" default XDG_RUNTIME_DIR "/run/user/$UID" cfgdir="$XDG_CONFIG_HOME/ssm" rundir="$XDG_RUNTIME_DIR/ssm" logdir="$HOME/log/ssm" else cfgdir='/etc/ssm' rundir='/run/ssm' logdir='/var/log/ssm' fi # Load custom functions for f in "$cfgdir/functions"/*; do source "$f" || die 9 "Failed to source functions from $f" done # Now create the needed runtime stuff for d in "$rundir" "$logdir"; do mkdir -p "$d" || die 3 "Failed to create runtime dir: $d" done # service_name is just $1 service_name=$1 # Semi-hardcoded stuff svc_pidfile="$rundir/$service_name.pid" # Get the service defaults [[ -f "$cfgdir/conf.d/$1" ]] && { source "$cfgdir/conf.d/$1" || die 5 "Failed to read service defaults: $cfgdir/conf.d/$1" } # Get the service config source "$cfgdir/init.d/$1" || die 7 "Failed to read the service config: $cfgdir/init.d/$1" # Legacy [[ "$service_args" ]] && service_command=( "${service_command[@]}" "${service_args[@]}" ) [[ "$service_respawn" == 'true' ]] && service_respawn=1 [[ "$service_type" == 'oneshot' ]] && { service_oneshot=1; service_managed=0; } [[ "$service_pidfile" ]] && service_managed=0 if ! (( service_managed )); then (( service_respawn )) && die 21 "Refusing to respawn a service that manages itself." fi # Service-level defaults default service_pidfile "$svc_pidfile" default service_logfile "$logdir/$service_name.log" default service_stop_timeout 30 default service_ready_timeout 15 default service_stop_signal 15 default service_reload_signal 1 default service_ready_flag "$rundir/$service_name.ready" default service_enabled_flag "$rundir/$service_name.enabled" # default does not support arrays [[ "$service_signals" ]] || service_signals=( 1 10 12 ) # Let's see if there's a PID if [[ -f "$service_pidfile" ]]; then service_pid=$(<$service_pidfile) # Let's see if it's running if nullexec kill -0 "$service_pid"; then service_running=1 fi fi # Maybe the service is enabled? if [[ -f "$service_enabled_flag" ]]; then # Yay, it is! service_enabled=1 fi # Check if action is even defined is_function "$2" || die 17 "Function $2 is not defined for $service_name." # Run pre_$action function if is_function "pre_$2"; then "pre_$2" || { printf 'pre_%s failed!\n' "$2" die 13 } fi # Run the function case "$2" in stop) printf 'Stopping %s... ' "$service_name" stop; res=$? case "$res" in 0) printf 'ok.\n';; 3) printf 'not running.\n' "$service_name";; 5) printf 'timed out.\n';; *) printf 'fail.\n';; esac ;; start) printf 'Starting %s... ' "$service_name" start; res=$? case "$res" in 0) printf 'ok.\n';; 3) printf 'already running.\n';; 5) printf 'readyness check timed out.\n';; 7) printf 'dependencies failed: %s.\n' "${failed_deps[@]}";; 9) printf 'service_command does not exist: %s.\n' "${service_command[0]}";; 13) printf '%s: failed to create temporary files.\n';; *) printf 'fail.\n';; esac ;; reload) printf 'Reloading %s... ' "$service_name" reload; res=$? case "$res" in 0) printf 'ok.\n';; *) printf 'fail.\n';; esac ;; status) status; res=$? case "$res" in 0) if (( service_oneshot )); then printf '%s is enabled.\n' "$service_name" else printf '%s is running.\n' "$service_name" fi;; 1) if (( service_oneshot )); then printf '%s is not enabled.\n' "$service_name" else printf '%s is not running.\n' "$service_name" fi;; *) printf '%s: status unknown.\n' "$service_name";; esac ;; logs) logs;; *) "$2"; res=$?;; esac (( res )) && return "$res" # Run post_$action function if is_function "post_$2"; then "post_$2" || { printf 'post_%s failed!\n' "$2" die 15 } fi } main "$@"