Постепенная потоковая передача 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 загружается в байтах, это невозможно. Вот почему я спрашиваю здесь.

Что я могу сделать (если что-нибудь), чтобы разрешить прогрессивную потоковую передачу?
Есть ли более подходящий формат контейнера, который я мог бы использовать, который служил бы той же цели?


person Jamie Taylor    schedule 25.02.2014    source источник
comment
Лучшее, что я думаю, - это найти библиотеку для PHP, предназначенную для работы с видеофайлами, поддерживающими формат mp4. Тогда я бы подумал, что вам почти нужен вызов AJAX, который может обновить поток до точки, выбранной в плеере, или перезагрузить его в браузере.   -  person Patrick    schedule 25.02.2014


Ответы (5)


Я думаю, что у вас концептуальный провал. Вы обрабатываете файл mp4, как если бы он был «файлом необработанных данных». Я пытаюсь объясниться.

Представьте, что у вас есть текстовый файл, и вы хотите получить символы из позиции X. Вы можете открыть файл, навести курсор на правильную позицию и затем побайтно читать текст. Это будет работать нормально.

Но теперь изображение, которое вы хотите сделать то же самое, но с файлом текстового процессора. Вы бы ожидали таких же результатов? Нет, потому что у вас в файле много метаданных, которые мешают вам это сделать.

Ваша проблема в основном такая же. Вам необходимо учитывать формат файла, управлять файлом с помощью библиотек, предназначенных для этого, или делать это самостоятельно.

Другой вариант - работать с файлами сырых данных, но в случае с видеофайлами это будут действительно большие файлы.

Я надеюсь, что помог тебе.

person dlopez    schedule 25.02.2014
comment
Да, это то, чего я боялся. Думаю, мне придется использовать флэш-решение, бу. Спасибо за Ваш ответ! - person Jamie Taylor; 26.02.2014

В дополнение к тому, что сказал dlopez, я рекомендую использовать стороннее решение для прогрессивной загрузки с возможностями поиска (псевдопотоковая передача AKA). Вы можете ознакомиться с решениями PD, перечисленными в Википедии: https://en.wikipedia.org/wiki/Progressive_download

Большинство из них также могут предотвратить защиту от хотлинкинга видео.

person Dmitry Shovtyuk    schedule 26.02.2014

Я столкнулся с похожей проблемой. Использование stream_get_content($fp, $start) вместо fseek решило проблему. Мне удалось настроить потоковую передачу видео во всех браузерах, и поиск (или пропуск) по всему видео работал без проблем.

person C3RO14    schedule 15.01.2016

Полнофункциональный код

$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;
}
person iamnamrud    schedule 29.05.2017

Вы не можете использовать fseek (5654), если используете путь к файлу с http://example.com ... ... Вместо этого вы должны использовать c: /file/abc.mp4. Это может решить вашу проблему ...

person Shambhulal Verma    schedule 22.06.2014