php - 来自静态文件的 HLS 直播

标签 php amazon-s3 video.js http-live-streaming m3u8

我有一个客户端,其中包含数千个内部流式传输的音频/视频文件,所有这些文件都经过分段 (.ts) 并保存在 S3 存储桶中,并在 SQL 数据库中使用适当的元数据。现在他们要求我创建两个“实时”流,一个用于音频,一个用于视频,他们可以设置并忘记。

不想重新分段所有内容或连接所有我试图破解的“实时”m3u8 文件,该文件会在现有文件中滑动(它们的编码方式完全相同)。

我所做的是生成一个“广播播放列表”,以 40 秒的间隔(每个 m3u8 x3 .ts)保存到数据库中,每个播放列表都标记有开始和结束时间以及适当的 EXT-X-MEDIA-顺序。然后我在 NOW() 之间进行选择并推送文件。

它可以工作,但有时时机正确,它会为第一个和最后一个文件命中相同的分组并缓冲。我可以完全控制播放器 (VideoJS) 和服务器以使其正常工作。

这是我到目前为止所拥有的代码...有什么方法可以使其工作吗?我还没有尝试过在 vJS 上使用缓冲区(不知道如何...)

所有的基本文件信息都这样存储在数据库中

INSERT INTO `contenido_audio_hls` (`id`, `audio_s`, `duration`) VALUES ('f2z7dcwc0l7rleig', '["10.000000","10.000000","10.000000","10.000000","10.000000","10.000000","10.000000","10.000000","10.000000","10.000000","10.000000","10.000000","10.000000","10.000000","10.000000","10.000000","10.000000","10.000000","10.000000","4.100000"]', 10);

生成播放列表后,我提取所需的数据

$radio = sql("SELECT lista_contenido.orden,lista_contenido.contenido,contenido_audio_hls.audio_s FROM lista_listas LEFT JOIN lista_contenido ON (lista_listas.id = lista_contenido.lista) LEFT JOIN contenido_audio_hls ON (lista_contenido.contenido = contenido_audio_hls.id) WHERE (lista_listas.tipo = 'radio') ORDER BY lista_contenido.orden ASC");
foreach($radio['data'] as $k=>$v) {
    $arreglo = json_decode($v['audio_s'],TRUE);
    foreach($arreglo as $kk=>$vv) {
        $puro[] = array("extinf"=>'#EXTINF:'.$vv.',',"id"=>$v['contenido'],"segment"=>$kk);
    }
}

我循环遍历它们来创建组

$segundos = 0;
$grupo = 1;
$contador = 1;
foreach($puro as $k=>$v) {
    if($segundos <= 30) {
        $m3u8[$grupo][] = $puro[$k];
        $contador++;
    } else {
        $m3u8[$grupo][] = $puro[$k];
        $grupo = $grupo + $contador;
        $segundos = 0;
    }
    $segundos = $segundos + 10;
}

然后将它们放入自己的表中

$largo = 0;
foreach($m3u8 as $k=>$v) {
    $ini = sprintf('%02d:%02d:%02d',($largo/3600),($largo/60%60),$largo%60);
    $localfin = $largo + 40;
    $fin = sprintf('%02d:%02d:%02d',($localfin/3600),($localfin/60%60),$localfin%60);

    $query = "INSERT INTO lista_m3u8 (ini,fin,tipo,sequence,data) VALUES('".$ini."','".$fin."','radio','".$k."','".json_encode($v)."')";

    sql($query);

    $largo = $largo + 40;
}

这给了我这个

