Правильное выполнение оболочки в PHP

Проблема

Я использовал функцию, которая использовала proc_open () для вызова команд оболочки. Кажется, что то, как я делал STDIO, было неправильным и иногда приводило к зависанию PHP или целевой команды. Это исходный код:

function execute($cmd, $stdin=null){
    $proc=proc_open($cmd,array(0=>array('pipe','r'),1=>array('pipe','w'),2=>array('pipe','w')),$pipes);
    fwrite($pipes[0],$stdin);                fclose($pipes[0]);
    $stdout=stream_get_contents($pipes[1]);  fclose($pipes[1]);
    $stderr=stream_get_contents($pipes[2]);  fclose($pipes[2]);
    return array( 'stdout'=>$stdout, 'stderr'=>$stderr, 'return'=>proc_close($proc) );
}

Он работает большую часть времени , но этого недостаточно, я хочу, чтобы он работал всегда.

Проблема заключается в stream_get_contents () блокируется, если буферы STDIO превышают 4 КБ данных.

Контрольный пример

function out($data){
    file_put_contents('php://stdout',$data);
}
function err($data){
    file_put_contents('php://stderr',$data);
}
if(isset($argc)){
    // RUN CLI TESTCASE
    out(str_repeat('o',1030);
    err(str_repeat('e',1030);
    out(str_repeat('O',1030);
    err(str_repeat('E',1030);
    die(128); // to test return error code
}else{
    // RUN EXECUTION TEST CASE
    $res=execute('php -f '.escapeshellarg(__FILE__));
}

Мы дважды выводим строку в STDERR и STDOUT с общей длиной 4120 байт (превышающей 4 КБ). Это приводит к блокировке PHP с обеих сторон.

Решение

По-видимому, stream_select () - это правильный путь. У меня есть следующий код:

function execute($cmd,$stdin=null,$timeout=20000){
    $proc=proc_open($cmd,array(0=>array('pipe','r'),1=>array('pipe','w'),2=>array('pipe','w')),$pipes);
    $write  = array($pipes[0]);
    $read   = array($pipes[1], $pipes[2]);
    $except = null;
    $stdout = '';
    $stderr = '';
    while($r = stream_select($read, $write, $except, null, $timeout)){
        foreach($read as $stream){

            // handle STDOUT
            if($stream===$pipes[1])
/*...*/         $stdout.=stream_get_contents($stream);

            // handle STDERR
            if($stream===$pipes[2])
/*...*/         $stderr.=stream_get_contents($stream);
        }

        // Handle STDIN (???)
        if(isset($write[0])) ;

// the following code is temporary
$n=isset($n) ? $n+1 : 0; if($n>10)break; // break while loop after 10 iterations

    }
}

Единственная оставшаяся часть головоломки - это обработка STDIN (см. Строку, отмеченную (???) ). Я понял, что STDIN должен предоставляться тем, что вызывает мою функцию, execute () . Но что, если я вообще не хочу использовать STDIN? В моем тестовом примере выше я не запрашивал ввод, но я должен был что-то сделать с STDIN.

Тем не менее, вышеупомянутый подход все еще зависает в stream_get_contents () . Я совершенно не уверен, что делать / попробовать дальше.

Источники

Решение было предложено Якобом Труэльсеном, а также обнаружена исходная проблема. Наконечник 4k также был его идеей. До этого я был озадачен, почему функция работает нормально (не знал, что все зависит от размера буфера).

9
задан Christian 16 May 2011 в 21:50
поделиться