PHP를 이용한 대용량 파일 스트리밍
다운로드로 사용자에게 주고 싶은 200MB 파일이 있습니다.그러나 사용자가 이 파일을 한 번만 다운로드하기를 원하므로 다음 작업을 수행합니다.
echo file_get_contents('http://some.secret.location.com/secretfolder/the_file.tar.gz');
다운로드를 강제합니다.그러나 이는 전체 파일을 메모리에 로드해야 한다는 것을 의미하며, 이는 일반적으로 작동하지 않습니다.청크당 KB 단위로 이 파일을 어떻게 스트리밍할 수 있습니까?
이와 같은 것을 시도해 보세요 (source 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.';
}
?>
사용합니다. 이름에서 알 수 있듯이 파일을 보내기 전에 전체 파일을 메모리로 읽는 것이 아니라 클라이언트에 바로 출력합니다.
설명서의 예제에서 수정:
<?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 등), 콘텐츠 배치 헤더를 제거합니다.
의 매뉴얼 페이지에서 예제를 살펴봅니다.
$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바이트 이상으로 하고 싶을 수도 있습니다.
저는 이 방법을 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();
읽기 파일()을 사용하여 다운로드를 강제하기도 했습니다.메모리 문제는 읽기 파일 내에 있는 것이 아니라 출력 버퍼링에 있습니다.
파일을 읽기 전에 출력 버퍼링을 끄는지 확인하고 문제를 해결해야 합니다.
if (ob_get_level()) ob_end_clean();
readfile($yourfile);
할당된 메모리 제한보다 크기가 훨씬 큰 파일에 대해 작동합니다.
언급URL : https://stackoverflow.com/questions/6914912/streaming-a-large-file-using-php
'programing' 카테고리의 다른 글
| 더블/플로트를 문자열로 변환 (0) | 2023.10.19 |
|---|---|
| 여러 프로젝트가 node_modules 디렉토리를 공유하도록 하려면 어떻게 해야 합니까? (0) | 2023.10.19 |
| .map() 자바스크립트 ES6 지도? (0) | 2023.10.19 |
| C++ 표준 라이브러리가 OS 대신 컴파일러와 함께 번들로 제공되는 이유는 무엇입니까? (0) | 2023.10.19 |
| 마이그레이션 수준 5.1에서 자동 증분 필드 시작을 1000부터 설정 (0) | 2023.10.19 |