Поток mp4 через файл php постепенно
Я работаю над более безопасным методом потоковой передачи для нашего видеоплеера. Поскольку каждый файл требует специальной аутентификации токена, а также позволяет загружать каждый токен только один раз, я запускаю MP4 через контейнер php. Это прекрасно работает внутри тега видео 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 загружается в байтах, это невозможно. Вот почему я спрашиваю здесь.
Что я могу сделать (если что-нибудь), чтобы позволить прогрессивную потоковую передачу?
Есть ли более подходящий формат контейнера, который я мог бы использовать для той же цели?
5 ответов
Я думаю, что у вас есть концептуальная ошибка. Вы рассматриваете файл mp4, как если бы это был "файл необработанных данных". Я пытаюсь объяснить себя.
Представьте, что у вас есть текстовый файл, и вы хотите получить символы из позиции X. Вы можете открыть файл, навести курсор на правильную позицию и затем прочитать побайтный текст. Это будет работать нормально.
Но теперь изображение, которое вы хотите сделать то же самое, но с файлом текстового процессора. Ожидаете ли вы таких же результатов? Нет, потому что у вас есть много метаданных в файле, которые мешают вам сделать это.
Ваша проблема в основном та же. Вы должны принять во внимание формат файла, управлять файлом с помощью библиотек, предназначенных для него, или делать это самостоятельно.
Другой вариант будет работать с файлами необработанных данных, но в случае видеофайлов это будут действительно большие файлы.
Я надеюсь, что я тебе помог.
В дополнение к тому, что сказал dlopez, я рекомендую использовать стороннее решение для прогрессивной загрузки с возможностями поиска (псевдопоток AKA). Вы можете взглянуть на решения PD, перечисленные в Википедии: https://en.wikipedia.org/wiki/Progressive_download
Большинство из них также могут предотвратить защиту от хотлинкинга видео.
Полностью функциональный код
$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;
}
Я столкнулся с подобной проблемой. С помощью stream_get_content($fp, $start)
вместо fseek
исправил проблему. Мне удалось нормально транслировать видео во всех браузерах, и поиск (или пропуск) по всему видео работал без проблем.
Вы не можете использовать fseek(5654), если вы используете путь к файлу с http://example.com/...... Вместо этого вы должны использовать c:/file/abc.mp4. Это может решить вашу проблему...