redhat – 从init脚本运行任意程序作为守护进程

我需要在Red Hat中安装一个程序作为服务.它不是后台自身,管理其PID文件或管理自己的日志.它只是运行并打印到STDOUT和STDERR.

使用标准的init脚本作为指南,我开发了以下内容:

#!/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启动它时,程序打印到终端并且命令没有完成.
>如果我按CTRL-C,它会打印“会话终止,杀死shell ……杀死.失败”.我必须这样做再次让我的shell提示回来.
>现在,当我运行服务someprog状态时,它表示它正在运行并列出其PID.我可以在ps中看到它,所以它正在运行.
>现在当我运行服务someprog停止它无法停止.我可以验证它仍在使用ps运行.

我需要更改什么才能将someprog发送到后台并作为服务进行管理?

编辑:我现在发现了几个相关的问题,除了“做其他事情”之外,他们都没有真正的答案:

> Call to daemon in a /etc/init.d script is blocking,not running in background
> Getting shell script to run as a daemon on CentOS?

编辑:关于双重分叉的这个答案可能已经解决了我的问题,但现在我的程序本身是双叉并且有效:https://stackoverflow.com/a/9646251/898699

解决方法

命令“未完成”,因为守护程序功能不会在后台运行您的应用程序.您需要添加&到你的守护进程命令的末尾,如下所示:

守护进程–user someproguser $exec&

如果someprog不处理SIGHUP,你应该使用nohup运行命令,以确保你的进程不会收到SIGHUP,告诉你的进程在父shell退出时退出.这看起来像这样:

守护进程 – 用户someproguser“nohup $exec”&

在你的stop函数中,killproc“exec”没有做任何事情来阻止你的程序.它应该是这样读的:

killproc $exec

killproc需要应用程序的完整路径才能正确停止它.我以前遇到过killproc的麻烦,所以你也可以在PIDFILE中杀死PID,你应该用这样的东西写someprog的PID:

cat $pidfile | xargs杀了

您可以像这样编写PIDFILE:

ps aux | grep $exec | grep -v grep | tr -s“”| cut -d“” – f2> $pidfile进程文件

其中$pidfile指向/var/run/someprog.pid.

如果要在停止功能上使用[OK]或[FAILED],则应使用/etc/rc.d/init.d/functions中的成功和失败功能.您在启动函数中不需要这些,因为守护程序会为您调用适当的函数.

您还只需要带空格的字符串引号.不过,这是一种风格选择,所以这取决于你.

所有这些更改看起来像这样:

#!/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

相关文章

linux常用进程通信方式包括管道(pipe)、有名管道(FIFO)、...
Linux性能观测工具按类别可分为系统级别和进程级别,系统级别...
本文详细介绍了curl命令基础和高级用法,包括跳过https的证书...
本文包含作者工作中常用到的一些命令,用于诊断网络、磁盘占满...
linux的平均负载表示运行态和就绪态及不可中断状态(正在io)的...
CPU上下文频繁切换会导致系统性能下降,切换分为进程切换、线...