讓PHP提供更好的檔案下載

餘二五發表於2017-11-16

  一般來說, 我們可以通過直接讓URL指向一個位於Document Root下面的檔案, 來引導使用者下載檔案.

  但是, 這樣做, 就沒辦法做一些統計, 許可權檢查, 等等的工作. 於是, 很多時候, 我們採用讓PHP來做轉發, 為使用者提供檔案下載.

1
2
3
4
5
$file "/tmp/dummy.tar.gz";
 header("Content-type: application/octet-stream");
 header(`Content-Disposition: attachment; filename="` basename($file) . `"`);
 header("Content-Length: "filesize($file));
 readfile($file);

複製程式碼但是這個有一個問題, 就是如果檔案是中文名的話, 有的使用者可能下載後的檔名是亂碼.

  於是, 我們做一下修改(參考: :

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
$file "/tmp/中文名.tar.gz";
  $filename basename($file);
  header("Content-type: application/octet-stream");
  //處理中文檔名
  $ua $_SERVER["HTTP_USER_AGENT"];
  $encoded_filename = urlencode($filename);
  $encoded_filename str_replace("+""%20"$encoded_filename);
  if (preg_match("/MSIE/"$ua)) {
  header(`Content-Disposition: attachment; filename="` $encoded_filename `"`);
  } else if (preg_match("/Firefox/"$ua)) {
  header("Content-Disposition: attachment; filename*="utf8``" $filename `"`);
  } else {
  header(`Content-Disposition: attachment; filename="` $filename `"`);
  }
  header(`Content-Disposition: attachment; filename="` $filename `"`);
  header("Content-Length: "filesize($file));
  readfile($file);


  複製程式碼恩, 現在看起來好多了, 不過還有一個問題, 那就是readfile, 雖然PHP的readfile嘗試實現的儘量高效, 不佔用PHP本身的記憶體, 但是實際上它還是需要採用MMAP(如果支援), 或者是一個固定的buffer去迴圈讀取檔案, 直接輸出.

  輸出的時候, 如果是Apache + PHP mod, 那麼還需要傳送到Apache的輸出緩衝區. 最後才傳送給使用者. 而對於Nginx + fpm如果他們分開部署的話, 那還會帶來額外的網路IO.

  那麼, 能不能不經過PHP這層, 直接讓Webserver直接把檔案傳送給使用者呢?

  今天, 我看到了一個有意思的文章: How I PHP: X-SendFile.

  我們可以使用Apache的module mod_xsendfile, 讓Apache直接傳送這個檔案給使用者:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
$file "/tmp/中文名.tar.gz";
$filename basename($file);
header("Content-type: application/octet-stream");
//處理中文檔名
$ua $_SERVER["HTTP_USER_AGENT"];
$encoded_filename = urlencode($filename);
$encoded_filename str_replace("+""%20"$encoded_filename);
if (preg_match("/MSIE/"$ua)) {
header(`Content-Disposition: attachment; filename="` $encoded_filename `"`);
else if (preg_match("/Firefox/"$ua)) {
header("Content-Disposition: attachment; filename*="utf8``" $filename `"`);
else {
header(`Content-Disposition: attachment; filename="` $filename `"`);
}
header(`Content-Disposition: attachment; filename="` basename($file) . `"`);
//讓Xsendfile傳送檔案
header("X-Sendfile: $file");

 複製程式碼X-Sendfile頭將被Apache處理, 並且把響應的檔案直接傳送給Client.

  Lighttpd和Nginx也有類似的模組, 大家有興趣的可以去找找看

本文轉自 3147972 51CTO部落格,原文連結:http://blog.51cto.com/a3147972/1230950,如需轉載請自行聯絡原作者


相關文章