1. 程式人生 > >【nginx】記錄nginx+php-fpm實現大檔案下載排坑的過程

【nginx】記錄nginx+php-fpm實現大檔案下載排坑的過程

先上一段程式碼,支援大檔案下載和斷點續傳,程式碼來源網際網路。


set_time_limit(0);

// 省略取檔案路徑的過程,這裡直接是檔案完整路徑
$filePath = get_save_path() . $File['save_name'];
$filePath = realpath($filePath);

$outFileExtension = strtolower(substr(strrchr($filePath, "."), 1)); //獲取副檔名
//根據副檔名 指出輸出瀏覽器格式
switch ($outFileExtension) {
    case "exe" :
        $ctype = "application/octet-stream";
        break;
    case "zip" :
        $ctype = "application/zip";
        break;
    case "mp3" :
        $ctype = "audio/mpeg";
        break;
    case "mpg" :
        $ctype = "video/mpeg";
        break;
    case "avi" :
        $ctype = "video/x-msvideo";
        break;
    default :
        $ctype = "application/force-download";
}

header("Cache-Control:");
header("Cache-Control: public");

//設定輸出瀏覽器格式
header("Content-Type: $ctype");
header("Content-Disposition: attachment; filename=" . basename($filePath));
header("Accept-Ranges: bytes");
$size = filesize($filePath);

//如果有$_SERVER['HTTP_RANGE']引數
if (isset ($_SERVER['HTTP_RANGE'])) {
    /*Range頭域   Range頭域可以請求實體的一個或者多個子範圍。
    例如,
    表示頭500個位元組:bytes=0-499
    表示第二個500位元組:bytes=500-999
    表示最後500個位元組:bytes=-500
    表示500位元組以後的範圍:bytes=500-   
    第一個和最後一個位元組:bytes=0-0,-1   
    同時指定幾個範圍:bytes=500-600,601-999   
    但是伺服器可以忽略此請求頭,如果無條件GET包含Range請求頭,響應會以狀態碼206(PartialContent)返回而不是以200 (OK)。
    */
    // 斷點後再次連線 $_SERVER['HTTP_RANGE'] 的值 bytes=4390912-
    list ($a, $range) = explode("=", $_SERVER['HTTP_RANGE']);
    //if yes, download missing part
    str_replace($range, "-", $range); //這句幹什麼的呢。。。。
    $size2 = $size - 1; //檔案總位元組數
    $new_length = $size2 - $range; //獲取下次下載的長度
    header("HTTP/1.1 206 Partial Content");
    header("Content-Length: $new_length"); //輸入總長
    header("Content-Range: bytes $range$size2/$size"); //Content-Range: bytes 4908618-4988927/4988928   95%的時候
} else {
    //第一次連線
    $size2 = $size - 1;
    header("Content-Range: bytes 0-$size2/$size"); //Content-Range: bytes 0-4988927/4988928
    header("Content-Length: " . $size); //輸出總長
}
//開啟檔案
$fp = fopen("$filePath", "rb");
//設定指標位置
if (!empty($range)) {
    fseek($fp, $range);
}

//虛幻輸出
while (!feof($fp)) {
    print (fread($fp, 1024 * 8)); //輸出檔案
    flush(); //輸出緩衝
    ob_flush();
}
fclose($fp);
exit ();

程式碼有詳細的解釋,也很清楚,但是在實際使用時還是小檔案可以下載,大檔案只能下載前半部分或者出現檔案已損壞的情況。檢視nginx日誌發現如下報錯


2018/08/01 07:43:20 [crit] 13906#0: *1479 open() "/usr/local/nginx/fastcgi_temp/0/02/0000000020" failed (13: Permission denied) while reading upstream,

原來在下載大檔案時,檔案大小超過配置的proxy_temp_file_write_size值時,nginx會將檔案寫入到臨時目錄下,如果該目錄沒有許可權,就寫不了,那下載只能下載緩衝區的內容了。
核實/usr/local/nginx/fastcgi_temp/目錄的許可權分組,並不在nginx執行賬號組下,即然知道了問題原因,那就好辦了。給予寫許可權,或者將目錄改為nginx執行賬號組下就OK了


cd /usr/local/nginx/ 
chmod -R 766 fastcgi_temp/
或者
chown -R nginx:nginx fastcgi_temp/ #nginx根據各自情況可能是不能的賬戶
解決PHP超大檔案下載,斷點續傳下載的方法詳解

原文地址:https://segmentfault.com/a/1190000015840045