5

所以我做了一件简单的事情,首先我通过 ssh2_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:[OK] 启动 httpd:[OK]

但是当我尝试例如在没有 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。

4

2 回答 2

0

我决定相信我的服务器大约需要 2 秒来检查是否存在错误。并以防万一为第二个循环设置最大时间。所以我的代码如下。它执行的比我想要的要多...

    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;
于 2015-04-24T12:09:47.893 回答
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);
}

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

于 2020-02-03T02:10:18.577 回答