我正在使用 PHP 脚本来控制下载文件的访问。这对于 2Gb 以下的任何文件都适用,但对于较大的文件则失败。
- Apache 和 PHP 都是 64 位
- Apache will如果直接访问则允许下载文件(我不允许)
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;
错误日志显示文件大小正常
[Tue Apr 08 11:01:16 2014] [error] [client *.*.*.*] FILETEST: /downloads/file.name : 2251373807, referer: http://myurl/files/
但访问日志的大小为负:
*.*.*.* - - [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"
因此浏览器拒绝下载该文件。事实上,使用 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.
尝试分块读取文件并将其暴露给浏览器,而不是用 2GB 填充本地内存并一次性刷新所有内容。
Replace readfile($path);
by:
@ob_end_flush();
flush();
$fileDescriptor = fopen($file, 'rb');
while ($chunk = fread($fileDescriptor, 8192)) {
echo $chunk;
@ob_end_flush();
flush();
}
fclose($fileDescriptor);
exit;
8192字节在某些情况下是一个临界点,参考php.net/fread http://php.net/fread.
添加一些微时间变量(并与文件描述符的指针位置进行比较)还可以让您控制下载的最大速度。
*(刷新输出缓冲区也稍微取决于网络服务器,使用这些命令以确保它至少尝试尽可能多地刷新。)
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)