2015-06-24 21:36:14 +10:00
#!/bin/sh
# Exit on 1st error
set -e
2015-09-02 09:38:04 +10:00
# NFS exports file. Some code below keeps a cache of output derived
# from exportfs(8). When this file is updated the cache is invalid
# and needs to be regenerated.
nfs_exports_file="${CTDB_NFS_EXPORTS_FILE:-/var/lib/nfs/etab}"
2024-03-27 14:24:09 +11:00
# Do not set CTDB_NFS_DISTRO_STYLE - it isn't a configuration
# variable, just a hook for testing. To change the style, edit the
# default value below.
2019-03-26 15:05:58 +11:00
nfs_distro_style="${CTDB_NFS_DISTRO_STYLE:-systemd-redhat}"
2019-03-26 14:49:49 +11:00
2022-03-18 12:52:10 +11:00
# As above, edit the default value below. CTDB_SYS_ETCDIR is a
# test variable only.
etc_dir="${CTDB_SYS_ETCDIR:-/etc}"
2022-03-18 12:47:10 +11:00
# A value of "AUTO" for any service means that service is usually
# automatically started and stopped by one of the other services.
# Such services will still be restarted by hand on failure, if
# configured to do so. This allows services that should not be
# running to be set to "".
2019-03-26 14:49:49 +11:00
case "$nfs_distro_style" in
2019-03-20 17:45:10 +11:00
systemd-*)
# Defaults
nfs_service="nfs-server"
nfs_lock_service="rpc-statd"
nfs_mountd_service="nfs-mountd"
nfs_status_service="rpc-statd"
nfs_rquotad_service="rpc-rquotad"
2022-03-18 12:52:10 +11:00
nfs_config="${etc_dir}/sysconfig/nfs"
2019-03-20 17:45:10 +11:00
nfs_rquotad_config="" # Not use with systemd, restart via service
case "$nfs_distro_style" in
2022-03-18 10:53:45 +11:00
*-redhat | *-suse)
2019-03-20 17:45:10 +11:00
: # Defaults only
;;
*-debian)
nfs_rquotad_service="quotarpc"
;;
*)
echo "Internal error"
exit 1
2022-03-18 10:53:45 +11:00
;;
2019-03-20 17:45:10 +11:00
esac
;;
2019-03-26 14:49:49 +11:00
sysvinit-*)
# Defaults
nfs_service="nfs"
2022-03-18 12:47:10 +11:00
nfs_lock_service="AUTO"
nfs_mountd_service="AUTO"
nfs_status_service="AUTO"
nfs_rquotad_service="AUTO"
2022-03-18 12:52:10 +11:00
nfs_config="${etc_dir}/sysconfig/nfs"
2019-03-20 17:35:44 +11:00
nfs_rquotad_config="$nfs_config"
2019-03-26 14:49:49 +11:00
case "$nfs_distro_style" in
*-redhat)
nfs_lock_service="nfslock"
;;
*-suse)
nfs_service="nfsserver"
;;
*-debian)
nfs_service="nfs-kernel-server"
2022-03-18 12:52:10 +11:00
nfs_config="${etc_dir}/default/nfs-kernel-server"
nfs_rquotad_config="${etc_dir}/default/quota"
2019-03-26 14:49:49 +11:00
;;
*)
echo "Internal error"
exit 1
2022-03-18 10:53:45 +11:00
;;
2019-03-26 14:49:49 +11:00
esac
;;
*)
echo "Internal error"
exit 1
2022-03-18 10:53:45 +11:00
;;
2019-03-26 14:49:49 +11:00
esac
2015-06-24 21:36:14 +10:00
# Override for unit testing
2022-03-18 10:53:45 +11:00
if [ -z "$PROCFS_PATH" ]; then
PROCFS_PATH="/proc"
2015-06-24 21:36:14 +10:00
fi
##################################################
2022-03-18 10:53:45 +11:00
usage()
2015-06-24 21:36:14 +10:00
{
2022-03-18 10:53:45 +11:00
_c=$(basename "$0")
cat <<EOF
2015-06-24 21:36:14 +10:00
usage: $_c { shutdown | startup }
$_c { stop | start } { nfs | nlockmgr }
$_c { monitor-list-shares | monitor-post }
2015-07-14 12:11:39 +10:00
$_c { register }
2015-06-24 21:36:14 +10:00
EOF
2022-03-18 10:53:45 +11:00
exit 1
2015-06-24 21:36:14 +10:00
}
2019-03-20 16:11:32 +11:00
##################################################
2022-03-18 10:53:45 +11:00
nfs_load_config()
2019-03-20 16:11:32 +11:00
{
_config="${1:-${nfs_config}}"
2022-03-18 10:53:45 +11:00
if [ -r "$_config" ]; then
2019-03-20 16:11:32 +11:00
. "$_config"
fi
}
2015-06-24 21:36:14 +10:00
2022-03-18 12:47:10 +11:00
##################################################
service_is_auto_started()
{
[ "$1" = "AUTO" ]
}
service_is_defined()
{
_service="$1"
[ -n "$_service" ] && ! service_is_auto_started "$_service"
}
service_if_defined()
{
_service="$1"
_action="$2"
if service_is_defined "$_service"; then
service "$_service" "$_action"
fi
}
2015-06-24 21:36:14 +10:00
##################################################
2022-03-18 10:41:26 +11:00
# Overall NFS service stop and start
2015-06-24 21:36:14 +10:00
2022-03-18 10:41:26 +11:00
nfs_service_stop()
2015-06-24 21:36:14 +10:00
{
2022-03-18 12:47:10 +11:00
service_if_defined "$nfs_rquotad_service" stop
2019-03-21 15:30:00 +11:00
2022-03-18 10:41:26 +11:00
service "$nfs_service" stop
2019-03-21 15:30:00 +11:00
2022-03-18 12:47:10 +11:00
service_if_defined "$nfs_lock_service" stop
2015-06-24 21:36:14 +10:00
}
2022-03-18 10:41:26 +11:00
nfs_service_start()
2015-06-24 21:36:14 +10:00
{
2022-03-18 12:47:10 +11:00
service_if_defined "$nfs_lock_service" start
2019-03-21 15:30:00 +11:00
2022-03-18 10:41:26 +11:00
service "$nfs_service" start
2019-03-21 15:30:00 +11:00
2022-03-18 12:47:10 +11:00
service_if_defined "$nfs_rquotad_service" start
2015-06-24 21:36:14 +10:00
}
##################################################
# service "stop" and "start" options for restarting
2022-05-23 14:21:37 +10:00
manual_stop()
2015-06-24 21:36:14 +10:00
{
2019-03-18 14:42:36 +11:00
case "$1" in
2022-05-23 14:21:37 +10:00
mountd)
killall -q -9 rpc.mountd
;;
rquotad)
killall -q -9 rpc.rquotad
;;
status)
killall -q -9 rpc.statd
;;
*)
echo "$0: Internal error - invalid call to manual_stop()"
exit 1
;;
esac
}
service_or_manual_stop()
{
_rpc_service="$1"
_system_service="$2"
2022-03-18 12:47:10 +11:00
if service_is_defined "$_system_service"; then
2022-05-23 14:21:37 +10:00
service "$_system_service" stop
2022-03-18 12:47:10 +11:00
elif service_is_auto_started "$_system_service"; then
2022-05-23 14:21:37 +10:00
manual_stop "$_rpc_service"
fi
}
service_stop()
{
_rpc_service="$1"
case "$_rpc_service" in
2015-06-24 21:36:14 +10:00
nfs)
2019-03-18 14:42:36 +11:00
echo 0 >"${PROCFS_PATH}/fs/nfsd/threads"
2022-03-18 10:41:26 +11:00
nfs_service_stop >/dev/null 2>&1 || true
2019-03-18 14:42:36 +11:00
pkill -9 nfsd
;;
2015-06-24 21:36:14 +10:00
nlockmgr)
2022-03-18 12:47:10 +11:00
if service_is_defined "$nfs_lock_service" ; then
2022-03-18 10:33:05 +11:00
service "$nfs_lock_service" stop >/dev/null 2>&1 || true
else
service "$nfs_service" stop >/dev/null 2>&1 || true
fi
2019-03-18 14:42:36 +11:00
;;
2019-03-20 17:35:44 +11:00
mountd)
2022-05-23 14:21:37 +10:00
service_or_manual_stop "$_rpc_service" "$nfs_mountd_service"
2019-03-20 17:35:44 +11:00
;;
rquotad)
2022-05-23 14:21:37 +10:00
service_or_manual_stop "$_rpc_service" "$nfs_rquotad_service"
2019-03-20 17:35:44 +11:00
;;
status)
2022-05-23 14:21:37 +10:00
service_or_manual_stop "$_rpc_service" "$nfs_status_service"
2019-03-20 17:35:44 +11:00
;;
2015-06-24 21:36:14 +10:00
*)
2019-03-18 14:42:36 +11:00
usage
2022-03-18 10:53:45 +11:00
;;
2019-03-18 14:42:36 +11:00
esac
2015-06-24 21:36:14 +10:00
}
2022-05-23 14:21:37 +10:00
manual_start()
2015-06-24 21:36:14 +10:00
{
2019-03-18 14:42:36 +11:00
case "$1" in
2019-03-20 17:35:44 +11:00
mountd)
nfs_load_config
2022-03-18 10:53:45 +11:00
if [ -z "$RPCMOUNTDOPTS" ]; then
2019-03-20 17:35:44 +11:00
RPCMOUNTDOPTS="${MOUNTD_PORT:+-p }$MOUNTD_PORT"
fi
# shellcheck disable=SC2086
rpc.mountd $RPCMOUNTDOPTS
;;
rquotad)
nfs_load_config "$nfs_rquotad_config"
2022-03-18 10:53:45 +11:00
if [ -z "$RPCRQUOTADOPTS" ]; then
2019-03-20 17:35:44 +11:00
RPCRQUOTADOPTS="${RQUOTAD_PORT:+-p }$RQUOTAD_PORT"
fi
# shellcheck disable=SC2086
rpc.rquotad $RPCRQUOTADOPTS
;;
status)
nfs_load_config
# Red Hat uses STATDARG, Debian uses STATDOPTS
opts="${STATDARG:-${STATDOPTS:-''}}"
2022-03-18 10:53:45 +11:00
if [ -z "$opts" ]; then
2019-03-20 17:35:44 +11:00
# shellcheck disable=SC2086
set -- \
2022-03-18 10:53:45 +11:00
${STATD_HA_CALLOUT:+-H} $STATD_HA_CALLOUT \
${STATD_HOSTNAME:+-n} $STATD_HOSTNAME \
${STATD_PORT:+-p} $STATD_PORT \
${STATD_OUTGOING_PORT:+-o} $STATD_OUTGOING_PORT
2019-03-20 17:35:44 +11:00
opts="$*"
fi
# shellcheck disable=SC2086
rpc.statd $opts
;;
2022-05-23 14:21:37 +10:00
*)
echo "$0: Internal error - invalid call to manual_start()"
exit 1
;;
esac
}
service_or_manual_start()
{
_rpc_service="$1"
_system_service="$2"
2022-03-18 12:47:10 +11:00
if service_is_defined "$_system_service"; then
2022-05-23 14:21:37 +10:00
service "$_system_service" start
2022-03-18 12:47:10 +11:00
elif service_is_auto_started "$_system_service"; then
2022-05-23 14:21:37 +10:00
manual_start "$_rpc_service"
fi
}
service_start()
{
_rpc_service="$1"
case "$_rpc_service" in
nfs)
nfs_service_start
;;
nlockmgr)
2022-03-18 12:47:10 +11:00
if service_is_defined "$nfs_lock_service" ; then
2022-05-23 14:21:37 +10:00
service "$nfs_lock_service" start
else
service "$nfs_service" start
fi
;;
mountd)
service_or_manual_start "$_rpc_service" "$nfs_mountd_service"
;;
rquotad)
service_or_manual_start "$_rpc_service" "$nfs_rquotad_service"
;;
status)
service_or_manual_start "$_rpc_service" "$nfs_status_service"
;;
2015-06-24 21:36:14 +10:00
*)
2019-03-18 14:42:36 +11:00
usage
2022-03-18 10:53:45 +11:00
;;
2019-03-18 14:42:36 +11:00
esac
2015-06-24 21:36:14 +10:00
}
##################################################
# service init startup and final shutdown
2022-03-18 10:53:45 +11:00
nfs_shutdown()
2015-06-24 21:36:14 +10:00
{
2022-03-18 10:41:26 +11:00
nfs_service_stop
2015-06-24 21:36:14 +10:00
}
2022-03-18 10:53:45 +11:00
nfs_startup()
2015-06-24 21:36:14 +10:00
{
2022-03-18 10:41:26 +11:00
nfs_service_stop || true
nfs_service_start
2022-03-18 10:53:45 +11:00
_f="${PROCFS_PATH}/sys/net/ipv4/tcp_tw_recycle"
if [ -f "$_f" ]; then
echo 1 >"$_f"
fi
2015-06-24 21:36:14 +10:00
}
##################################################
# monitor-post support
2022-03-18 10:53:45 +11:00
nfs_check_thread_count()
2015-06-24 21:36:14 +10:00
{
2022-03-18 10:53:45 +11:00
# Load NFS configuration to get desired number of threads.
nfs_load_config
# If $RPCNFSDCOUNT/$USE_KERNEL_NFSD_NUMBER isn't set then we could
# guess the default from the initscript. However, let's just
# assume that those using the default don't care about the number
# of threads and that they have switched on this feature in error.
_configured_threads="${RPCNFSDCOUNT:-${USE_KERNEL_NFSD_NUMBER}}"
if [ -z "$_configured_threads" ] && type nfsconf >/dev/null 2>&1; then
_configured_threads=$(nfsconf --get nfsd threads) || true
fi
[ -n "$_configured_threads" ] || return 0
_threads_file="${PROCFS_PATH}/fs/nfsd/threads"
# nfsd should be running the configured number of threads. If
# there are a different number of threads then tell nfsd the
# correct number.
2022-03-18 13:40:20 +11:00
read -r _running_threads <"$_threads_file" || {
2022-03-18 10:53:45 +11:00
echo "WARNING: Reading \"${_threads_file}\" unexpectedly failed"
exit 0
}
# Intentionally not arithmetic comparison - avoids extra errors
# when above read fails in an unexpected way...
if [ "$_running_threads" != "$_configured_threads" ]; then
echo "Attempting to correct number of nfsd threads from ${_running_threads} to ${_configured_threads}"
echo "$_configured_threads" >"$_threads_file"
fi
2015-06-24 21:36:14 +10:00
}
##################################################
# list share directories
2022-03-18 10:53:45 +11:00
nfs_monitor_list_shares()
2015-06-24 21:36:14 +10:00
{
2022-03-18 10:53:45 +11:00
_cache_file="${CTDB_NFS_CALLOUT_STATE_DIR}/list_shares_cache"
# -nt operator is well supported in Linux: dash, bash, ksh, ...
# shellcheck disable=SC2039,SC3013
if [ ! -r "$nfs_exports_file" ] || [ ! -r "$_cache_file" ] ||
[ "$nfs_exports_file" -nt "$_cache_file" ]; then
mkdir -p "$CTDB_NFS_CALLOUT_STATE_DIR"
# We could just use the contents of $nfs_exports_file.
# However, let's regard that file as internal to NFS and use
# exportfs, which is the public API.
if ! _exports=$(exportfs -v); then
echo "WARNING: failed to run exportfs to list NFS shares" >&2
return
fi
2015-09-02 09:38:04 +10:00
2022-03-18 10:53:45 +11:00
echo "$_exports" |
grep '^/' |
sed -e 's@[[:space:]][[:space:]]*[^[:space:]()][^[:space:]()]*([^[:space:]()][^[:space:]()]*)$@@' |
sort -u >"$_cache_file"
fi
2015-09-02 09:38:04 +10:00
2022-03-18 10:53:45 +11:00
cat "$_cache_file"
2015-06-24 21:36:14 +10:00
}
##################################################
2022-03-18 10:53:45 +11:00
nfs_register()
2015-07-14 12:13:58 +10:00
{
2022-03-18 10:53:45 +11:00
cat <<EOF
2015-07-14 12:13:58 +10:00
shutdown
startup
stop
start
monitor-list-shares
monitor-post
EOF
}
##################################################
2015-06-24 21:36:14 +10:00
case "$1" in
2022-03-18 10:53:45 +11:00
shutdown)
2015-06-24 21:36:14 +10:00
nfs_shutdown
;;
2022-03-18 10:53:45 +11:00
startup)
2015-06-24 21:36:14 +10:00
nfs_startup
;;
2022-03-18 10:53:45 +11:00
stop)
2015-06-24 21:36:14 +10:00
service_stop "$2"
;;
2022-03-18 10:53:45 +11:00
start)
2015-06-24 21:36:14 +10:00
service_start "$2"
;;
2022-03-18 10:53:45 +11:00
monitor-list-shares)
2015-06-24 21:36:14 +10:00
nfs_monitor_list_shares
;;
2022-03-18 10:53:45 +11:00
monitor-post)
2015-06-24 21:36:14 +10:00
nfs_check_thread_count
;;
2022-03-18 10:53:45 +11:00
register)
2015-07-14 12:13:58 +10:00
nfs_register
2015-07-14 12:11:39 +10:00
;;
2023-10-26 23:38:22 -07:00
monitor-pre | releaseip | takeip | startipreallocate)
2015-06-24 21:36:14 +10:00
# Not required/implemented
:
;;
2022-03-18 10:53:45 +11:00
*)
2015-06-24 21:36:14 +10:00
usage
2022-03-18 10:53:45 +11:00
;;
2015-06-24 21:36:14 +10:00
esac