Идентификатор процесса проверки PHP

Это то, что я задавался вопросом на некоторое время и решил спросить об этом.

У нас есть функция getmypid(), которая возвращает текущий идентификатор процесса скрипта. Есть ли какая-то функция, например

checkifpidexists () в php? Я имею в виду встроенный, а не какое-то решение для пакетного сценария.

И есть ли способ изменить pid скриптов?

Некоторое уточнение:

Я хочу проверить, существует ли pid, чтобы увидеть, запущен ли уже скрипт, чтобы он больше не запускался, если хотите, поддельное задание cron.

Причина, по которой я хотел изменить pid, заключается в том, что я могу установить pid скрипта на что-то действительно высокое, например 60000, и жестко закодировать это значение, чтобы этот скрипт мог работать только с этим pid, чтобы работал только 1 экземпляр.

РЕДАКТИРОВАТЬ----

Чтобы помочь кому-либо еще с этой проблемой, я создал этот класс:

class instance {

    private $lock_file = '';
    private $is_running = false;

    public function __construct($id = __FILE__) {

        $id = md5($id);

        $this->lock_file = sys_get_temp_dir() . $id;

        if (file_exists($this->lock_file)) {
            $this->is_running = true;
        } else {
            $file = fopen($this->lock_file, 'w');
            fclose($file);
        }
    }

    public function __destruct() {
        if (file_exists($this->lock_file) && !$this->is_running) {
            unlink($this->lock_file);
        }
    }

    public function is_running() {
        return $this->is_running;
    }

}

и вы используете его так:

$instance = new instance('abcd'); // the argument is optional as it defaults to __FILE__

if ($instance->is_running()) {
    echo 'file already running';    
} else {
    echo 'file not running';
}

person Ozzy    schedule 01.11.2009    source источник
comment
поэтому я могу установить pid сценария на что-то действительно высокое, например 60000, и жестко закодировать это значение - предположение, что вы можете установить PID, ошибочно. Такой подход не может работать.   -  person Ulrich Eckhardt    schedule 10.07.2020


Ответы (6)


В Linux вы должны посмотреть /proc.

return file_exists( "/proc/$pid" );

В Windows вы можете выполнить shell_exec() tasklist.exe, и это найдет соответствующий идентификатор процесса:

$processes = explode( "\n", shell_exec( "tasklist.exe" ));
foreach( $processes as $process )
{
     if( strpos( "Image Name", $process ) === 0
       || strpos( "===", $process ) === 0 )
          continue;
     $matches = false;
     preg_match( "/(.*?)\s+(\d+).*$/", $process, $matches );
     $pid = $matches[ 2 ];
}

Я считаю, что вы хотите сделать, это сохранить файл PID. В демонах, которые я написал, они проверяют файл конфигурации, ищут экземпляр файла pid, получают pid из файла pid, проверяют, существует ли /proc/$pid, и если нет, удаляют pid. файл.

   if( file_exists("/tmp/daemon.pid"))
   {
       $pid = file_get_contents( "/tmp/daemon.pid" );
       if( file_exists( "/proc/$pid" ))
       {
           error_log( "found a running instance, exiting.");
           exit(1);
       }
       else
       {
           error_log( "previous process exited without cleaning pidfile, removing" );
           unlink( "/tmp/daemon.pid" );
       }
   }
   $h = fopen("/tmp/daemon.pid", 'w');
   if( $h ) fwrite( $h, getmypid() );
   fclose( $h );

Идентификаторы процессов предоставляются ОС, и нельзя зарезервировать идентификатор процесса. Вы должны написать свой демон, чтобы он уважал файл pid.

