php获取hls流媒体信息需先解析.m3u8文件。1. 使用file_get_contents()读取m3u8文件内容;2. 通过explode()逐行解析文本;3. 提取关键标签如#ext-x-version、#ext-x-media-sequence、#ext-x-targetduration及#extinf等;4. 处理相对url转换为绝对路径;5. 若存在aes-128加密,解析#ext-x-key获取密钥url并解密媒体片段;6. 针对多码率流,识别#ext-x-stream-inf标签并递归解析不同码率的m3u8文件;7. 性能优化方面采用流式读取、正则表达式优化、缓存解析结果及高效字符串处理函数以提升效率。
PHP获取HLS流媒体信息,核心在于解析HLS的索引文件(.m3u8文件)。这文件里包含了视频分片和其他必要信息,我们可以用PHP读取并处理它。
解决方案:
- 读取M3U8文件内容
使用file_get_contents()函数读取M3U8文件的URL,将其内容保存到字符串中。
立即学习“PHP免费学习笔记(深入)”;
<?php $m3u8_url = 'YOUR_M3U8_URL'; $m3u8_content = file_get_contents($m3u8_url); if ($m3u8_content === false) { die('Failed to fetch M3U8 file.'); } // 现在 $m3u8_content 包含了 M3U8 文件的内容 ?>
- 解析M3U8内容
M3U8文件是文本文件,我们需要逐行解析。使用explode()函数按行分割字符串。
<?php $lines = explode("n", $m3u8_content); ?>
- 提取关键信息
遍历每一行,根据M3U8文件的格式提取信息。常见的标签包括:
- #EXTM3U: 文件头,表示这是一个M3U8文件。
- #EXT-X-VERSION: HLS协议版本。
- #EXT-X-MEDIA-SEQUENCE: 第一个媒体片段的序列号。
- #EXT-X-TARGETDURATION: 每个媒体片段的目标持续时间(秒)。
- #EXTINF: 媒体片段的信息,包括持续时间和文件名。
- #EXT-X-ENDLIST: 表示流的结束。
<?php $version = null; $media_sequence = null; $target_duration = null; $segments = []; foreach ($lines as $line) { $line = trim($line); // 去除空白 if (strpos($line, '#EXT-X-VERSION') === 0) { $version = intval(substr($line, strlen('#EXT-X-VERSION:'))); } elseif (strpos($line, '#EXT-X-MEDIA-SEQUENCE') === 0) { $media_sequence = intval(substr($line, strlen('#EXT-X-MEDIA-SEQUENCE:'))); } elseif (strpos($line, '#EXT-X-TARGETDURATION') === 0) { $target_duration = intval(substr($line, strlen('#EXT-X-TARGETDURATION:'))); } elseif (strpos($line, '#EXTINF') === 0) { // 获取片段持续时间 preg_match('/EXTINF:([d.]+)/', $line, $matches); $duration = floatval($matches[1]); // 获取片段文件名 (下一行) $segment_url = trim(next($lines)); $segments[] = ['duration' => $duration, 'url' => $segment_url]; } elseif (strpos($line, '#EXT-X-STREAM-INF') === 0) { //处理多码率情况,提取不同码率的url preg_match('/URI="([^"]+)"/', $line, $matches); if(isset($matches[1])){ $variant_url = $matches[1]; //这里可以递归调用这个函数,解析不同码率的m3u8文件 } } } // 现在你可以使用这些信息了 var_dump($version, $media_sequence, $target_duration, $segments); ?>
- 处理相对URL
如果媒体片段的URL是相对路径,需要将其转换为绝对路径。
<?php function makeAbsoluteUrl($base_url, $relative_url) { // 使用PHP内置函数处理 return SabreUriresolve($base_url, $relative_url); } $base_url = dirname($m3u8_url) . '/'; // M3U8文件所在的目录 foreach ($segments as &$segment) { $segment['url'] = makeAbsoluteUrl($base_url, $segment['url']); } ?>
如何处理M3U8文件中的加密内容(如AES-128)?
M3U8文件中可能包含加密信息,例如使用AES-128加密。 遇到这种情况,你需要:
-
查找密钥URL:M3U8文件中会有一个#EXT-X-KEY标签,其中包含密钥的URL。
#EXT-X-KEY:METHOD=AES-128,URI="YOUR_KEY_URL",IV=0x...
-
获取密钥:使用file_get_contents()从密钥URL获取密钥。
<?php // 假设你已经获取了密钥和IV $key_url = 'YOUR_KEY_URL'; $key = file_get_contents($key_url); //IV 一般在EXT-X-KEY中指定,如果没有,则默认为0 $iv = 'YOUR_IV'; // 使用 openssl 解密 $encrypted_data = file_get_contents('YOUR_ENCRYPTED_TS_FILE'); $decrypted_data = openssl_decrypt($encrypted_data, 'aes-128-cbc', $key, OPENSSL_RAW_DATA, $iv); if ($decrypted_data === false) { echo "解密失败: " . openssl_error_string() . "n"; } else { // 保存解密后的数据 file_put_contents('YOUR_DECRYPTED_TS_FILE', $decrypted_data); } ?>
如何处理多码率(Adaptive Bitrate Streaming)的M3U8文件?
多码率HLS使用#EXT-X-STREAM-INF标签来描述不同的码率流。每个#EXT-X-STREAM-INF标签后面跟着一个M3U8文件的URL,这个文件描述了特定码率的流。
-
提取码率信息:解析#EXT-X-STREAM-INF标签,提取带宽、分辨率等信息,以及对应的M3U8 URL。
-
选择合适的码率:根据网络状况和设备性能,选择合适的码率。
-
递归解析:下载并解析选定的M3U8文件,获取媒体片段信息。
<?php // 在主循环中,处理 #EXT-X-STREAM-INF elseif (strpos($line, '#EXT-X-STREAM-INF') === 0) { preg_match('/BANDWIDTH=(d+),.*URI="([^"]+)"/', $line, $matches); $bandwidth = intval($matches[1]); $variant_url = trim($matches[2]); // 递归调用函数,解析不同码率的m3u8文件 $variant_data = parseM3U8($variant_url); $variants[] = ['bandwidth' => $bandwidth, 'url' => $variant_url, 'data' => $variant_data]; } //辅助函数,递归解析不同码率的m3u8文件 function parseM3U8($url){ $m3u8_content = file_get_contents($url); //... 这里可以复用之前的解析逻辑 return $m3u8_content; } ?>