php下载在下载过程中失败

时间:2015-07-22 07:56:17

标签: php download

我使用此脚本在php中下载文件 Here是我复制脚本的原始来源。

if(file_exists($path)){
        $file_size = filesize($path);
        $file = fopen($path,"rb");
        $chunksize = 2*1024*1024; // how many bytes per chunk
        if($file){
            header("Content-Description: File Transfer");
            header("Pragma: public");
            header("Expires: 0");
            header("Cache-Control: public, must-revalidate, post-check=0, pre-check=0");
            header("Content-Disposition: attachment; filename=\"$name.$format\"");
            header("Content-Type: $mimetype");

            //check if http_range is sent by browser (or download manager)
            if(isset($_SERVER['HTTP_RANGE'])){
                list($size_unit, $range_orig) = explode('=', $_SERVER['HTTP_RANGE'], 2);
                if ($size_unit == 'bytes'){
                    //multiple ranges could be specified at the same time, but for simplicity only serve the first range
                    //http://tools.ietf.org/id/draft-ietf-http-range-retrieval-00.txt
                    list($range, $extra_ranges) = explode(',', $range_orig, 2);
                }
                else{
                    $range = '';
                    header('HTTP/1.1 416 Requested Range Not Satisfiable');
                    exit;
                }
            }
            else{
                $range = '';
            }

            //figure out download piece from range (if set)
            list($seek_start, $seek_end) = explode('-', $range, 2);

            //set start and end based on range (if set), else set defaults
            //also check for invalid ranges.
            $seek_end   = (empty($seek_end)) ? ($file_size - 1) : min(abs(intval($seek_end)),($file_size - 1));
            $seek_start = (empty($seek_start) || $seek_end < abs(intval($seek_start))) ? 0 : max(abs(intval($seek_start)),0);

            //Only send partial content header if downloading a piece of the file (IE workaround)
            if ($seek_start > 0 || $seek_end < ($file_size - 1)){
                header('HTTP/1.1 206 Partial Content');
                header('Content-Range: bytes '.$seek_start.'-'.$seek_end.'/'.$file_size);
                header('Content-Length: '.($seek_end - $seek_start + 1));
            }
            else
                header("Content-Length: $file_size");

            header('Accept-Ranges: bytes');

            set_time_limit(0);
            @fseek($file, $seek_start);

            while(!feof($file)){
                print(@fread($file, $chunksize));
                ob_flush();
                flush();
                if (connection_status()!=0){
                    @fclose($file);
                    exit;
                }
            }
            @fclose($file);
            exit;
        }
        else{
            // file couldn't be opened
            header("HTTP/1.0 500 Internal Server Error");
            exit;
        }
    }
    else {
        // file does not exist
        header("HTTP/1.0 404 Not Found");
        exit;
    }

现在,问题。
下载大文件(e,g> 20MB)有时下载成功完成。有时它会在下载过程中断开。例如我测试并发现对于100MB文件,下载40MB后失败。 (不是一次,多次。)
在浏览器中,似乎文件已完全下载。 (Firefox和Chrome在下载栏中没有Failed文本。但是&#34;打开包含文件夹&#34;选项存在)
我的一些网络连接速度很慢的朋友说,对于较小的文件(小于1 MB),他也遇到了同样的问题。但我从未体验过它。 (也许是因为我的网络速度不慢)
所以问题是,当连接速度慢的人想要下载时,在中间,它会打破他。 (我猜服务器不再向他发送文件了。)
注1:我也有这两个标题:

Connection: Keep-Alive
Keep-Alive: timeout=5

对于想要测试它的人来说,这是一个示例下载文件: https://goo.gl/9flVng
还有一个问题,是不是更好地将用户直接发送到header: location下载文件而不是readfile(), fread(), ...

对不起,很长的问题,谢谢你的帮助。

1 个答案:

答案 0 :(得分:0)

您可以尝试提高脚本的最大执行时间限制,默认设置为30秒。将其设置为0将删除时间限制。

set_time_limit (0)