在Centos上守护PHP脚本

时间:2015-03-19 18:38:09

标签: php linux daemon

尝试在Centos Linux环境中守护php脚本时遇到一些问题。不幸的是,Centos只有有限的守护进程功能,而不是start-stop-daemon ...

我试图创建一个连续运行php脚本的服务,直到捕获到信号。我也想要一个PHP脚本来启动服务。

  • foolauncher.php通过系统启动服务(' service foo start')
  • /etc/init.d/foo启动foo.php
  • foo.php生成子进程以进行守护进程并使父进程退出

然而,当我启动foolauncher.php时,它会显示启动我的foo.php服务的init.d脚本,不会释放终端,而是输出我的系统调用。当我发送SIGINT退出父节点时,foo服务也会停止运行。

我知道问题的位置在foo.php:

$output = system('service foo status');

如果省略系统调用的那一行,发送SIGINT退出父级将不会影响foo服务。

我知道fclose(STDERR),fclose(STDOUT)和fclose(STDIN),但我不知道我实际关闭这些输出的位置,因为它似乎会导致类似的问题向父母发送SIGINT; foo.php仍然在系统调用中死亡。

似乎因为我正在向STDOUT输出内容,父母无法完全释放孩子...我不打算使用其他的deamonizing方法(即系统守护程序,nohup) ,或者使用cron作业来执行(尽管我听说过daemonizing php不是一个好主意),但解决了如何解决这个问题。

任何建议,建议或解决方案都会非常受欢迎!谢谢!

以下是文件:

foolauncher.php:

<?php
system('service foo start');

foo.php:

<?php

declare(ticks = 1);
$pid = pcntl_fork();

if ($pid < 0)
{
    system("echo 'failed to fork' >> /foo/foo.log");
    exit;
}
else if ($pid)
{
    system("echo 'parent exiting' >> /foo/foo.log");
    exit;
}
$sid = posix_setsid();
if ($sid < 0)
{
    exit;
}

pcntl_signal(SIGTERM, "signalHandler");
pcntl_signal(SIGINT, "signalHandler");

$daemonPid = posix_getpid();
system("echo 'Process started on $daemonPid!\n' >> /foo/foo.log");
system("echo $daemonPid > /var/run/foo.pid");

while(true)
{
    // If I dont have this line, everything daemonizes nicely
    $output = system("service foo status");
    system("echo 'In infinite loop still! $output\n' >> /foo/foo.log");
    sleep(3);
}

function signalHandler($signo)
{

    switch($signo)
    {
        case SIGTERM:
        {
            system("echo 'SIGTERM caught!\n' >> /foo/foo.log");
            exit();
        }
        case SIGINT:
        {
            system("echo 'SIGINT caught!\n' >> /foo/foo.log");
            exit();
        }
    }
}

/etc/init.d/foo

# Source function library.
. /etc/rc.d/init.d/functions

php=${PHP-/usr/bin/php}
foo=${FOO-/foo/foo.php}
prog=`/bin/basename $foo`
lockfile=${LOCKFILE-/var/lock/subsys/foo}
pidfile=${PIDFILE-/var/run/foo.pid}
RETVAL=0

start() {
    echo -n $"Starting $prog: "

    daemon --pidfile=${pidfile} ${php} ${foo}
    RETVAL=$?
    echo
    [ $RETVAL = 0 ] && touch ${lockfile}
    return $RETVAL
}

stop() {
    echo -n $"Stopping $prog: "
    killproc -p ${pidfile} ${prog}
    RETVAL=$?
    echo
    [ $RETVAL = 0 ] && rm -f ${lockfile} ${pidfile}
}


rh_status() {
    status -p ${pidfile} ${foo}
}

# See how we were called.
case "$1" in
    start)
        rh_status >/dev/null 2>&1 && exit 0
        start
        ;;
    stop)
        stop
        ;;
    status)
        rh_status
        RETVAL=$?
        ;;
    restart)
        configtest -q || exit $RETVAL
        stop
        start
        ;;
    *)
        echo $"Usage: $prog {start|stop|restart|status}"
        RETVAL=2
esac
exit $RETVAL

1 个答案:

答案 0 :(得分:0)

找到解决方案!

通过将stdout和stderr重定向到/ dev / null

,我能够很好地获得daemonize服务

来自foolauncher.php

<?php
system('service foo start &> /dev/null');

似乎做我想要的!