#!/bin/sh /etc/rc.common
|
|
# Copyright © 2012 OpenWrt.org
|
|
#
|
|
# This is free software, licensed under the GNU General Public License v2.
|
|
# See /LICENSE for more information.
|
|
#
|
|
START=70
|
|
STOP=30
|
|
|
|
USERS_C=/var/etc/nut/upsd.users
|
|
UPSD_C=/var/etc/nut/upsd.conf
|
|
UPS_C=/var/etc/nut/ups.conf
|
|
|
|
USE_PROCD=1
|
|
|
|
get_write_driver_config() {
|
|
local cfg="$1"
|
|
local var="$2"
|
|
local def="$3"
|
|
local flag="$4"
|
|
local val
|
|
|
|
[ -z "$flag" ] && {
|
|
config_get val "$cfg" "$var" "$def"
|
|
[ -n "$val" ] && [ "$val" != "0" ] && echo "$var = $val" >>"$UPS_C"
|
|
}
|
|
|
|
[ -n "$flag" ] && {
|
|
config_get_bool val "$cfg" "$var" "$def"
|
|
[ "$val" = 1 ] && echo "$var" >>"$UPS_C"
|
|
}
|
|
}
|
|
|
|
upsd_statepath() {
|
|
local statepath
|
|
|
|
config_get statepath upsd statepath /var/run/nut
|
|
STATEPATH="$statepath"
|
|
}
|
|
|
|
upsd_runas() {
|
|
local runas
|
|
|
|
[ -n "$RUNAS" ] && return 0
|
|
|
|
config_get runas upsd runas nut
|
|
RUNAS="$runas"
|
|
}
|
|
|
|
listen_address() {
|
|
local cfg="$1"
|
|
|
|
config_get address "$cfg" address "::1"
|
|
config_get port "$cfg" port
|
|
echo "LISTEN $address $port" >>"$UPSD_C"
|
|
}
|
|
|
|
upsd_config() {
|
|
local cfg="$1"
|
|
local maxage maxconn certfile runas statepath
|
|
|
|
# Note runas support requires you make sure USB device file is readable by
|
|
# the runas user
|
|
config_get runas "$cfg" runas nut
|
|
RUNAS="$runas"
|
|
|
|
config_get statepath "$cfg" statepath /var/run/nut
|
|
STATEPATH="$statepath"
|
|
|
|
config_get maxage "$cfg" maxage
|
|
[ -n "$maxage" ] && echo "MAXAGE $maxage" >>"$UPSD_C"
|
|
|
|
[ -n "$statepath" ] && echo "STATEPATH $statepath" >>"$UPSD_C"
|
|
|
|
config_get maxconn "$cfg" maxconn
|
|
[ -n "$maxconn" ] && echo "MAXCONN $maxconn" >>"$UPSD_C"
|
|
|
|
#NOTE: certs only apply to SSL-enabled version
|
|
config_get certfile "$cfg" certfile
|
|
[ -n "$certfile" ] && echo "CERTFILE $certfile" >>"$UPSD_C"
|
|
}
|
|
|
|
nut_user_add() {
|
|
local cfg="$1"
|
|
local a
|
|
local val
|
|
|
|
config_get val "$cfg" username "$1"
|
|
echo "[$val]" >> "$USERS_C"
|
|
|
|
config_get val "$cfg" password
|
|
echo " password = $val" >> "$USERS_C"
|
|
|
|
config_get val "$cfg" actions
|
|
for a in $val; do
|
|
echo " actions = $a" >> "$USERS_C"
|
|
done
|
|
|
|
instcmd() {
|
|
local val="$1"
|
|
echo " instcmds = $val" >> "$USERS_C"
|
|
}
|
|
|
|
config_list_foreach "$cfg" instcmd instcmd
|
|
|
|
config_get val "$cfg" upsmon
|
|
if [ -n "$val" ]; then
|
|
echo " upsmon $val" >> "$USERS_C"
|
|
fi
|
|
}
|
|
|
|
build_server_config() {
|
|
mkdir -p "$(dirname "$UPSD_C")"
|
|
chmod 0640 "$UPS_C"
|
|
rm -f "$USERS_C"
|
|
rm -f "$UPSD_C"
|
|
rm -f /var/etc/nut/nut.conf
|
|
|
|
echo "# Config file automatically generated from UCI config" > "$USERS_C"
|
|
echo "# Config file automatically generated from UCI config" > "$UPSD_C"
|
|
|
|
config_foreach nut_user_add user
|
|
config_foreach listen_address listen_address
|
|
config_foreach upsd_config upsd
|
|
echo "MODE=netserver" >>/var/etc/nut/nut.conf
|
|
|
|
chmod 0640 "$USERS_C"
|
|
chmod 0640 "$UPSD_C"
|
|
chmod 0644 /var/etc/nut/nut.conf
|
|
|
|
[ -d "${STATEPATH}" ] || {
|
|
mkdir -p "${STATEPATH}"
|
|
chmod 0750 "${STATEPATH}"
|
|
}
|
|
|
|
if [ -n "$RUNAS" ]; then
|
|
chown "$RUNAS":"$(id -gn "$RUNAS")" "${STATEPATH}"
|
|
chgrp "$(id -gn "$RUNAS")" "$USERS_C"
|
|
chgrp "$(id -gn "$RUNAS")" "$UPSD_C"
|
|
fi
|
|
haveserver=1
|
|
}
|
|
|
|
build_driver_config() {
|
|
local cfg="$1"
|
|
|
|
echo "[$cfg]" >>"$UPS_C"
|
|
|
|
get_write_driver_config "$cfg" bus
|
|
get_write_driver_config "$cfg" community
|
|
get_write_driver_config "$cfg" desc
|
|
get_write_driver_config "$cfg" driver "usbhid-ups"
|
|
get_write_driver_config "$cfg" ignorelb 0 1
|
|
get_write_driver_config "$cfg" interruptonly 0 1
|
|
get_write_driver_config "$cfg" interruptsize
|
|
get_write_driver_config "$cfg" maxreport
|
|
get_write_driver_config "$cfg" maxstartdelay
|
|
get_write_driver_config "$cfg" mfr
|
|
get_write_driver_config "$cfg" model
|
|
get_write_driver_config "$cfg" nolock 0 1
|
|
get_write_driver_config "$cfg" notransferoids 0 1
|
|
get_write_driver_config "$cfg" offdelay
|
|
get_write_driver_config "$cfg" ondelay
|
|
get_write_driver_config "$cfg" pollfreq
|
|
get_write_driver_config "$cfg" port "auto"
|
|
get_write_driver_config "$cfg" product
|
|
get_write_driver_config "$cfg" productid
|
|
get_write_driver_config "$cfg" retrydelay
|
|
get_write_driver_config "$cfg" sdorder
|
|
get_write_driver_config "$cfg" sdtime
|
|
get_write_driver_config "$cfg" serial
|
|
get_write_driver_config "$cfg" snmp_version
|
|
get_write_driver_config "$cfg" snmp_retries
|
|
get_write_driver_config "$cfg" snmp_timeout
|
|
get_write_driver_config "$cfg" synchronous
|
|
get_write_driver_config "$cfg" vendor
|
|
get_write_driver_config "$cfg" vendorid
|
|
|
|
defoverride() {
|
|
local overvar="$1"
|
|
local defover="$2"
|
|
local overtype="$(echo "$overvar" | tr '.' '_')"
|
|
local overval
|
|
|
|
config_get overval "${defover}_${overtype}" value
|
|
[ -n "$overval" ] && echo "${defover}.${overvar} = $overval" >>"$UPS_C"
|
|
}
|
|
|
|
config_list_foreach "$cfg" override defoverride override
|
|
config_list_foreach "$cfg" default defoverride default
|
|
|
|
other() {
|
|
local othervar="$1"
|
|
local othervarflag="$2"
|
|
local otherval
|
|
|
|
if [ "$othervarflag" = "otherflag" ]; then
|
|
config_get_bool otherval "${othervarflag}_${othervar}" value
|
|
[ "$otherval" = "1" ] && echo "${othervar}" >>"$UPS_C"
|
|
else
|
|
config_get otherval "${othervarflag}_${othervar}" value
|
|
[ -n "$otherval" ] && echo "${othervar} = $otherval" >>"$UPS_C"
|
|
fi
|
|
}
|
|
|
|
config_list_foreach "$cfg" other other
|
|
config_list_foreach "$cfg" other otherflag
|
|
echo "" >>$UPS_C
|
|
havedriver=1
|
|
}
|
|
|
|
build_global_driver_config() {
|
|
local cfg="$1"
|
|
|
|
# Global driver config
|
|
get_write_driver_config "$cfg" chroot
|
|
get_write_driver_config "$cfg" driverpath
|
|
get_write_driver_config "$cfg" maxstartdelay
|
|
get_write_driver_config "$cfg" maxretry
|
|
get_write_driver_config "$cfg" retrydelay
|
|
get_write_driver_config "$cfg" pollinterval
|
|
get_write_driver_config "$cfg" synchronous
|
|
config_get runas "$cfg" user nut
|
|
RUNAS="$runas"
|
|
|
|
echo "" >>"$UPS_C"
|
|
}
|
|
|
|
build_config() {
|
|
local STATEPATH=/var/run/nut
|
|
|
|
mkdir -p "$(dirname "$UPS_C")"
|
|
rm -f "$UPS_C"
|
|
echo "# Config file automatically generated from UCI config" > "$UPS_C"
|
|
chmod 0640 "$UPS_C"
|
|
|
|
config_load nut_server
|
|
|
|
upsd_runas
|
|
config_foreach build_global_driver_config driver_global
|
|
config_foreach build_driver_config driver
|
|
upsd_statepath
|
|
build_server_config
|
|
[ -n "$RUNAS" ] && chgrp "$(id -gn "$RUNAS")" "$UPS_C"
|
|
}
|
|
|
|
start_driver_instance() {
|
|
local cfg="$1"
|
|
local requested="$2"
|
|
local driver
|
|
local STATEPATH=/var/run/nut
|
|
local RUNAS=nut
|
|
|
|
[ "$havedriver" != 1 ] && return
|
|
|
|
# If wanting a specific instance, only start it
|
|
if [ "$requested" != "$cfg" ] && [ "$request" != "" ]; then
|
|
return 0
|
|
fi
|
|
|
|
mkdir -p "$(dirname "$UPS_C")"
|
|
chmod 0755 "$UPS_C"
|
|
|
|
upsd_statepath
|
|
build_config
|
|
|
|
# Avoid hotplug inadvertenly restarting driver during
|
|
# forced shutdown
|
|
[ -f /var/run/killpower ] && return 0
|
|
if [ -d /var/run/nut ] && [ -f /var/run/nut/disable-hotplug ]; then
|
|
return 0
|
|
fi
|
|
|
|
if [ -n "$RUNAS" ]; then
|
|
chown "$RUNAS":"$(id -gn "$RUNAS")" "${STATEPATH}"
|
|
chgrp "$(id -gn "$RUNAS")" "$UPS_C"
|
|
fi
|
|
|
|
config_get driver "$cfg" driver "usbhid-ups"
|
|
procd_open_instance "$cfg"
|
|
procd_set_param respawn
|
|
procd_set_param stderr 0
|
|
procd_set_param stdout 1
|
|
procd_set_param command /lib/nut/"${driver}" -D -a "$cfg" ${RUNAS:+-u "$RUNAS"}
|
|
procd_close_instance
|
|
}
|
|
|
|
interface_triggers() {
|
|
local action="$1"
|
|
local triggerlist trigger
|
|
|
|
config_get triggerlist upsd triggerlist
|
|
|
|
. /lib/functions/network.sh
|
|
|
|
if [ -n "$triggerlist" ]; then
|
|
for trigger in $triggerlist; do
|
|
if [ "$action" = "add_trigger" ]; then
|
|
procd_add_interface_trigger "interface.*" "$trigger" /etc/init.d/nut-server reload
|
|
else
|
|
network_is_up "$trigger" && return 0
|
|
fi
|
|
done
|
|
else
|
|
if [ "$action" = "add_trigger" ]; then
|
|
procd_add_raw_trigger "interface.*.up" 2000 /etc/init.d/nut-server reload
|
|
else
|
|
ubus call network.device status | grep -q '"up": true' && return 0
|
|
fi
|
|
fi
|
|
[ "$action" = "add_trigger" ] || return 1
|
|
}
|
|
|
|
start_server_instance() {
|
|
local cfg="$1"
|
|
|
|
[ "$haveserver" != 1 ] && return
|
|
interface_triggers "check_interface_up" || return
|
|
|
|
procd_open_instance "$cfg"
|
|
procd_set_param respawn
|
|
procd_set_param stderr 0
|
|
procd_set_param stdout 1
|
|
procd_set_param command /usr/sbin/upsd -D ${RUNAS:+-u "$RUNAS"}
|
|
procd_close_instance
|
|
}
|
|
|
|
start_service() {
|
|
local STATEPATH=/var/run/nut
|
|
|
|
# Avoid hotplug inadvertenly restarting driver during
|
|
# forced shutdown
|
|
[ -f /var/run/killpower ] && return 0
|
|
|
|
config_load nut_server
|
|
build_config
|
|
|
|
case $@ in
|
|
"")
|
|
config_foreach start_driver_instance driver "$@"
|
|
start_server_instance upsd
|
|
;;
|
|
*upsd*)
|
|
start_server_instance upsd
|
|
;;
|
|
*)
|
|
config_foreach start_driver_instance driver "$@"
|
|
;;
|
|
esac
|
|
}
|
|
|
|
reload_service() {
|
|
stop_service "$@"
|
|
sleep 2
|
|
start_service "$@"
|
|
}
|
|
|
|
service_triggers() {
|
|
config_load nut_server
|
|
|
|
interface_triggers "add_trigger"
|
|
procd_add_reload_trigger "nut_server"
|
|
}
|