QT 实现解密m3u8文件

文章目录

    • 概要
    • 如何解密M3U8文件呢
    • 实现思路和代码
      • 序列图
      • 网络请求
      • 解密
    • 结论

概要

视频文件很多已M3U8文件格式来提供,先复习下什么是M3U8文件!用QT的 mutimedia框架来播放视频时,有的视频加载慢,有的视频加载快,为啥?结论再最后

m3u8 文件实质是一个播放列表(playlist),其可能是一个媒体播放列表(Media Playlist),或者是一个主列表(Master Playlist)。但无论是哪种播放列表,其内部文字使用的都是 utf-8 编码。

当 m3u8 文件作为媒体播放列表(Meida Playlist)时,其内部信息记录的是一系列媒体片段资源,顺序播放该片段资源,即可完整展示多媒体资源。其格式如下所示:

#EXTM3U
#EXT-X-TARGETDURATION:10#EXTINF:9.009,
http://media.example.com/first.ts
#EXTINF:9.009,
http://media.example.com/second.ts
#EXTINF:3.003,
http://media.example.com/third.ts

对于点播来说,客户端只需按顺序下载上述片段资源,依次进行播放即可。而对于直播来说,客户端需要 定时重新请求 该 m3u8 文件,看下是否有新的片段数据需要进行下载并播放。

当 m3u8 作为主播放列表(Master Playlist)时,其内部提供的是同一份媒体资源的多份流列表资源(Variant Stream)。其格式如下所示:

#EXTM3U
#EXT-X-STREAM-INF:BANDWIDTH=150000,RESOLUTION=416x234,CODECS="avc1.42e00a,mp4a.40.2"
http://example.com/low/index.m3u8
#EXT-X-STREAM-INF:BANDWIDTH=240000,RESOLUTION=416x234,CODECS="avc1.42e00a,mp4a.40.2"
http://example.com/lo_mid/index.m3u8
#EXT-X-STREAM-INF:BANDWIDTH=440000,RESOLUTION=416x234,CODECS="avc1.42e00a,mp4a.40.2"
http://example.com/hi_mid/index.m3u8
#EXT-X-STREAM-INF:BANDWIDTH=640000,RESOLUTION=640x360,CODECS="avc1.42e00a,mp4a.40.2"
http://example.com/high/index.m3u8
#EXT-X-STREAM-INF:BANDWIDTH=64000,CODECS="mp4a.40.5"
http://example.com/audio/index.m3u8

该备用流资源指定了多种不同码率,不同格式的媒体播放列表,并且,该备用流资源也可同时提供不同版本的资源内容,比如不同语言的音频文件,不同角度拍摄的视屏文件等等。客户可以根据不同的网络状态选取合适码流的资源,并且最好根据用户喜好选择合适的资源内容。

以上,就是 m3u8 文件的大概内容

如何解密M3U8文件呢

示例:

M3U8文件是一种播放列表文件,用于存储和组织HLS(HTTP Live Streaming)流媒体数据。在M3U8文件中,EXT-X-KEY、URI和IV等字段是用于描述流媒体的关键信息。

EXT-X-KEY: 这个字段用于指定加密密钥的信息。它通常包含一个URI,该URI指向包含密钥的媒体文件。该字段还可能包含其他参数,如密钥的加密算法和密码等。
URI: 这个字段指定了媒体文件的URL地址。它用于告诉播放器从哪个位置获取媒体数据。
IV: 这个字段是初始化向量(Initialization Vector)的缩写,用于加密算法的初始化过程。在HLS流媒体中,每个媒体片段都使用不同的初始化向量进行加密,以确保每个片段的加密是独立的。
这些字段通常以特定的格式出现在M3U8文件中。下面是一个示例:

#EXTM3U  
#EXT-X-STREAM-INF:PROGRAM-ID=1, BANDWIDTH=1280000  
http://example.com/stream.m3u8?token=1234567890  
#EXT-X-KEY:METHOD=AES-128, URI="http://example.com/key.txt", IV=0x00000000000000000000000000000001  
#EXT-X-KEY:METHOD=AES-128, URI="http://example.com/key2.txt", IV=0x00000001000000010000000100000002  
#EXT-X-I-FRAME-STREAM-INF:PROGRAM-ID=1, BANDWIDTH=512000, CODECS="mp4a.40.2", RESOLUTION=480x360, FRAME-RATE=15  
#EXTINF:16.733333,
http://example.com/iframe.m3u8?token=abcdefghijklmnopqrstuvwxyz

