2

我有一个输入字段,我在其中粘贴下载 url。之后,我使用 AJAX 请求来获取文件信息,例如 headerinfo、content-length、mime type &,以防我使用 curl 接受范围。

然后我开始一个连续的 xhr2 请求循环,其中包含我的 php 文件的范围。

http://www.example.com/chunks.php?url=http://url.com/someFile.ext&range=0-1024
http://www.example.com/chunks.php?url=http://url.com/someFile.ext&range=1024-2048
....

我也可以将其更改为

http://www.example.com/chunks.php?url=http://url.com/someFile.ext&range=0-1024
http://www.example.com/chunks.php?url=http://url.com/someFile.ext&range=1025-2049
....

取决于我的脚本开始读取文件的位置。

我的第一种方法是使用 cUrl 并设置范围

<?php
$ch=curl_init();
curl_setopt($ch,CURLOPT_URL,$_GET['url']);
curl_setopt($ch,CURLOPT_RANGE,$_GET['range']);
curl_setopt($ch,CURLOPT_BINARYTRANSFER,1);
curl_setopt($ch,CURLOPT_RETURNTRANSFER,1);
$result=curl_exec($ch);
curl_close($ch);
echo $result;
?>

效果很好,但如果范围块大于 1mb,则onprogress使用 ajax 的客户端事件没有动画。

我可以使用自定义CURLOPT_READFUNCTION...但我不知道它是如何工作的...所以我改变了方法并使用了简单的fopen

<?php
$r=explode('-',$_GET['range']);//get (from to) ranges
$cc=($r[1]-$r[0]); //Calculate Client Chunk length
$sc=128; //Set the Server chunk length

$b=""; //Buffer
$bytes=0; //bytes read

$h=fopen($_GET['url'],"rb"); // open the url
fseek($h,$r[0]); // jump to the from pointer retrieved from links

while($bytes<$cc){ //while bytes read is smaller than my client chunk
 $sc=(($bytes+$sc)>$cc?($cc-$bytes):$sc); //prolley an error here 
 //if the server chunk + bytes read is bigger than the client chunk
 //then the server chunk is clinet chunk - bytes read
 $b=fread($h,$sc); // read the buffer
 $bytes+=strlen($b); //add the buffer length to bytes read  
 echo $b;// echo the buffer
 ob_flush(); // flush 
 flush(); // flush
}
fclose($h); //close
?>

现在这个工作......我在客户端上得到了正确的动画,最终大小也是正确的,因为我使用 fseek && fread,指针应该没问题(0-1024,1024-2048)。

但文件已损坏。

现在经过一些测试......这很慢。

更好的方法是cUrl使用CURLOPT_READFUNCTIONfsoket打开...

所以我猜:

<?php
function $READ(){
 //here i need small chuncks of the response flushed. 
}
$ch=curl_init();
curl_setopt($ch,CURLOPT_URL,$_GET['url']);
curl_setopt($ch,CURLOPT_RANGE,$_GET['range']);
curl_setopt($ch,CURLOPT_BINARYTRANSFER,1);
curl_setopt($ch,CURLOPT_RETURNTRANSFER,1);
curl_setopt($ch,CURLOPT_READFUNCTION,$READ);
$result=curl_exec($ch);
curl_close($ch);
echo $result;
?>

如果您有更好的解决方案,我对使用 javascript 和 php 的所有内容持开放态度。

这样做的目的是创建一个带有简历的下载管理器,将文件存储到window.webkitRequestFileSystem不填满浏览器内存的情况下。

假设客户端有 8mb 的块,而服务器的块是 256kb ..

然后每 8mb 的块被附加到之前创建的文件中window.webkitRequestFileSystem

并且每 256kb 我都会更新平均下载速度,这样我就可以创建一个漂亮的动画。

服务器上的 php 仅使用 256kb ram,客户端浏览器可以每 8mb 清空垃圾收集(理论上)。

编辑2

对于这段代码,我找到了一个解决方案:

该代码允许您获取范围,例如:0-100 并获得这 100 字节的输出分块!

这允许您拥有一个具有连续完美 PROGRESSBAR 的 AJAX 脚本

<?php
function w($ch,$chunk){ 
 echo $chunk;
 ob_flush();
 flush();
 return strlen($chunk);
};
$ch=curl_init();
curl_setopt($ch,CURLOPT_URL,$_GET['url']);
curl_setopt($ch,CURLOPT_RANGE,$_GET['range']);
curl_setopt($ch,CURLOPT_BINARYTRANSFER,1);
curl_setopt($ch,CURLOPT_WRITEFUNCTION,w);
curl_exec($ch);
curl_close($ch);
?>

但我希望你们有更好的解决方案!谢谢

4

1 回答 1

1

我可以让它与 PHP curl 的CURLOPT_WRITEFUNCTION回调设置一起工作。以下curl_write_flush用于该 curl 选项的示例回调函数将接收到的每个块写入并将输出刷新到浏览器。

<?php

/**
 * CURLOPT_WRITEFUNCTION which flushes the output buffer and the SAPI buffer.
 *
 * @param resource $curl_handle
 * @param string   $chunk
 */
function curl_write_flush($curl_handle, $chunk)
{ 
    echo $chunk;

    ob_flush(); // flush output buffer (Output Control configuration specific)
    flush();    // flush output body (SAPI specific)

    return strlen($chunk); // tell Curl there was output (if any).
};

$curl_handle = curl_init($_GET['url']);
curl_setopt($curl_handle, CURLOPT_RANGE, $_GET['range']);
curl_setopt($curl_handle, CURLOPT_BINARYTRANSFER, 1);
curl_setopt($curl_handle, CURLOPT_WRITEFUNCTION, 'curl_write_flush');
curl_exec($curl_handle);
curl_close($curl_handle);

我尝试使用小文件和大文件,效果很好,但您无法设置自定义块大小。

下载流的速度与我通过 ISP 获得的速度相同。

如果你有更好的我愿意接受任何答案。

于 2013-08-06T13:29:32.603 回答