person memnoch_proxy    schedule 01.11.2009
comment
pcntl_fork() не изменит pid текущего процесса! Функция pcntl_fork() создает дочерний процесс, который отличается от родительского процесса только своими PID и PPID. Пожалуйста, обратитесь к справочной странице fork(2) вашей системы для получения подробной информации о том, как fork работает в вашей системе. - person ennuikiller; 01.11.2009
comment
хм, я надеялся на более кросс-платформенный способ проверки существования pid :( - person Ozzy; 01.11.2009
comment
В моем редактировании я упомянул, что не думаю, что он хочет pcntl_fork(), и при этом он не получит выбор pid. - person memnoch_proxy; 01.11.2009
comment
Цикл для Windows должен быть примерно таким: <? foreach ($processes as $process) { if (preg_match('/^(.*)\s+'.$pid.'/', $process)) { return true; } } ?> - person Kouber Saparev; 17.06.2011

Лучшим способом добиться этого было бы использование pid или файла блокировки. Просто проверьте наличие файла pid, создайте его по мере необходимости и заполните его текущим pid.

<?
  class pidfile {
    private $_file;
    private $_running;

    public function __construct($dir, $name) {
      $this->_file = "$dir/$name.pid";

      if (file_exists($this->_file)) {
        $pid = trim(file_get_contents($this->_file));
        if (posix_kill($pid, 0)) {
          $this->_running = true;
        }
      }

      if (! $this->_running) {
        $pid = getmypid();
        file_put_contents($this->_file, $pid);
      }
    }

    public function __destruct() {
      if ((! $this->_running) && file_exists($this->_file)) {
        unlink($this->_file);
      }
    }

    public function is_already_running() {
      return $this->_running;
    }
  }
?>

И используйте его следующим образом:

<?
  $pidfile = new pidfile('/tmp', 'myscript');
  if($pidfile->is_already_running()) {
    echo "Already running.\n";
    exit;
  } else {
    echo "Started...\n";
  }
?>

Здесь не так много проверок ошибок, но быстрый запуск показывает, что это работает в моей системе.

person jheddings    schedule 01.11.2009
comment
Проблема в том, что __destruct не запускается при сбросе Apache. - person Elia Weiss; 19.10.2019

Для проверки наличия PID на компьютере с Windows я использую:

function pidExists($pid)
{
    exec('TASKLIST /NH /FO "CSV" /FI "PID eq '.$pid.'"', $outputA );
    $outputB = explode( '","', $outputA[0] );
    return isset($outputB[1])?true:false;
}

Обратите внимание, что $outputB[0] содержит сообщения о том, что pid не может быть найден, если pid действительно не существует! Поэтому для проверки я использую второй аргумент.

ИЗМЕНИТЬ:

Чтобы расширить это, также можно динамически создавать сценарии в окнах в фоновом режиме с помощью powershell, например:

    // this function builds an argument list to parse into the newly spawned script. 
    // which can be accessed through the superglobal  global $argv;
    function buildArgList( array $arguments){
     return ' '. implode(' ', $arguments) .' ';
    }

    $arguments = buildArgList(['argument1','argument2','argument3']);
    $windowstyle = 'normal'; // or you can use hidden to hide the CLI

    pclose(popen("powershell start-process -FilePath '" . PHP_BINARY . "' -ArgumentList '-f\"" . $file . " " . $arguments . "\"' -WindowStyle " . $windowstyle,"r"));   

Скрипт, который вы создаете, может затем использовать: cli_set_process_title установить заголовок этого процесса на какой-то уникальный хэш.

внутри родительского процесса, породившего дочерний процесс, вы можете использовать следующий код, чтобы найти этот процесс в списке задач, используя его заголовок окна для поиска уникального хэша.

exec('TASKLIST /NH /FO "CSV" /FI "windowtitle eq ' . escapeshellarg($uniquehash) . '"', $output );

В сочетании с базой данных вы можете создать workermanager, взаимодействующий между различными php-скриптами.

person spoofie    schedule 15.08.2017
comment
Также следует учитывать, что powershell не очень хорошо обрабатывает пути с пробелами, поэтому это может помешать его работе. - person spoofie; 06.06.2019

Нет, вы не можете изменить идентификатор процесса. Он назначается ядром и является частью структур данных ядра.

person ennuikiller    schedule 01.11.2009

Как уже говорили другие, вы не можете изменить идентификатор процесса - он назначается и полностью управляется ядром ОС. Кроме того, вы не сказали, основано ли это на командной строке или на веб-сервере: если это последнее, вы, возможно, даже не получите pid вашего скрипта.

страница руководства для getmypid() содержит несколько примеров "оптимистичной" блокировки. Я использую слово optimisitc, поскольку PHP никогда не будет приближаться к подобным веб-приложениям asp.net, где у вас есть настоящая многопоточная среда с общими/статическими классами и, следовательно, Singleton для использования/злоупотребления. В основном у вас есть возможность:

  • Прикосновение к «файлу блокировки» где-то в файловой системе. Затем ваш скрипт проверяет, существует ли этот файл: если он существует, завершите работу, в противном случае коснитесь этого файла и продолжите обработку.
  • Установка флага на основе базы данных, чтобы сказать, что скрипт запущен. Как и выше, но используйте таблицу/поле БД, чтобы пометить скрипт как работающий.

Оба они полагаются на правильное завершение сценария (поскольку последним шагом будет удаление флага блокировки файла/БД). Если по какой-либо причине произойдет сбой скрипта (или самого компьютера), вы можете оставить процесс ручной очистки, чтобы удалить флаг. Для этого нет простого решения, но один из способов, который можно изучить, — это посмотреть на отметку даты блокировки с помощью произвольного подхода «если старше, чем X, последний запуск должен был дать сбой».

person Ian    schedule 01.11.2009

Не забывайте также, что вы можете получить доступ к командам оболочки с помощью обратных кавычек (`), что даст вам доступ к стандартным инструментам * nix для работы с pids.

источник: http://www.php.net/manual/en/language.operators.execution.php

person Chris    schedule 01.11.2009