在上述示例中,#EXTM3U标识了文件为M3U8播放列表的开始。#EXT-X-STREAM-INF指定了流媒体的信息,如节目ID和带宽。http://example.com/stream.m3u8?token=1234567890是媒体文件的URI。接下来的#EXT-X-KEY字段指定了加密密钥的信息,包括加密方法和密钥的URI以及初始化向量(IV)。在这个例子中,有两个密钥,每个密钥对应一个媒体片段。最后,#EXT-X-I-FRAME-STREAM-INF指定了I帧媒体流的信息,包括节目ID、带宽、编解码器、分辨率和帧率等。http://example.com/iframe.m3u8?token=abcdefghijklmnopqrstuvwxyz是I帧媒体文件的URI。

实现思路和代码

序列图

在这里插入图片描述

网络请求

	QNetworkRequest request(url);  //请求m3u8地址request.setHeader(QNetworkRequest::ContentTypeHeader, "application/x-www-form-urlencoded");addAllCookie(request);QNetworkReply *pNetworkResponse = pManager->get(request);m_tsContent.clear();m_keyContent.clear();m_ivData.clear();QObject::connect(pNetworkResponse, &QNetworkReply::finished, [=]{if (pNetworkResponse->error() == QNetworkReply::NoError){QByteArray bytes = pNetworkResponse->readAll();QJsonObject json_object = QJsonDocument::fromJson(bytes).object();if(json_object["code"].toInt() == 10000){if(!json_object["data"].isUndefined()){QJsonValue data = json_object["data"];QJsonValue urls = data["videoUrl"];// Test url hereQUrl videoUrl(urls["normal"].toString());QNetworkRequest videoRequest(videoUrl);videoRequest.setHeader(QNetworkRequest::ContentTypeHeader, "application/x-www-form-urlencoded");m_elaptimer.restart();QNetworkReply *pVideoNetworkResponse = pManager->get(videoRequest);connect(pVideoNetworkResponse, &QNetworkReply::finished, [=]{m3u8time = m_elaptimer.elapsed();if(pVideoNetworkResponse->error() == QNetworkReply::NoError) {QByteArray data = pVideoNetworkResponse->readAll();QString m3u8Content = QString::fromUtf8(data);QStringList lines = m3u8Content.split('\n');QString keyLine;QString tsLine;foreach (const QString &line, lines) {if (line.startsWith("#EXT-X-KEY:")) {keyLine = line;qDebug() << "[www]: keyline - " << keyLine;} else if (line.startsWith("https:")) {tsLine = line;qDebug() << "[www]: tsLine - " << tsLine;break;qDebug() << "[www]: test fist snippet and stop";}}QRegularExpression keyRegex("#EXT-X-KEY:METHOD=([A-Za-z0-9-]+),URI=\"([^\"]+)\",IV=([A-Za-z0-9-]+)");QRegularExpressionMatchIterator matchIterator = keyRegex.globalMatch(keyLine);QString method;QString keyUri;QString IVString;if (matchIterator.hasNext()) {QRegularExpressionMatch match = matchIterator.next();method = match.captured(1);keyUri = match.captured(2);IVString = match.captured(3);}qDebug() << "[www] method: " << method;qDebug() << "[www] keyUri: " << keyUri;qDebug() << "[www] tsLine: " << tsLine;qDebug() << "[www] IVString: " << IVString;QByteArray tsData = QByteArray();QByteArray keyData = QByteArray();QByteArray ivData = QByteArray::fromHex(IVString.right(IVString.size() - 2).toLatin1());m_ivData = ivData;qDebug() << "[www] ivData: " << ivData;QString keyContent;QString tsContent;//Get key from uriQUrl keyUrl(keyUri);QNetworkRequest keyQuest(keyUrl);keyQuest.setHeader(QNetworkRequest::ContentTypeHeader, "application/x-www-form-urlencoded");beginKey = m_elaptimer.elapsed();QNetworkReply *keyResponse = pManager->get(keyQuest);connect(keyResponse, SIGNAL(error(QNetworkReply::NetworkError)),this, SLOT(httpError(QNetworkReply::NetworkError)));connect(keyResponse, &QNetworkReply::finished, [=]{if(keyResponse->error() == QNetworkReply::NoError) {QByteArray keyData = keyResponse->readAll();QString keyContent = QString::fromUtf8(keyData);qDebug() << "[www] keyContent:" << keyContent;qDebug() << "[www] keyData:" << keyData.toHex();m_keyContent = keyData;checkAesPara();} else {qDebug() << "[xiaole]" <<  keyResponse->errorString() ;}});//Get ts data from uriQUrl tsUrl(tsLine);QNetworkRequest tsRequest(tsUrl);tsRequest.setHeader(QNetworkRequest::ContentTypeHeader, "application/x-www-form-urlencoded");QNetworkReply *tsResponse = pManager->get(tsRequest);connect(tsResponse, SIGNAL(error(QNetworkReply::NetworkError)),this, SLOT(httpError(QNetworkReply::NetworkError)));connect(tsResponse, &QNetworkReply::finished, [=]{if(tsResponse->error() == QNetworkReply::NoError) {QByteArray tsData = tsResponse->readAll();QString tsContent = QString::fromUtf8(tsData);m_tsContent = tsData;checkAesPara();} });} else {}});}}}pNetworkResponse->close();pNetworkResponse->deleteLater();});

