Preciso instalar um programa como serviço no Red Hat. Ele não se destaca, gerencia seu arquivo PID ou gerencia seus próprios logs. Apenas executa e imprime em STDOUT e STDERR.
Usando os scripts init padrão como guias, desenvolvi o seguinte:
#!/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
Pode ser que meu erro tenha sido copiar e colar e modificar alguns dos scripts existentes no /etc/init.d. De qualquer forma, o serviço resultante se comporta de maneira estranha:
- quando inicio,
service someprog start
o programa é impresso no terminal e o comando não é concluído. - se eu CTRL-C, ele imprime "Sessão terminada, matando shell ... ... morto. FAILED". Eu tenho que fazer isso para obter meu prompt de shell novamente.
- Agora, quando corro,
service someprog status
ele diz que está sendo executado e lista seu PID. Eu posso vê-lo,ps
então ele está sendo executado. - Agora, quando eu corro,
service someprog stop
ele não para. Posso verificar se ele ainda está sendo executadops
.
O que preciso alterar para que someprog
seja enviado para segundo plano e gerenciado como um serviço?
Edit: Agora encontrei algumas perguntas relacionadas, nenhuma delas com uma resposta real que não seja "faça outra coisa":
- A chamada para o daemon em um script /etc/init.d está bloqueando, não sendo executada em segundo plano
- Obtendo o script de shell para executar como um daemon no CentOS?
Edit: esta resposta sobre a bifurcação dupla pode ter resolvido o meu problema, mas agora o meu próprio programa bifurca-se e funciona: /programming//a/9646251/898699
Respostas:
O comando "não é concluído" porque a
daemon
função não executa seu aplicativo em segundo plano para você. Você precisará adicionar um&
no final do seudaemon
comando da seguinte forma:daemon --user someproguser $exec &
Se
someprog
não funcionarSIGHUP
, você deve executar o comandonohup
para garantir que seu processo não seja recebido, oSIGHUP
que instrui o processo a sair quando o shell pai sair. Isso seria assim:daemon --user someproguser "nohup $exec" &
Na sua
stop
função,killproc "exec"
não está fazendo nada para interromper o seu programa. Deve ler-se assim:killproc $exec
killproc
requer o caminho completo para o seu aplicativo para pará-lo corretamente. Eu tive alguns problemaskillproc
no passado, então você também pode matar o PID no PIDFILE no qual você deve escreversomeprog
o PID com algo assim:cat $pidfile | xargs kill
Você pode escrever o PIDFILE assim:
ps aux | grep $exec | grep -v grep | tr -s " " | cut -d " " -f2 > $pidfile
para onde
$pidfile
aponta/var/run/someprog.pid
.Se você deseja [OK] ou [FAILED] em sua
stop
função, use as funçõessuccess
efailure
de/etc/rc.d/init.d/functions
. Você não precisa disso nastart
função porquedaemon
chama o apropriado para você.Você também só precisa de aspas em torno de strings com espaços. É uma escolha de estilo, porém, é com você.
Todas essas mudanças são assim:
fonte
Se este é o seu programa, escreva-o como um daemon adequado. Especialmente se for para redistribuição. :)
Você pode tentar monit . Ou talvez algo como runit ou daemontools. Os poderosos não têm pacotes prontamente disponíveis. Daemontools é da DJB, se isso influencia sua decisão (em qualquer direção).
fonte
Eu fiz mais algumas pesquisas e parece que a resposta é "você não pode fazer isso". O programa a ser executado deve realmente se daemonizar adequadamente: bifurque e desanexe seus manipuladores de arquivos padrão, desconecte-se do terminal e inicie uma nova sessão.
Edit: aparentemente eu estou errado - bifurcação dupla iria funcionar. /programming//a/9646251/898699
fonte