从初始化脚本中将任意程序作为守护程序运行


10

我需要在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


您是否正在使用libslack提供的“ daemon”实用程序启动程序。libslack.org/daemon/#documentation在这种情况下,可以将程序作为守护程序-n name --stop停止。另外,尝试将输出重定向(在启动程序时)到文件或/ dev / null并检查。
Ankit

2
根据您的redhat版本,您可以在upstart中为其创建一个简单的包装,然后直接在upstart中对其进行调用。然后,新贵将为您管理服务。不过这是EL6。
马修·伊夫

Answers:


1

命令“未完成”,因为该daemon功能不会在后台为您运行应用程序。您将需要&daemon命令末尾添加一个,如下所示:

daemon --user someproguser $exec &

如果someprog不处理SIGHUP,则应使用命令with nohup来确保您的进程不会收到SIGHUP,该命令告诉您的进程在父外壳退出时退出。看起来像这样:

daemon --user someproguser "nohup $exec" &

在您的stop函数中,killproc "exec"没有做任何事情来停止您的程序。它应如下所示:

killproc $exec

killproc需要应用程序的完整路径才能正确停止它。killproc过去我遇到了一些麻烦,因此您也可以使用以下命令杀死应该将someprogPID 写入PID的PIDFILE中的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功能上选择[确定]或[失败] ,则应使用中的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

抱歉,您花了将近4.5年的时间才能得到正确的答案。
Stuporman

-1

如果这是您的程序,请将其编写为适当的守护程序。特别是如果要重新分配。:)

您可以尝试监控。或者也许是runit或daemontools之类的东西。那些强大的软件包没有现成的。Daemontools来自DJB,如果这会影响您的决策(在任何一个方向上)。


-1

我进行了更多研究,似乎答案是“您不能这样做”。要运行的程序实际上必须正确地守护自己:分支并分离其标准文件句柄,从终端分离,并开始新的会话。

编辑:显然我错了-双叉将起作用。https://stackoverflow.com/a/9646251/898699


您可以使用评论中所述的新贵来做到这一点。
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.