INSERT INTO `lista_m3u8` (`ini`, `fin`, `tipo`, `sequence`, `data`) VALUES ('06:54:00', '06:54:40', 'radio', 580636, '[{"extinf":"#EXTINF:10.000000,","id":"f2z7de0quwgehw23","segment":14},{"extinf":"#EXTINF:10.000000,","id":"f2z7de0quwgehw23","segment":15},{"extinf":"#EXTINF:10.000000,","id":"f2z7de0quwgehw23","segment":16},{"extinf":"#EXTINF:10.000000,","id":"f2z7de0quwgehw23","segment":17}]');
INSERT INTO `lista_m3u8` (`ini`, `fin`, `tipo`, `sequence`, `data`) VALUES ('06:54:40', '06:55:20', 'radio', 582504, '[{"extinf":"#EXTINF:10.000000,","id":"f2z7de0quwgehw23","segment":18},{"extinf":"#EXTINF:10.000000,","id":"f2z7de0quwgehw23","segment":19},{"extinf":"#EXTINF:0.766667,","id":"f2z7de0quwgehw23","segment":20},{"extinf":"#EXTINF:10.000000,","id":"f2z7dft8c217xyp","segment":0}]');
INSERT INTO `lista_m3u8` (`ini`, `fin`, `tipo`, `sequence`, `data`) VALUES ('06:55:20', '06:56:00', 'radio', 584375, '[{"extinf":"#EXTINF:10.000000,","id":"f2z7dft8c217xyp","segment":1},{"extinf":"#EXTINF:10.000000,","id":"f2z7dft8c217xyp","segment":2},{"extinf":"#EXTINF:10.000000,","id":"f2z7dft8c217xyp","segment":3},{"extinf":"#EXTINF:10.000000,","id":"f2z7dft8c217xyp","segment":4}]');
INSERT INTO `lista_m3u8` (`ini`, `fin`, `tipo`, `sequence`, `data`) VALUES ('06:56:00', '06:56:40', 'radio', 586249, '[{"extinf":"#EXTINF:10.000000,","id":"f2z7dft8c217xyp","segment":5},{"extinf":"#EXTINF:10.000000,","id":"f2z7dft8c217xyp","segment":6},{"extinf":"#EXTINF:10.000000,","id":"f2z7dft8c217xyp","segment":7},{"extinf":"#EXTINF:10.000000,","id":"f2z7dft8c217xyp","segment":8}]');

然后m3u8就生成了

$audio = sql("SELECT sequence, data FROM lista_m3u8 WHERE tipo = 'radio' AND ini <= DATE_FORMAT(NOW(),'%H:%i:%s') AND fin >= DATE_FORMAT(NOW(),'%H:%i:%s')");

$sale = '#EXTM3U'.PHP_EOL;
$sale .= '#EXT-X-VERSION:3'.PHP_EOL;
$sale .= '#EXT-X-MEDIA-SEQUENCE:'.$audio['data'][0]['sequence'].PHP_EOL;
$sale .= '#EXT-X-TARGETDURATION:10'.PHP_EOL;

$arreglo = json_decode($audio['data'][0]['data'],TRUE);
foreach($arreglo as $k=>$v) {
    $sale .= $v['extinf'].PHP_EOL;
    $sale .= S3URL("bucket-audio",$v['id']."/segment".sprintf('%05d',$v['segment']).".ts",(count($arreglo) * 25)).PHP_EOL;
}

header("Content-type: application/x-mpegURL");
echo $sale.PHP_EOL;

最佳答案

我相信我已经解决了这个问题。我刚把这个游戏玩了 16 个小时回来,它仍在继续,我的 AWS 日志证实了这一点。

我最初以错误的方式处理这个问题,试图生成 jar 装的 m3u8 文件;我真正需要做的是知道两件事:

1.- 现在应该播放哪个片段(无论原始文件是什么)?

2.- 从“流”开始(文件 0,片段 0)会播放多少片段?

新方法现在采用原始播放列表并为每个片段创建一行,指示其开始时间、持续时间、片段文件和在流中的位置。然后生成 m3u8,后面有几个段,后面有几个段,从流的开头计算正确的 EXT-X-MEDIA-SEQUENCE。我还在文件之间添加了 EXT-X-DISCONTINUITY,这样它就不会因接收意外 header 而挂起。

现在,我从原始表中获取文件/段列表:

$ini = 0;
$conteo = 0;
$radio = sql("SELECT lista_contenido.orden,lista_contenido.contenido,contenido_audio_hls.audio_s FROM lista_listas LEFT JOIN lista_contenido ON (lista_listas.id = lista_contenido.lista) LEFT JOIN contenido_audio_hls ON (lista_contenido.contenido = contenido_audio_hls.id) WHERE (lista_listas.tipo = 'radio') ORDER BY lista_contenido.orden ASC");
foreach($radio['data'] as $k=>$v) {
    $arreglo = json_decode($v['audio_s'],TRUE);
    $seg = 0;
    foreach($arreglo as $kk=>$vv) {
        sql("INSERT INTO lista_m3u8 (tipo,orden,contenido,segmento,extinf,ini) VALUES('radio','".$conteo."','".$v['contenido']."','segment".sprintf('%05d',$seg).".ts','".$vv."','".sprintf('%02d:%02d:%02d',($ini/3600),($ini/60%60),$ini%60)."')");
        $ini = $ini + ceil($vv * 1);
        $seg++;
        $conteo++;
    }
}

这给了我一个像这样的表格:

