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



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

我正在从这里改编一些代码

<?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>标签(和镶边)崩溃。

我假设由于HTML5播放器在几秒钟内搜索并且PHP以字节为单位加载,因此无法做到这一点。这就是我在这里问的原因。

我可以做什么(如果有的话)来允许渐进式流式传输?
是否有更合适的容器格式可供我使用,以达到相同的目的?

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

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

我认为你有一个概念上的失败。您正在将mp4文件视为"原始数据文件"。我试着解释自己。

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

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

你的问题基本是一样的。您需要考虑文件的格式,使用为其设计的库管理文件,或者自己做。

另一种选择是处理原始数据文件,但在视频文件的情况下,这些文件将是非常大的文件。

我希望我能帮助你。

我遇到了类似的问题。使用stream_get_content($fp, $start)而不是fseek解决了问题。我能够让视频在所有浏览器上正常流式传输,并且在整个视频中搜索(或跳过)没有问题。

全功能代码

$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;
}

如果你将文件路径与 http://example.com 一起使用,则不能使用 fseek(5654)。 你应该改用 c:/file/abc.mp4。它可能会解决您的问题...

最新更新