복붙노트

PHP를 사용하여 큰 파일 스트리밍

PHP

PHP를 사용하여 큰 파일 스트리밍

200MB 파일을 다운로드하여 사용자에게 제공하고 싶습니다. 그러나 사용자가이 파일을 한 번만 다운로드하도록하기 때문에 다음을 수행합니다.

echo file_get_contents('http://some.secret.location.com/secretfolder/the_file.tar.gz');

강제로 다운로드합니다. 그러나 이것은 일반적으로 작동하지 않는 전체 파일을 메모리에로드해야 함을 의미합니다. 청크 당 몇 kb에서이 파일을 어떻게 스트리밍 할 수 있습니까?

해결법

  1. ==============================

    1.다음과 같이 시도해보십시오 (출처 : http://teddy.fr/2007/11/28/how-serve-big-files-through-php/).

    다음과 같이 시도해보십시오 (출처 : http://teddy.fr/2007/11/28/how-serve-big-files-through-php/).

    <?php
    define('CHUNK_SIZE', 1024*1024); // Size (in bytes) of tiles chunk
    
    // Read a file and display its content chunk by chunk
    function readfile_chunked($filename, $retbytes = TRUE) {
        $buffer = '';
        $cnt    = 0;
        $handle = fopen($filename, 'rb');
    
        if ($handle === false) {
            return false;
        }
    
        while (!feof($handle)) {
            $buffer = fread($handle, CHUNK_SIZE);
            echo $buffer;
            ob_flush();
            flush();
    
            if ($retbytes) {
                $cnt += strlen($buffer);
            }
        }
    
        $status = fclose($handle);
    
        if ($retbytes && $status) {
            return $cnt; // return num. bytes delivered like readfile() does.
        }
    
        return $status;
    }
    
    // Here goes your code for checking that the user is logged in
    // ...
    // ...
    
    if ($logged_in) {
        $filename = 'path/to/your/file';
        $mimetype = 'mime/type';
        header('Content-Type: '.$mimetype );
        readfile_chunked($filename);
    
    } else {
        echo 'Tabatha says you haven\'t paid.';
    }
    ?>
    
  2. ==============================

    2.fpassthru ()를 사용하십시오. 이름에서 알 수 있듯이 파일을 보내기 전에 전체 파일을 메모리로 읽어 들이지 않고 곧바로 클라이언트로 출력합니다.

    fpassthru ()를 사용하십시오. 이름에서 알 수 있듯이 파일을 보내기 전에 전체 파일을 메모리로 읽어 들이지 않고 곧바로 클라이언트로 출력합니다.

    설명서의 예에서 수정되었습니다.

    <?php
    
    // the file you want to send
    $path = "path/to/file";
    
    // the file name of the download, change this if needed
    $public_name = basename($path);
    
    // get the file's mime type to send the correct content type header
    $finfo = finfo_open(FILEINFO_MIME_TYPE);
    $mime_type = finfo_file($finfo, $path);
    
    // send the headers
    header("Content-Disposition: attachment; filename=$public_name;");
    header("Content-Type: $mime_type");
    header('Content-Length: ' . filesize($path));
    
    // stream the file
    $fp = fopen($path, 'rb');
    fpassthru($fp);
    exit;
    

    다운로드가 아닌 브라우저로 직접 내용을 스트리밍하려는 경우 (그리고 비디오, 오디오, pdf 등의 브라우저에서 지원하는 컨텐츠 유형 인 경우) Content-Disposition 헤더를 제거하십시오.

  3. ==============================

    3.fsockopen () 매뉴얼 페이지의 예제를 보자.

    fsockopen () 매뉴얼 페이지의 예제를 보자.

    $fp = fsockopen("www.example.com", 80, $errno, $errstr, 30);
    if (!$fp) {
        echo "$errstr ($errno)<br />\n";
    } else {
        $out = "GET / HTTP/1.1\r\n";
        $out .= "Host: www.example.com\r\n";
        $out .= "Connection: Close\r\n\r\n";
        fwrite($fp, $out);
        while (!feof($fp)) {
            echo fgets($fp, 128);
        }
        fclose($fp);
    }
    

    그러면 www.example.com에 연결되어 요청을 보내고 128 바이트 청크로 응답을 보내고 에코합니다. 128 바이트 이상으로 만들 수 있습니다.

  4. ==============================

    4.http://codesamplez.com/programming/php-html5-video-streaming-tutorial에서이 방법을 찾았습니다.

    http://codesamplez.com/programming/php-html5-video-streaming-tutorial에서이 방법을 찾았습니다.

    그리고 그것은 저를 위해 아주 잘 작동합니다.

       <?php
    
    class VideoStream
    {
        private $path = "";
        private $stream = "";
        private $buffer = 102400;
        private $start  = -1;
        private $end    = -1;
        private $size   = 0;
    
        function __construct($filePath) 
        {
            $this->path = $filePath;
        }
    
        /**
         * Open stream
         */
        private function open()
        {
            if (!($this->stream = fopen($this->path, 'rb'))) {
                die('Could not open stream for reading');
            }
    
        }
    
        /**
         * Set proper header to serve the video content
         */
        private function setHeader()
        {
            ob_get_clean();
            header("Content-Type: video/mp4");
            header("Cache-Control: max-age=2592000, public");
            header("Expires: ".gmdate('D, d M Y H:i:s', time()+2592000) . ' GMT');
            header("Last-Modified: ".gmdate('D, d M Y H:i:s', @filemtime($this->path)) . ' GMT' );
            $this->start = 0;
            $this->size  = filesize($this->path);
            $this->end   = $this->size - 1;
            header("Accept-Ranges: 0-".$this->end);
    
            if (isset($_SERVER['HTTP_RANGE'])) {
    
                $c_start = $this->start;
                $c_end = $this->end;
    
                list(, $range) = explode('=', $_SERVER['HTTP_RANGE'], 2);
                if (strpos($range, ',') !== false) {
                    header('HTTP/1.1 416 Requested Range Not Satisfiable');
                    header("Content-Range: bytes $this->start-$this->end/$this->size");
                    exit;
                }
                if ($range == '-') {
                    $c_start = $this->size - substr($range, 1);
                }else{
                    $range = explode('-', $range);
                    $c_start = $range[0];
    
                    $c_end = (isset($range[1]) && is_numeric($range[1])) ? $range[1] : $c_end;
                }
                $c_end = ($c_end > $this->end) ? $this->end : $c_end;
                if ($c_start > $c_end || $c_start > $this->size - 1 || $c_end >= $this->size) {
                    header('HTTP/1.1 416 Requested Range Not Satisfiable');
                    header("Content-Range: bytes $this->start-$this->end/$this->size");
                    exit;
                }
                $this->start = $c_start;
                $this->end = $c_end;
                $length = $this->end - $this->start + 1;
                fseek($this->stream, $this->start);
                header('HTTP/1.1 206 Partial Content');
                header("Content-Length: ".$length);
                header("Content-Range: bytes $this->start-$this->end/".$this->size);
            }
            else
            {
                header("Content-Length: ".$this->size);
            }  
    
        }
    
        /**
         * close curretly opened stream
         */
        private function end()
        {
            fclose($this->stream);
            exit;
        }
    
        /**
         * perform the streaming of calculated range
         */
        private function stream()
        {
            $i = $this->start;
            set_time_limit(0);
            while(!feof($this->stream) && $i <= $this->end) {
                $bytesToRead = $this->buffer;
                if(($i+$bytesToRead) > $this->end) {
                    $bytesToRead = $this->end - $i + 1;
                }
                $data = fread($this->stream, $bytesToRead);
                echo $data;
                flush();
                $i += $bytesToRead;
            }
        }
    
        /**
         * Start streaming video content
         */
        function start()
        {
            $this->open();
            $this->setHeader();
            $this->stream();
            $this->end();
        }
    }
    

    이 클래스를 사용하려면 다음과 같은 간단한 코드를 작성해야합니다.

    $stream = new VideoStream($filePath);
    $stream->start();
    
  5. from https://stackoverflow.com/questions/6914912/streaming-a-large-file-using-php by cc-by-sa and MIT license