INSERT INTO `lista_m3u8` (`tipo`, `orden`, `contenido`, `segmento`, `extinf`, `ini`) VALUES ('radio', 20, 'f2z7ddw7r6bb7gfy', 'segment00018.ts', 10.000000000000, '00:03:11');
INSERT INTO `lista_m3u8` (`tipo`, `orden`, `contenido`, `segmento`, `extinf`, `ini`) VALUES ('radio', 21, 'f2z7ddw7r6bb7gfy', 'segment00019.ts', 10.000000000000, '00:03:21');
INSERT INTO `lista_m3u8` (`tipo`, `orden`, `contenido`, `segmento`, `extinf`, `ini`) VALUES ('radio', 22, 'f2z7ddw7r6bb7gfy', 'segment00020.ts', 6.066667079926, '00:03:31');
INSERT INTO `lista_m3u8` (`tipo`, `orden`, `contenido`, `segmento`, `extinf`, `ini`) VALUES ('radio', 23, 'f2z7df1bb66be7h3', 'segment00000.ts', 10.000000000000, '00:03:38');
INSERT INTO `lista_m3u8` (`tipo`, `orden`, `contenido`, `segmento`, `extinf`, `ini`) VALUES ('radio', 24, 'f2z7df1bb66be7h3', 'segment00001.ts', 10.000000000000, '00:03:48');
INSERT INTO `lista_m3u8` (`tipo`, `orden`, `contenido`, `segmento`, `extinf`, `ini`) VALUES ('radio', 25, 'f2z7df1bb66be7h3', 'segment00002.ts', 10.000000000000, '00:03:58');
INSERT INTO `lista_m3u8` (`tipo`, `orden`, `contenido`, `segmento`, `extinf`, `ini`) VALUES ('radio', 26, 'f2z7df1bb66be7h3', 'segment00003.ts', 10.000000000000, '00:04:08');

这会创建数千个数据库行(对于 24 小时广播流而言约为 9000 个),但它们按时间索引,因此选择是即时的。

最终的 m3u8 脚本执行以下操作:

$actual = sql("SELECT orden, extinf, contenido, segmento, ini FROM lista_m3u8 WHERE tipo = 'radio' AND ini >= DATE_FORMAT(DATE_SUB(NOW(), INTERVAL 1 MINUTE),'%H:%i:%s') AND ini <= DATE_FORMAT(DATE_ADD(NOW(), INTERVAL 3 MINUTE),'%H:%i:%s') ORDER BY orden ASC");

$sale = '#EXTM3U'.PHP_EOL;
$sale .= '#EXT-X-VERSION:3'.PHP_EOL;
$sale .= '#EXT-X-MEDIA-SEQUENCE:'.($actual['data'][$actual['total']-1]['orden'] - $actual['total']).PHP_EOL;
$sale .= '#EXT-X-TARGETDURATION:10'.PHP_EOL;

$contenido = $actual['data'][0]['contenido'];
foreach($actual['data'] as $k=>$v) {
    if($v['contenido'] != $contenido) { $sale .= "#EXT-X-DISCONTINUITY".PHP_EOL; }
    $sale .= "#EXTINF:".$v['extinf'].",".PHP_EOL;
    $sale .= S3URL("audio-bucket",$v['contenido']."/".$v['segmento'],180).PHP_EOL;
    $contenido = $v['contenido'];
}

header("Content-type: application/x-mpegURL");
echo $sale.PHP_EOL;

请注意这里发生的两件事,EXT-X-MEDIA-SEQUENCE 是通过从整个列表中减去当前段的位置来计算的,而 XT-X-DISCONTINUITY 则放在文件更改之间。

我将进行更多测试,看看这是否适用于跨浏览器(到目前为止我只测试了 Chrome 和 IEG);但我相信这是一个可行的解决方案。

关于php - 来自静态文件的 HLS 直播,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/37347894/

相关文章:

amazon-s3 - Amazon S3 SelectObjectContent 错误 - OverMaxParquetBlockSize

wordpress - Chrome上的video.js WordPress插件仅音频(无视频)

java - 数据库到android,用php

php - 如何根据一周的下一个周日查找数据

node.js - 如何使用 node、createPresignedPost 和 fetch 将图像文件直接从客户端上传到 AWS S3

amazon-web-services - 如何获取S3存储桶上每个文件夹的总对象数和总存储量

javascript - 在 Bootstrap 模式内显示视频 js 视频时出现问题

javascript - 如何在视频js中在时间间隔后显示弹出窗口

php - 处理 Zend Framework 1 中消失的 MySQL

php - 具有 EC2 AssumeRole 的 AWS STS 角色