Как сделать блокировку фреда и читать до конца? - PullRequest
2 голосов
/ 25 августа 2009

Я выполняю сценарий TCL из PHP, используя proc_open.

  1. Я сначала открываю оболочку TCL 2) Отправить команду, используя fwrite 3) Что мне нужно, так это хладнокровно ждать / блокировать до команда, отправленная fwrite, завершена и получить все содержимое. Команда может занять некоторое время для завершения. (Я могу прочитать только 2 строки, а затем он переходит к следующему циклу)

Может ли кто-нибудь направить меня.

Настоящий код

<?php

$app = 'tclsh84';
$descriptorspec = array(
0 => array("pipe","r"),
1 => array("pipe","w"),
2 => array("file","C:/wamp/www/tcl/bin/g.txt","w")
) ;
$process = proc_open($app, $descriptorspec, $pipes);
if (is_resource($process)) 
{


for($i=0;$i<4;$i++)
{

 fwrite($pipes[0], 'source c:/wamp/www/tcl/bin/test.tcl'."\n");
$content= fread($pipes[1],8192)
print "$content";

}    
   fclose($pipes[0]);    

   fclose($pipes[1]);


   proc_close($process);
}
?>

Ответы [ 2 ]

1 голос
/ 25 августа 2009

Я думаю о комбинации

  • stream_select и / или feof ()
  • fread () и объединение частичных результатов ($ result. = Fread ())
  • и, возможно, proc_get_status () , чтобы определить окончание процесса

Вы хотите подождать, пока приложение tcl не запишет что-либо в свой стандартный вывод в течение определенного периода времени (предполагая, что это означает конец последней команды), а затем отправить следующую команду / строку в свой стандартный ввод?

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

incr a 1
after 1000
puts [concat [clock seconds] $a]

и

<?php
$app = 'c:/programme/tcl/bin/tclsh85.exe';
$descriptorspec = array(
  0 => array("pipe","r"),
  1 => array("pipe","w"),
  2 => array("file","C:/god.txt","w")
) ;
$process = proc_open($app, $descriptorspec, $pipes);
if (is_resource($process)) {
  fwrite($pipes[0], "set a 1\n");
  for($i=0;$i<4;$i++) {
    fwrite($pipes[0], "source c:/helloworld.tcl\n");
  }
  // when all scripts are done the shell shall exit
  fwrite($pipes[0], "exit\n");
  fclose($pipes[0]);

  do {
    $read=array($pipes[1]); $write=array(); $except=array($pipes[1]);
    // wait up to 1 second for new output of the tcl process
    $ready = stream_select($read, $write, $except, 1, 0);
    if ( $ready && $read /* is not empty */) {
      // get the partial output
      $r = fread($pipes[1], 2048);
      echo $r;
    }
    // is the process still running?
    $status = proc_get_status($process);
  } while($status['running']);
  fclose($pipes[1]);
  proc_close($process);
}
?>

Возможно, вы захотите добавить еще обработку ошибок. Например. если stream_select () возвращает x раз с таймаутом, возможно, что-то пошло не так.

edit2:
Позвольте оболочке напечатать что-то, что вы можете сканировать после каждого сценария.

<?php
// something that's not in the "normal" output of the scripts
$id = 'done'. time();

$app = 'c:/programme/tcl/bin/tclsh85.exe';
$descriptorspec = array(
  0 => array("pipe","r"),
  1 => array("pipe","w"),
  2 => array("file","C:/god.txt","w")
) ;
$process = proc_open($app, $descriptorspec, $pipes);
if (is_resource($process)) {
  fwrite($pipes[0], "set a 1\n");
  for($i=0;$i<4;$i++) {
    $output = '';
    $continue = true;
    $cTimeout = 0;
    echo 'loop ', $i, "\n";
    fwrite($pipes[0], "source c:/helloworld.tcl\n");
    fwrite($pipes[0], "puts $id\n");
    echo "waiting for idle\n";
    do {
      $read=array($pipes[1]);
      $write=array();
      $except=array($pipes[1]);
      $ready = stream_select($read, $write, $except, 1, 0);
      if ( $ready && $read ) {
        $output .= fread($pipes[1], 2048);
        // if the delimiter id shows up in $output
        if ( false!==strpos($output, $id) ) {
            // the script is done
          $continue = false;
        }
      }
    } while($continue);
    echo 'loop ', $i, " finished\n";
  }
  proc_close($process);
}
?>
0 голосов
/ 25 августа 2009

Попробуйте:

$content = '';

while(!feof($pipes[1]))
{
    $content .= fread($pipes[1],8192);
}

Это ждет?

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...