PHP:在web浏览器中输出system/Shell_exec命令输出


PHP: Outputting the system / Shell_exec command output in web browser


我尝试在网页上输出一个简单的ping命令,其方式与在终端中显示的方式类似(同时),使用shell_exec;但它只在完全执行后显示,而我需要它在终端上显示时显示,

我的代码是

<?php
$i= shell_exec("ping -c 4 google.com");
echo "<pre> $i <pre>";
?>

它要等一段时间,然后把整个事情都放在一个镜头上。。PHP能识别每一行的输出并将其显示在网页上吗

编辑
我也试过这个

    <?php
    $proc = popen("ping -c 4 google.com", 'r');
    echo '<pre>';
    while (!feof($proc)) {
        echo fread($proc, 4096);
    }
    echo '</pre>';
    ?>

但我还是得到了同样的结果。。

EDIT
当我尝试在终端中执行此PHP代码时,(phptest.PHP)的工作方式与我们直接在服务器上执行ping时的工作方式相同。但在网页上它仍然是一样的。

嗯,来自web浏览器的奇怪行为。我使用的是这个代码:

<?php
ob_end_flush();
ini_set("output_buffering", "0");
ob_implicit_flush(true);
function pingtest()
{
    $proc = popen("ping -c 5 google.com", 'r');
    while (!feof($proc))
    {
        echo "[".date("i:s")."] ".fread($proc, 4096);
    }
}
?>
<!DOCTYPE html>
<html>
<body>
  <pre>
Immediate output: 
<?php
pingtest();
?>
  </pre>
</body>
</html>

在浏览器中,内容将在收到所有字节后显示。但是,内容实际上是按时交付的,做这个测试:

wget -O - -q "http://localhost/ping.php"

您将看到响应是由php&apache2准时。

我在长任务上使用这种执行已经有一段时间了,但使用了一个更复杂的解决方案:

  • 接口的html文件
  • 运行长任务的php文件
  • 使用EventSource对象连接html接口和php长执行(可在html5上获得)

界面(test.html)

<!DOCTYPE html>
<html>
<head>
    <title>Simple EventSource example</title>
</head>
<body>
    <script type="text/javascript">
    function eventsourcetest() {
        var ta = document.getElementById('output');
        var source = new EventSource('test.php');
        source.addEventListener('message', function(e) {
            if (e.data !== '') {
               ta.value += e.data + ''n';
            }
        }, false);
        source.addEventListener('error', function(e) {
            source.close();
        }, false);
    }
    </script>
    <p>Output:<br/><textarea id="output" style="width: 80%; height: 25em;"></textarea></p>
    <p><button type="button" onclick="eventsourcetest();">ping google.com</button>
</html>

服务器端组件(test.php)

<?php
ob_end_flush();
ini_set("output_buffering", "0");
ob_implicit_flush(true);
header('Content-Type: text/event-stream');
header('Cache-Control: no-cache');
function echoEvent($datatext) {
    echo "data: ".implode("'ndata: ", explode("'n", $datatext))."'n'n";
}
echoEvent("Start!");
$proc = popen("ping -c 5 google.com", 'r');
while (!feof($proc)) {
    echoEvent(fread($proc, 4096));
}
echoEvent("Finish!");

将两个文件放在Web服务器上的一个位置,然后输入test.html,我认为这就是您从一开始就要查找的内容。

使用输出缓冲和flush。您可能还想了解Symfony 2流程组件。

这不是PHP的问题,或者更确切地说,这是PHP和浏览器之间的共享问题。

在PHP中:确保输出缓冲关闭,可以在输出任何内容之前运行ob_end_clean()来完成此操作。

正如这篇SO文章所建议的那样,您必须将输出的第一个字符串填充为512字节,或者通过http头指定一个字符集编码。填充解决方案可能是解决这一问题的最简单方法,基本上是这样的:echo(str_pad("Live Ping Test!",512));,然后开始回显fread的结果。

您可能需要尝试使用flush()在输出就绪时刷新输出,并使用passthru()执行命令。

Carlos C Soto是对的,您必须使用javascript。EventSource就是一条路。基本上,是javascript代码会不断地调用url

您可以在一个文件中编写ping的输出,并编写一个php脚本来读取最后一行,然后使用eventsource调用该脚本。

在web上搜索"服务器发送的事件"以查找更多示例

if可以使用apache执行用户进行解析。如果根用户不同,服务器用户不同,则不允许执行命令行命令。

我测试了卡洛斯的答案。。。

我不得不添加flush();ob_flush();使其正常工作(需要刷新和ob_flush)

像这个

<?php
$proc = popen("ping -c 5 google.com", 'r');
while (!feof($proc))
{
    echo "[".date("i:s")."] ".fread($proc, 4096).'<br>';flush();ob_flush();
}
?>