Мне нужно установить программу как службу в Red Hat. Он не работает в фоновом режиме, не управляет своим файлом PID и не управляет собственными журналами. Он просто запускается и печатает в STDOUT и STDERR.
Используя стандартные сценарии инициализации в качестве руководства, я разработал следующее:
#!/bin/bash
#
# /etc/rc.d/init.d/someprog
#
# Starts the someprog daemon
#
# chkconfig: 345 80 20
# description: the someprog daemon
# processname: someprog
# config: /etc/someprog.conf
# Source function library.
. /etc/rc.d/init.d/functions
prog="someprog"
exec="/usr/local/bin/$prog"
[ -e "/etc/sysconfig/$prog" ] && . "/etc/sysconfig/$prog"
lockfile="/var/lock/subsys/$prog"
RETVAL=0
check() {
[ `id -u` = 0 ] || exit 4
test -x "$exec" || exit 5
}
start() {
check
if [ ! -f "$lockfile" ]; then
echo -n $"Starting $prog: "
daemon --user someproguser "$exec"
RETVAL=$?
[ $RETVAL -eq 0 ] && touch "$lockfile"
echo
fi
return $RETVAL
}
stop() {
check
echo -n $"Stopping $prog: "
killproc "exec"
RETVAL=$?
[ $RETVAL -eq 0 ] && rm -f "$lockfile"
echo
return $RETVAL
}
restart() {
stop
start
}
case "$1" in
start)
start
;;
stop)
stop
;;
restart)
restart
;;
status)
status "$prog"
RETVAL=$?
;;
*)
echo $"Usage: $0 {start|stop|restart|status}"
RETVAL=2
esac
exit $RETVAL
Возможно, моей ошибкой было копирование-вставка и изменение некоторых существующих скриптов в /etc/init.d. В любом случае результирующий сервис ведет себя странно:
- когда я запускаю его,
service someprog start
программа печатает на терминал, и команда не завершается. - если я нажимаю CTRL-C, выводится сообщение «Сеанс завершен, оболочка уничтожена ... ... убита. Сбой». Я должен сделать это, чтобы снова получить приглашение оболочки.
- теперь, когда я запускаю,
service someprog status
он говорит, что он работает и перечисляет его PID. Я вижу этоps
так, что он работает. - теперь, когда я бегу,
service someprog stop
он не может остановиться. Я могу убедиться, что он все еще работает сps
.
Что мне нужно изменить, чтобы someprog
он отправлялся в фоновый режим и управлялся как сервис?
Изменить: Теперь я нашел пару связанных вопросов, ни один из них с фактическим ответом, кроме "сделать что-то еще":
- Вызов демону в скрипте /etc/init.d блокируется, не работает в фоновом режиме
- Заставить скрипт оболочки работать в CentOS как демон?
Изменить: этот ответ о двойном разветвлении мог бы решить мою проблему, но теперь моя программа сама раздваивается, и это работает: https://stackoverflow.com/a/9646251/898699
Ответы:
Команда «не завершена», потому что
daemon
функция не запускает ваше приложение в фоновом режиме. Вам нужно будет добавить&
в конец вашейdaemon
команды, например, так:daemon --user someproguser $exec &
Если
someprog
не обрабатываетSIGHUP
, вы должны запустить команду с,nohup
чтобы убедиться, что ваш процесс не будет получать,SIGHUP
который говорит ваш процесс, чтобы выйти, когда родительская оболочка выходит. Это будет выглядеть так:daemon --user someproguser "nohup $exec" &
В вашей
stop
функцииkillproc "exec"
ничего не делает для остановки вашей программы. Это должно читаться так:killproc $exec
killproc
требуется полный путь к вашему приложению, чтобы правильно его остановить. Вkillproc
прошлом у меня были некоторые проблемы , поэтому вы также можете просто убить PID в PIDFILE, для которого вы должны писатьsomeprog
PID, примерно так:cat $pidfile | xargs kill
Вы можете написать PIDFILE так:
ps aux | grep $exec | grep -v grep | tr -s " " | cut -d " " -f2 > $pidfile
где
$pidfile
указывает на/var/run/someprog.pid
.Если вы хотите [OK] или [FAILED] на
stop
функции, вы должны использоватьsuccess
иfailure
функцию из/etc/rc.d/init.d/functions
. Вам не нужно это вstart
функции, потому чтоdaemon
вызывает подходящий для вас.Вам также нужны только кавычки вокруг строк с пробелами. Однако это выбор стиля, так что решать вам.
Все эти изменения выглядят так:
источник
Если это ваша программа, пожалуйста, напишите ее как правильный демон. Особенно если его для перераспределения. :)
Вы можете попробовать Монит . Или, может быть, что-то вроде runit или daemontools. У тех могущественных нет легкодоступных пакетов. Daemontools от DJB, если это влияет на ваше решение (в любом направлении.)
источник
Я провел еще несколько исследований, и похоже, что ответ «ты не можешь этого сделать». Программа, которая должна быть запущена, должна на самом деле должным образом демонизировать себя: разветвить и отсоединить стандартные дескрипторы файлов, отсоединиться от терминала и начать новый сеанс.
Редактировать: по-видимому, я не прав - двойное разветвление будет работать. https://stackoverflow.com/a/9646251/898699
источник