我正在使用PHP脚本来控制对下载文件的访问.这适用于2Gb以下的任何内容,但对于较大的文件则无效.
PHP的内核(忽略访问控制):
if (ob_get_level()) ob_end_clean();
error_log('FILETEST: '.$path.' : '.filesize($path));
header('Content-Description: File Transfer');
header('Content-Type: application/octet-stream');
header('Content-Disposition: attachment; filename='.basename($path));
header('Expires: 0');
header('Cache-Control: must-revalidate');
header('Pragma: public');
header('Content-Length: ' . filesize($path));
readfile($path);
exit;
Run Code Online (Sandbox Code Playgroud)
错误日志显示文件大小正常
[Tue Apr 08 11:01:16 2014] [error] [client *.*.*.*] FILETEST: /downloads/file.name : 2251373807, referer: http://myurl/files/
Run Code Online (Sandbox Code Playgroud)
但访问日志的大小为负:
*.*.*.* - - [08/Apr/2014:11:01:16 +0100] "GET /files/file.name HTTP/1.1" 200 -2043593489 "http://myurl/files/" "Mozilla/5.0 (Windows NT 6.1; WOW64; rv:24.0) Gecko/20100101 Firefox/24.0"
Run Code Online (Sandbox Code Playgroud)
因此浏览器拒绝下载该文件.事实上,使用wget,它不会发送任何东西:
$ wget -S -O - http://myurl/files/file.name
--2014-04-08 11:33:38-- http://myurl/files/file.name
HTTP request sent, awaiting response... No data received.
Retrying.
Run Code Online (Sandbox Code Playgroud)
尝试以块的形式读取文件并将它们暴露给浏览器,而不是用2GB填充本地内存并立即刷新所有文件.
替换readfile($path);为:
@ob_end_flush();
flush();
$fileDescriptor = fopen($file, 'rb');
while ($chunk = fread($fileDescriptor, 8192)) {
echo $chunk;
@ob_end_flush();
flush();
}
fclose($fileDescriptor);
exit;
Run Code Online (Sandbox Code Playgroud)
在某些情况下,8192字节是关键点,参考php.net/fread.
添加一些microtime变量(并与文件描述符的指针位置进行比较)也可以控制下载的最大速度.
*(刷新输出缓冲区也略微取决于Web服务器,使用这些命令确保它至少尝试尽可能地刷新.)