PHP的ftp_raw函数与MLSD和fsockopen:奇怪的阻塞行为和警告消息

时间:2015-07-20 19:46:45

标签: php sockets ftp

我目前在使用FTP ftp_raw功能时遇到问题。

我需要获得一个非常精确且详细的FTP目录文件和目录列表(ftp_rawlist对日期时间不够精确且未指定)。所以我想使用更规范化的命令:MLSD

我可以获取我想要的文件列表,但是从套接​​字获取数据后,再也无法使用PHP的ftp*函数了......

以下是我的代码摘录:

//Current folder = "/"
$directory = "/www/";

$currentFolder = ftp_pwd($ftpStream);
echo $currentFolder;
echo ' : ';
ftp_chdir($ftpStream, $directory);

$ret = ftp_raw($ftpStream, 'PASV');

if (preg_match('#^227.*\(([0-9]+,[0-9]+,[0-9]+,[0-9]+),([0-9]+),([0-9]+)\)$#', $ret[0], $matches)) {
    $controlIp = str_replace(',', '.', $matches[1]);
    $controlPort = intval($matches[2]) * 256 + intval($matches[3]);
    $socket = fsockopen($controlIp, $controlPort);
    ftp_raw($ftpStream, 'MLSD');
    $s = '';
    while (!feof($socket)) {
        $s .= fread($socket, 4096);
    }
    fclose($socket);
}

echo ftp_pwd($ftpStream); //line 256
echo ' --> ';
ftp_chdir($ftpStream, $currentFolder); //line 258

echo ftp_pwd($ftpStream); //line 260

显示:

/ :    <br />
<b>Warning</b>:  ftp_pwd(): Transfer complete. in <b>/media/sf_web/x/webService/models/Ftp.class.php</b> on line <b>256</b><br />
--><br />
<b>Warning</b>:  ftp_chdir(): &quot;/www&quot; is current directory. in <b>/media/sf_web/x/webService/models/Ftp.class.php</b> on line <b>258</b><br />
<br />
<b>Warning</b>:  ftp_pwd(): CWD command successful. in <b>/media/sf_web/x/webService/models/Ftp.class.php</b> on line <b>260</b><br />

错误不一致。

有关信息,如果我评论整个&#34;如果&#34;条件块,我得到一个正常的行为,没有警告:

/ : /www --> /
*/

哪里可能是我的错?如何解释不连贯的警告信息以及之后无法正确使用$ftpStream

我一直在寻找解决方案几个小时,所以非常感谢你的帮助... :)

提前谢谢。

1 个答案:

答案 0 :(得分:1)

PHP现在支持MLSD命令nativelly,因为7.2及其ftp_mlsd function

原始答案:

MLSD作为所有FTP数据传输命令,会发出多个响应。

通常会有一个或多个1xx初步回复,例如

150 Opening data channel for directory listing of "/path"

最后一个回复,在你的情况下:

226 Transfer complete.

ftp_raw函数只读取一个响应,即前一个150响应。将最终226响应保留在队列中。

一旦发出新的FTP命令,即PWD,它就会找到待处理的226响应,并在此处停止。

实际的PWD响应再次“留在队列中”:

257 /www is current directory.

等等。

您收到警告,因为状态代码与命令不匹配。对PWD的肯定回复是257,而不是226。对CWD的肯定回复是250,而不是257等。

我担心没有办法同步,因为没有PHP函数可以跳过/读取FTP回复。

因此,您唯一的选择可能是断开连接并重新连接。