我需要在 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,它会打印“会话终止,正在终止 shell……终止。失败”。我必须这样做才能重新获得 shell 提示符。
- 现在当我运行时,
service someprog status
它显示它正在运行并列出其 PID。我可以看到它ps
正在运行。 - 现在我运行时
service someprog stop
它无法停止。我可以使用 来验证它是否仍在运行ps
。
我需要做哪些更改以便someprog
将其发送到后台并作为服务进行管理?
编辑:我现在发现了几个相关的问题,除了“做点别的事”之外,都没有实际的答案:
编辑:这个关于双重分叉的答案可能已经解决了我的问题,但是现在我的程序本身进行了双重分叉并且可以工作:https://stackoverflow.com/a/9646251/898699
答案1
该命令“未完成”,因为该daemon
函数不会在后台为您运行应用程序。您需要&
在命令末尾添加一个daemon
,如下所示:
daemon --user someproguser $exec &
如果someprog
没有处理SIGHUP
,则应使用 运行命令,以nohup
确保您的进程不会收到SIGHUP
在父 shell 退出时告诉您的进程退出的命令。该命令应如下所示:
daemon --user someproguser "nohup $exec" &
在您的stop
函数中,killproc "exec"
没有执行任何操作来停止您的程序。它应该像这样:
killproc $exec
killproc
需要应用程序的完整路径才能正确停止它。我killproc
过去遇到过一些问题,所以您也可以直接在 PIDFILE 中终止 PID,您应该使用someprog
类似下面的命令将 的 PID 写入该 PID:
cat $pidfile | xargs kill
您可以像这样编写 PIDFILE:
ps aux | grep $exec | grep -v grep | tr -s " " | cut -d " " -f2 > $pidfile
其中$pidfile
指向/var/run/someprog.pid
。
如果你希望stop
函数中有 [OK] 或 [FAILED],则应使用中的success
和failure
函数/etc/rc.d/init.d/functions
。函数中不需要这些,start
因为daemon
会为您调用合适的函数。
您也只需要在带空格的字符串周围加上引号。不过,这是一种样式选择,因此由您决定。
所有这些变化看起来如下:
#!/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
pidfile=/var/run/$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 "nohup $exec" &
RETVAL=$?
if [ $RETVAL -eq 0 ]; then
touch $lockfile
ps aux | grep $exec | grep -v grep | tr -s " " | cut -d " " -f2 > $pidfile
fi
echo
fi
return $RETVAL
}
stop() {
check
echo -n $"Stopping $prog: "
killproc $exec && cat $pidfile | kill
RETVAL=$?
if [ $RETVAL -eq 0 ]; then
rm -f $lockfile
rm -f $pidfile
success; echo
else
failure; echo
fi
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
答案2
如果这是你的程序,请将其编写为适当的守护进程。特别是如果它用于重新分发。:)
你可以尝试监控。或者可能是 runit 或 daemontools 之类的东西。这些可能没有现成的软件包。Daemontools 来自 DJB,如果这会影响您的决定(无论哪个方向)。