feof产生无限循环

时间:2015-04-23 13:01:12

标签: php feof ssh2-exec

所以我做了一件简单的事情,首先我通过ssh2_exec执行exec命令(在成功验证后),然后在变量中读取答案。我的代码如下(未经过身份验证)

try {
        $stdout_stream = ssh2_exec($this->connection, $_cmd);
        $stderr_stream = ssh2_fetch_stream($stdout_stream, \SSH2_STREAM_STDERR);
    } catch (Exception $e) {
        $std_output = $e->getMessage();
        return false;
    }

    $output = "";

    while (!feof($stdout_stream)) {
        $output .= fgets($stdout_stream, 4096);
    }

    while (!feof($stderr_stream)) {
        $output .= fgets($stderr_stream, 4096);
    }

    fclose($stdout_stream);
    fclose($stderr_stream);     

    return $output;

例如,我尝试执行这样的cmd:

sudo service httpd stop && sudo service httpd start

因此,当命令执行良好时,一切都很好,响应是

  

关闭httpd:[确定]启动httpd:[确定]

但是当我尝试在没有sudo的情况下执行这样的命令时

service httpd stop && service httpd start

我知道服务器说像"命令未找到"或者像这样,但是我无法得到这个错误,这个脚本无限地执行。

我尝试以这种方式重写我的代码(或类似的其他代码)

    $dataString = fgets($stdout_stream);
        if($dataString == "\n" || $dataString == "\r\n" || $dataString == "") {
            //var_dump("Empty line found.");
        }

        if($dataString === false && !feof($stdout_stream)) {
            //var_dump("not string");
        } elseif($dataString === false && feof($stdout_stream)) {
            //var_dump("We are at the end of the file.\n");
            break;
        } else {
            //else all is good, process line read in
            $output .= $dataString;
        }
    }

但结果是一样的。

所以问题是我们不能事先说出导致无限循环$stdout_stream$stderr_stream的原因。

我使用的是PHP 5.3。

2 个答案:

答案 0 :(得分:0)

我决定相信我的服务器 ~2秒就足以检查是否存在错误。以防万一为secod循环设置最大时间。所以我的代码如下。它执行的次数超过我想要的......

    try {
        $stdout_stream = ssh2_exec($this->connection, $_cmd);
        $stderr_stream = ssh2_fetch_stream($stdout_stream, \SSH2_STREAM_STDERR);
    } catch (Exception $e) {
        $std_output = $e->getMessage();
        return false;
    }

    $output = "";

    $start_time = time();
    $max_time = 2; //time in seconds

    while(((time()-$start_time) < $max_time)) {
        $output .= fgets($stderr_stream, 4096);
    }

    if(empty($output)) {
        $start_time = time();
        $max_time = 10; //time in seconds

        while (!feof($stdout_stream)) {
            $output .= fgets($stdout_stream, 4096);
            if((time()-$start_time) > $max_time) break;
        }
    }

    fclose($stdout_stream);
    fclose($stderr_stream);     

    return $output;

答案 1 :(得分:0)

男孩,我很高兴我发现了这个!我遇到了同样的问题。我最终以这种方式解决了。

代替:

while (!feof($stdout_stream)) {
    $output .= fgets($stdout_stream, 4096);
}

while (!feof($stderr_stream)) {
    $output .= fgets($stderr_stream, 4096);
}

执行此操作:

while (!feof($stdout_stream) || !feof($stderr_stream)) {
    $output_stdout .= fgets($stdout_stream, 4096);
    $output_stderr .= fgets($stderr_stream, 4096);
}

技术细节超出了我的范围,但似乎两个描述符同时关闭,而不是一次关闭。