在linux中使用PHP的posix_kill()会导致奇怪的行为

时间:2012-01-17 00:07:04

标签: php c linux apache kill

我正在创建一个网页,用于监控和控制一些自定义C程序。我创建了一个页面,它将启动/停止一个C程序'启动器'(漂亮的通用名称),然后分叉并创建许多子进程。首发工作正常 - exec("cd launcher_dir; nohup ./launcher > outfile 2>&1 &");

停止是出现问题的地方。单击停止按钮后,随机发生两件事之一。有一个浏览器错误页面(101连接重置或324空响应)或页面加载两次,但您只能第二次看到它。我知道它加载两次的原因是因为代码中的调试消息。在这两种情况下,启动程序进程都被终止(发送SIGTERM)。但是如果页面加载两次,则第一次杀死启动器(此部分的页面上没有任何内容),第二次检查并发现没有启动器进程正在运行并显示一条消息:“启动器未运行”。

我正在将调试消息写入文件,发现重新加载发生在php代码中的某个可变行(有时会打印某个调试消息,有时则不会。)此外,还会设置php错误报告全部,没有错误。

Launcher捕获SIGTERM,依次将SIGTERM发送给它的子进程,然后调用exit(0)。

有趣的是,如果SIGKILL用于杀死启动器,那么php工作正常并且符合预期,但是这不允许启动器很好地关闭。这可能会发生什么?

以下是相关的php代码:

function stop_launcher(){
    $pid = get_launcher_pid(); // Definitely returns the correct pid
    debug_message("stop_launcher called, pid = ".$pid);
    if ($pid == "") {
        // If no 'connection reset' error occurs this is displayed
        // after first executing the else branch. Why is the php being run twice?
        print "Launcher doesn't seem to be running.. <br />";
        exit;
    } else {
        debug_message("killing");
        posix_kill(intval($pid), 15); //SIGTERM
        debug_message("kill finished"); // Sometimes this message is written, sometimes                      not
        if (ps_exists($pid)) { // Definitely works. This never gets displayed
            print "Shutdown failed. Try again</br>";
            exit;
        } 
    }
}

function debug_message($message){
    $fh = fopen(".debug", 'a') or die("can't open file");
    fwrite($fh, date("-r").":  ".$message."\n");
    fclose($fh);
}

任何建议都表示赞赏!

2 个答案:

答案 0 :(得分:2)

事实证明,这是让你觉得愚蠢的错误之一!发射器程序意外杀死了它的进程组......

答案 1 :(得分:0)

Launcher是shell。如果收到SIGTERM,有些shell会将SIGTERM发送给他们的孩子。将“nohup”改为“exec nohup”应该可以摆脱这个问题。