HTTP 정지점 전송(PHP 구현)

16929 단어 php 정지점 전송
1.단점 속전 원리
정지점 전송 이란 파일 이 다운 로드 된 곳 부터 계속 다운로드 해 야 한 다 는 것 이다.이전 버 전의 HTTP 프로 토 콜 은 정지점 을 지원 하지 않 았 고 HTTP/1.1 부터 지원 되 었 습 니 다.일반적으로 정지점 에서 다운로드 할 때 만 Range 와 Content-Range 실체 헤드 를 사용 합 니 다.
1.1 정지점 리 셋 사용 하지 않 음
get /down.zip http/1.1
accept: image/gif, image/x-xbitmap, image/jpeg, image/pjpeg, application/vnd.ms-
excel, application/msword, application/vnd.ms-powerpoint, */* accept-language: zh-cn accept-encoding: gzip, deflate user-agent: mozilla/4.0 (compatible; msie 5.01; windows nt 5.0) connection: keep-alive

서버 가 요청 을 받 은 후 요청 한 파일 을 찾 아 파일 의 정 보 를 추출 한 후 브 라 우 저 에 게 돌아 가 다음 과 같은 정 보 를 되 돌려 줍 니 다.
HTTP/1.1 200 Ok
content-length=106786028
accept-ranges=bytes
date=mon, 30 apr 2001 12:56:11 gmt
etag=w/"02ca57e173c11:95b"
content-type=application/octet-stream
server=microsoft-iis/5.0
last-modified=mon, 30 apr 2001 12:56:11 gmt

1.2 정지점 전송 사용
GET /down.zip HTTP/1.0
User-Agent: NetFox
RANGE: bytes=2000070-
Accept: text/html, image/gif, image/jpeg, *; q=.2, */*; q=.2

이 줄 이 많아 졌 습 니 다Range: bytes=2000070-이 줄 은 서버 다운.zip 라 는 파일 이 200070 바이트 부터 전달 되 고 앞 에 있 는 바이트 가 전달 되 지 않 아 도 된다 는 뜻 입 니 다.Range 의 전체 형식 은:
Range: bytes=startOffset-targetOffset/sum [   startOffset  ,     targetOffset  ,     sum  ]

Range: bytes=startOffset-targetOffset [         ]

서버 가 이 요청 을 받 은 후 돌아 오 는 정 보 는 다음 과 같 습 니 다.
HTTP/1.1 206 Partial Content
content-length=106786028
content-range=bytes 2000070-106786027/106786028
date=mon, 30 apr 2001 12:55:20 gmt
etag=w/"02ca57e173c11:95b"
content-type=application/octet-stream
server=microsoft-iis/5.0
last-modified=mon, 30 apr 2001 12:55:20 gmt

이전 서버 에서 돌아 온 정보 와 비교 해 보면 한 줄 이 추 가 된 것 을 발견 할 수 있 습 니 다.
Content-Range=bytes 2000070-106786027/106786028

돌아 오 는 코드 도 200 이 아니 라 206 으로 바 뀌 었 다.
HTTP/1.1 206 Partial Content

이상 의 원 리 를 알 게 되면 정지점 전송 프로 그래 밍 을 진행 할 수 있다.
2.PHP 구현
/** php   ,       * download:      * setSpeed:        * getRange:   header Range */

class FileDownload{

    /**    * @param String $file          * @param String $name     ,              * @param boolean $reload          */
    public function download($file, $name='', $reload=false){
        $fp = @fopen($file, 'rb');
        if($fp){
            if($name==''){
                $name = basename($file);
            }
            $header_array = get_headers($file, true);
            //var_dump($header_array);die;
            //       ,      
            if (!$header_array) {
                $file_size = filesize($file);
            } else {
                $file_size = $header_array['Content-Length'];
            }
            $ranges = $this->getRange($file_size);
            $ua = $_SERVER["HTTP_USER_AGENT"];//          
            header('cache-control:public');
            header('content-type:application/octet-stream');    

            $encoded_filename = urlencode($name);
            $encoded_filename = str_replace("+", "%20", $encoded_filename);

            //         
            if (preg_match("/MSIE/", $ua) ||  preg_match("/Trident/", $ua) ){               
                header('Content-Disposition: attachment; filename="' .$encoded_filename . '"');
            } else if (preg_match("/Firefox/", $ua)) {
                header('Content-Disposition: attachment; filename*="utf8\'\'' . $name . '"');
            }else if (preg_match("/Chrome/", $ua)) {
                header('Content-Disposition: attachment; filename="' . $encoded_filename . '"');
            } else {
                header('Content-Disposition: attachment; filename="' . $name . '"');
            }
            //header('Content-Disposition: attachment; filename="' . $name . '"');

            if($reload && $ranges!=null){ //     
                header('HTTP/1.1 206 Partial Content');
                header('Accept-Ranges:bytes');

                //     
                header(sprintf('content-length:%u',$ranges['end']-$ranges['start']));

                // range  
                header(sprintf('content-range:bytes %s-%s/%s', $ranges['start'], $ranges['end'], $file_size));
                //file_put_contents('test.log',sprintf('content-length:%u',$ranges['end']-$ranges['start']),FILE_APPEND);
                // fp        
                fseek($fp, sprintf('%u', $ranges['start']));
            }else{
                file_put_contents('test.log','2222',FILE_APPEND);
                header('HTTP/1.1 200 OK');
                header('content-length:'.$file_size);
            }

            while(!feof($fp)){
                //echo fread($fp, round($this->_speed*1024,0));
                //echo fread($fp, $file_size);
                echo fread($fp, 4096);
                ob_flush();
            }

            ($fp!=null) && fclose($fp);
        }else{
            return '';
        }
    }

    /**        * @param int $speed */
    public function setSpeed($speed){
        if(is_numeric($speed) && $speed>16 && $speed<4096){
            $this->_speed = $speed;
        }
    }

    /**   header range   * @param int $file_size      * @return Array */
    private function getRange($file_size){
        //file_put_contents('range.log', json_encode($_SERVER), FILE_APPEND);
        if(isset($_SERVER['HTTP_RANGE']) && !empty($_SERVER['HTTP_RANGE'])){
            $range = $_SERVER['HTTP_RANGE'];
            $range = preg_replace('/[\s|,].*/', '', $range);
            $range = explode('-', substr($range, 6));
            if(count($range)<2){
                $range[1] = $file_size;
            }
            $range = array_combine(array('start','end'), $range);
            if(empty($range['start'])){
                $range['start'] = 0;
            }
            if(empty($range['end'])){
                $range['end'] = $file_size;
            }
            return $range;
        }
        return null;
    }
}

$obj = new FileDownload();
$obj->download('http://down.golaravel.com/laravel/laravel-master.zip','', true);

좋은 웹페이지 즐겨찾기