Последующие сигналы pcntl_signal не запускают обработчик - PullRequest
8 голосов
/ 13 марта 2012

Позвольте мне начать с базового описания кода, который у меня есть. Я начинаю с основного родительского процесса (ПРИМЕЧАНИЕ. Я не показываю все функции для простоты. Дайте мне знать, если вам понадобится расширение в любой момент):

declare(ticks=1);
pcntl_signal(SIGHUP, array('forker', 'restartSignalHandler'));
if(forker_is_not_running()){
    new Forker();
}
class Forker {
    private $active_forks = array();
    private $parent_pid = null;

    public function __construct(){
        $this->parent_pid = getmypid();
        $this->create_fork();
        $this->wait_for_active();
    }

    public function wait_for_active(){
        while(!empty($this->active_forks)){
            foreach($this->active_forks as $k=>$fork){
                if($this->fork_no_longer_running($fork)){
                    unset($this->active_forks[$k]);
                }
            }
        }
    }

    // Pseudo code
    public function fork_no_longer_running($pid){
        // return true if 'ps -elf | grep $pid' doesn't returns only the grep command
        // else return false (aka the fork is still running)
    }

    public function create_fork(){
        $pid = pcntl_fork();
        if($pid == -1){
            posix_kill($this->parent_pid, SIGTERM);
        } else if($pid){
            // add the pid to the current fork
            $this->active_forks[] = $pid;
        } else {
            // Run our process
            pcntl_exec('/usr/bin/php', array('/domain/dev/www/index.php','holder','process'));
            exit(0);
        }
    }

    public function restartSignalHandler(){
        $forks = $this->active_forks;
        foreach($forks as $pid){
            $this->create_fork();
            posix_kill($pid, SIGINT);
        }
    }
}

class holder {
    public function process(){
        $x = new Processor();
    }
}

class Processor {
    public function __construct(){
        pcntl_signal(SIGINT, array($this, "shutdownSignalHandler"));
    }
    public function shutdownSignalHandler(){
        echo "Shutting down";
        exit;
    }
}

Вот что происходит:

  1. Я запускаю свой скрипт и правильно получаю процессы (например, Parentpid: 2, childpid: 3)
  2. Затем я посылаю родителю сигнал SIGHUP, и он правильно убивает и запускает новый дочерний процесс (например, Parentpid: 2, childpid: 4)
  3. Затем я посылаю родителю 2-й сигнал SIGHUP, и он правильно пытается добавить новый дочерний процесс, но отказывается убить 2-й дочерний процесс. (например, Parentpid: 2, undyingchildpid: 4, newchildpid: 5)

Дай мне знать, если это нужно больше деталей / не имеет смысла. Я не могу понять, почему в первый раз это правильно убило бы детей, но во второй раз - нет.

Чётная часть WEIRDER заключается в том, что когда я изменяю его так, что я изменяю свой обработчик перезапуска, чтобы он продолжал пытаться убить ребенка с помощью SIGINT, он каждый раз завершается неудачей, но когда я посылаю ему команду SIGKILL, это убивает ребенка процесс:

if($time_passed > 60){
    posix_kill($pid, SIGKILL);
}

Мне нужно, чтобы SIGINT убил ребенка, чтобы он справился с этим должным образом. Я не хочу просто SIGKILL это. Есть ли причина, по которой второй раз SIGINT не сработает, но SIGKILL сработает?

1 Ответ

1 голос
/ 02 апреля 2012

Прежде всего, вам не нужно разветвляться. Ваш код выполняет exec внутри дочернего элемента, вы можете просто запустить exec без разветвления, и ОС порождает вашу команду как дочерний элемент. Если вы хотите использовать fork, просто include файл в дочернем, а не execing.

public function create_fork(){
    //no need to actually fork!
    pcntl_exec('/usr/bin/php', array('/domain/dev/www/index.php','holder','process'));
}

//if you want to fork, better do it like this : 


public function create_fork(){
    $pid = pcntl_fork();
    if($pid == -1){
        posix_kill($this->parent_pid, SIGTERM);
    } else if($pid){
        // add the pid to the current fork
        $this->active_forks[] = $pid;
    } else {
       // Run our process
       include '/domain/dev/www/index.php';
       SomeClass::someMethod();
       exit(0);
    }
}

Также при использовании вилки нужно waitpid для детей. Итак, в ваш код нужно вставить что-то вроде:

//somewhere in a loop : 
$pidOfExittedChild = pcntl_waitpid (-1, $status, WNOHANG);
if ($pidOfExittedChild) {
    //a child has exitted, check its $status and do something
}

Проверьте больше на: http://php.net/manual/en/function.pcntl-waitpid.php

...