解密

#include "qaesencryption.h"extern "C" {#include <openssl/aes.h>#include <openssl/rand.h>QByteArray aesDecrypt(const QByteArray& cipherText, const QByteArray& key, const QByteArray& iv) {// 创建一个AES解密上下文AES_KEY decryptKey;AES_set_decrypt_key(reinterpret_cast<const unsigned char*>(key.constData()), 128, &decryptKey);// 解密数据QByteArray decryptedText(cipherText.size(), Qt::Uninitialized);AES_cbc_encrypt(reinterpret_cast<const unsigned char*>(cipherText.constData()),reinterpret_cast<unsigned char*>(decryptedText.data()),cipherText.size(),&decryptKey,const_cast<unsigned char*>(reinterpret_cast<const unsigned char*>(iv.data())),AES_DECRYPT);return decryptedText;}
}QByteArray qaesDecrypt(const QByteArray& cipherText, const QByteArray& key, const QByteArray& iv) {return QAESEncryption::Decrypt(QAESEncryption::AES_128, QAESEncryption::CBC, cipherText, key,iv, QAESEncryption::ZERO);
}
checkAesPara()
{if(!m_tsContent.isEmpty() &&  !m_keyContent.isEmpty()){qDebug()<<"keyContent"<< m_keyContent <<"tsContent"<< m_tsContent.length();QByteArray decTsData = aesDecrypt(m_tsContent, m_keyContent, m_ivData);QFile file("/home/test.ts");  //保存到本地文件if (file.open(QIODevice::ReadWrite | QIODevice::Truncate)){file.write(decTsData);file.close();}writeOver  = m_elaptimer.elapsed();writeTime = writeOver-jiemiOver;}
}

结论

`
● ts片段越大,缓冲时间越长,反之亦然
● qt播放器并不需要一个完整的ts下载完才开始播放。

所以视频加载快慢,最大的因素是网速,还有就是跟TS片段大小有关系。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/160133.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

回归预测 | Matlab实现POA-CNN-SVM鹈鹕算法优化卷积神经网络-支持向量机多变量回归预测

Matlab实现POA-CNN-SVM鹈鹕算法优化卷积神经网络-支持向量机多变量回归预测 目录 Matlab实现POA-CNN-SVM鹈鹕算法优化卷积神经网络-支持向量机多变量回归预测效果一览基本介绍程序设计参考资料 效果一览 基本介绍 1.POA-CNN-SVM鹈鹕算法优化卷积神经网络-支持向量机的多变量回归…

Rust语言和curl库编写程序

这是一个使用Rust语言和curl库编写的爬虫程序&#xff0c;用于爬取视频。 use std::env; use std::net::TcpStream; use std::io::{BufReader, BufWriter}; ​ fn main() {// 获取命令行参数let args: Vec<String> env::args().collect();let proxy_host args[1].clon…

[双指针] (三) LeetCode LCR 179. 查找总价格为目标值的两个商品 和 15. 三数之和

[双指针] (三) LeetCode LCR 179. 查找总价格为目标值的两个商品 和 15. 三数之和 文章目录 [双指针] (三) LeetCode LCR 179. 查找总价格为目标值的两个商品 和 15. 三数之和查找总价格为目标值的两个商品题目分析解题思路代码实现总结 三数之和题目分析解题思路代码实现总结 …

陕西某小型水库雨水情测报及大坝安全监测项目案例

项目背景 根据《陕西省小型病险水库除险加固项目管理办法》、《陕西省小型水库雨水情测报和大坝安全监测设施建设与运行管理办法》的要求&#xff0c;为保障水库安全运行&#xff0c;对全省小型病险水库除险加固&#xff0c;建设完善雨水情测报、监测预警、防汛道路、通讯设备、…

android studio 字节码查看工具jclasslib bytecode viewer

jclasslib bytecode viewer 是一款非常好用的.class文件查看工具&#xff1b; jclasslib bytecode editor is a tool that visualizes all aspects of compiled Java class files and the contained bytecode. Many aspects of class files can be edited in the UI. In addit…

YOLOv8在前代的基础上有哪些改进?

YOLO系列又双叒更新&#xff01; 只能说&#xff0c;YOLO系列发展地真快&#xff0c;已经有点跟不上了&#xff01; YOLOv1-YOLOv8系列回顾 YOLOv1&#xff1a;2015年Joseph Redmon和Ali Farhadi等人&#xff08;华盛顿大学&#xff09; YOLOv2&#xff1a;2016年Joseph Re…

Android和JNI交互 : 常见的图像格式转换 : NV21、RGBA、Bitmap等

1. 前言 最近在使用OpenCV处理图片的时候&#xff0c;经常会遇到需要转换图像的情况&#xff0c;网上相关资料比较少&#xff0c;也不全&#xff0c;有时候得费劲老半天才能搞定。 自己踩了坑后&#xff0c;在这里记录下&#xff0c;都是我在项目中遇到的图像转化操作&#xf…

Redo Log(重做日志)的刷盘策略

1. 概述 Redo Log&#xff08;重做日志&#xff09;是 InnoDB 存储引擎中的一种关键组件&#xff0c;用于保障数据库事务的持久性和崩溃恢复。InnoDB 将事务所做的更改先记录到重做日志&#xff0c;之后再将其应用到磁盘上的数据页。 刷盘策略&#xff08;Flush Policy&#x…

CDN与WAF防火墙:强强联手,守护您的网站安全

随着互联网的普及&#xff0c;网站安全问题变得愈发重要。恶意攻击、数据泄露和服务中断等问题都可能给网站和用户带来严重损害。在保护网站免受这些威胁的过程中&#xff0c;内容分发网络&#xff08;CDN&#xff09;和Web应用程序防火墙&#xff08;WAF&#xff09;是两个强大…

代码随想录算法训练营第23期day40|343. 整数拆分、96.不同的二叉搜索树

目录 一、&#xff08;leetcode 343&#xff09;整数拆分 1.动规五部曲 1&#xff09;确定dp数组&#xff08;dp table&#xff09;以及下标的含义 2&#xff09;确定递推公式 3&#xff09;dp的初始化 4&#xff09;确定遍历顺序 5&#xff09;举例推导dp数组 2.贪心算…

PostgreSQL 数据库日志相关参数

PostgreSQL数据库的配置主要是通过修改数据目录下的 postgresql.conf和pg_hba.conf文件来实现的。 如果想从其他机器上登录该数据 库&#xff0c;需要把监听地址改成实际网络的地址&#xff0c;一种简单的方法是把地址 改成“*”&#xff0c;表示在本地的所有地址上监听&#…

Windows搭建Web站点:免费内网穿透发布至公网

目录 什么是cpolar&#xff1f; 概述 1. 注册并安装cpolar内网穿透 2. 搭建一个静态Web站点 2.1 下载演示站点 2.2 本地运行演示站点 2.3 本地浏览测试站点是否正常 3. 本地站点发布公网可访问 3.1 登录cpolar web ui管理界面 3.2 启动website隧道 3.3 获取公网URL地…