逐步通过php文件流式传输mp4

时间:2014-02-25 15:46:41

标签: php html5-video mp4 progressive-download

我正在为我们的视频播放器开发更安全的流媒体方法。因为每个文件都需要特殊的令牌身份验证,并且只允许每个令牌加载一次,所以我通过php容器运行MP4。这在HTML5视频代码中完美运行,可防止用户轻松下载源代码。

我正在调整一些代码from here

<?php
include('...'); //include site functions

/*
   Here I connect to the database, and retrieve the
   location of the video which is returned as
   $video = "http://domain.tld/folder/file.mp4"

   This has been removed for this SO example.
*/

$file = $video;
$fp = @fopen($file, 'rb');

$head = array_change_key_case(get_headers($file, TRUE));
$size = $head['content-length'];


//$size   = filesize($file); // File size
$length = $size;           // Content length
$start  = 0;               // Start byte
$end    = $size - 1;       // End byte
header('Content-type: video/mp4');
//header("Accept-Ranges: 0-$length");
header("Accept-Ranges: bytes");
if (isset($_SERVER['HTTP_RANGE'])) {
    $c_start = $start;
    $c_end   = $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 $start-$end/$size");
        exit;
    }
    if ($range == '-') {
        $c_start = $size - substr($range, 1);
    }else{
        $range  = explode('-', $range);
        $c_start = $range[0];
        $c_end   = (isset($range[1]) && is_numeric($range[1])) ? $range[1] : $size;
    }
    $c_end = ($c_end > $end) ? $end : $c_end;
    if ($c_start > $c_end || $c_start > $size - 1 || $c_end >= $size) {
        header('HTTP/1.1 416 Requested Range Not Satisfiable');
        header("Content-Range: bytes $start-$end/$size");
        exit;
    }
    $start  = $c_start;
    $end    = $c_end;
    $length = $end - $start + 1;
    fseek($fp, $start);
    header('HTTP/1.1 206 Partial Content');
}
header("Content-Range: bytes $start-$end/$size");
header("Content-Length: ".$length);
$buffer = 1024 * 8;
while(!feof($fp) && ($p = ftell($fp)) <= $end) {
    if ($p + $buffer > $end) {
        $buffer = $end - $p + 1;
    }
    set_time_limit(0);
    echo fread($fp, $buffer);
    flush();
}
fclose($fp);
exit();
?>

现在出现问题,因为我希望能够进入视频。跳到视频中尚未加载的区域会导致<video>标记(和chrome)崩溃。

我认为,由于HTML5播放器在几秒钟内寻找并且PHP以字节为单位加载,因此无法完成。这就是我在这里问的原因。

我可以做什么(如果有的话)允许渐进式流媒体?
我可以使用哪种更合适的容器格式可以达到同样的目的吗?

5 个答案:

答案 0 :(得分:4)

除了dlopez所说的内容之外,我建议使用第三方解决方案进行渐进式下载,并具有搜索功能(AKA伪流)。您可以查看维基百科中列出的PD解决方案:https://en.wikipedia.org/wiki/Progressive_download

它们中的大多数也可以防止视频热链接保护。

答案 1 :(得分:2)

我认为你有一个概念上的失败。您正在处理mp4文件,就像它是“原始数据文件”一样。我试着解释一下自己。

想象一下,你有一个文本文件,并且想要从位置X获取字符。你可以打开文件,将光标指向正确的位置,然后逐字节读取文本。这样可以正常工作。

但现在想要使用文本处理器文件执行相同操作的图像。你会期待同样的结果吗?不,因为文件中有很多元数据阻止你这样做。

你的问题基本相同。您需要考虑文件的格式,使用为其设计的库来管理文件,或者自己完成。

另一种选择是使用原始数据文件,但对于视频文件,这些文件将是非常大的文件。

我希望我帮助过你。

答案 2 :(得分:0)

如果您使用http://example.com的文件路径,则无法使用fseek(5654)......您应该使用c:/file/abc.mp4。它可以解决你的问题...

答案 3 :(得分:0)

我遇到了类似的问题。使用stream_get_content($fp, $start)代替fseek修复了问题。我能够在所有浏览器上播放视频,并在整个视频中搜索(或跳过)没有问题。

答案 4 :(得分:0)

功能齐全的代码

$file = "z.mp4";
$length= filesize($file);
$offset = 0;
$f = fopen($file, 'r');
stream_get_contents($f, $offset);
$pos = 0;
while($pos < $length){
     $chunk = min($length-$pos, 1024*8);
     echo fread($f, $chunk);
     flush();
     ob_flush();
     $pos += $chunk;
}