从初始化脚本中以守护进程形式运行任意程序

从初始化脚本中以守护进程形式运行任意程序

我需要在 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],则应使用中的successfailure函数/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,如果这会影响您的决定(无论哪个方向)。

相关内容