Android中两种播放器API对应的两种DRM

2023-11-22 17:10

本文主要是介绍Android中两种播放器API对应的两种DRM,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

之前因为种种原因好久没有更新视频开发的文章了。今天刚刚从国内飞回来,趁着周末更新一下.不过关于DRM这块首先一般的开发者很少用到,而且DRM的开发需要前后台的密切合作,可以说后台的工作占了一大半,安卓前端这块DRM的API封装其实已经很到位了,只是因为接触的人少,所以文档并没有多么而已。所以这篇文章只是给大家过一遍概念,细节我就少讲一些。有具体的问题可以留言,


今天主要开始讲解一下安卓视频开发的 DRM 这个问题,DRM是英文 Digital rights management 的缩写,可以理解为版权保护。众所周知,视频或者音频的盗版问题是困扰发行商的一个大麻烦,因为盗版的横行会直接导致发行商利润的减少。那么像在PC端或者移动端的在线/离线多媒体内容的播放上,发行商又能怎么解决呢?

比如最近优酷很火的《白夜追凶》这种电视剧,


vip的会员可以享受离线观看。假如说这种类型的文件没有进行版权保护,或者说加密,那么会员机制就会轻易作废.(我可以申请一个会员,然后把文件从SD卡中复制粘贴并且发送到网上)

所以一般来说,对这种premiere content(因为我们公司也是做电视剧,老板都这么叫,中文翻译应该可以说是付费内容???),都需要对当前文件,例如MP4文件的audio或者video track部分的内容进行加密,但是metadata部分不加密。只有在用户登录之后,进行身份验证了才传一个密钥用来对该视频进行解密。

那么问题来了

在安卓平台上的视频怎么做DRM的解析?

在回答这个问题之前我们先了解一下当前移动端的DRM的一些概念。

1.DRM platform

DRM 平台可以理解为DRM服务的提供商,它提供了一整套DRM的服务方案,从前端到后端,这里大家可以把这种服务理解为一套 带引号的SDK ,不同的提供商在服务器端和客户端会要求不同的数据传输格式。因平台而异,安卓的设备普遍拥有Widevine这个服务(在framework层),是近几年才被google收购的。


从上图可以看出,现有的成熟的DRM平台并不多,安卓端的话一般使用的是第一个。

2.DRM 是怎么工作的?

简单点来讲,DRM的后台,即服务器端的工作其实和大部分视频内容分发处理后台没有太大的区别,唯一的不同就是它需要对视频数据部分进行适当的加密(在这里我们不讨论加密算法)。

而客户端呢就需要相应的获得解密的秘钥对视频内容解密, 值得注意的是DRM里面秘钥一般被称license而不是key 。整个过程可以用以下的流程图来解释。


3.一般的DRM平台提供商的任务

我们这一部分来详细的了解一下DRM的平台提供商的任务(当然平台提供商并不是必需的,如果企业自己有能力做一整套解决方案那也ok,不过这整篇文章你也不用看了:grin:)。

Widevine 的后台为例,widevine自从被google收购之后就将其后台发展为类似云平台的PASS服务了,企业的后台可以购买Widevine的服务,服务提供加密的API组还有数据库容量,用来保存企业视频的license,即解密秘钥。至于身份验证啊等等功能就留给企业后台自己完成。以下示意图可以大概解释一下Widevine的DRM流程


在第四步之后的流程里面,企业后台会根据播放器的请求去返回一个 向Widevine数据库 请求license 的URL,播放器在获得视频URL和license URL之后就可以开始播放了。

这是最naive的实现。原因是现在的视频服务出现了很多中间商。。。。 :cat:


因为中间涉及很多步骤,所以很多小型的企业决定不要自己去和google的Widevine服务器打交道,而是交给一些视频服务的中间商去做这个事情,然后和这些中间商做身份验证,这样免去了很多麻烦,api相对简单,弊端当然就是要多付钱了。。。


不过无论怎么样,安卓平台上播放DRM视频的宗旨就是获取视频url加上解密用的license。就是这么简单

4.Android的DRM实例代码

说了这么多,终于到安卓的重点了,代码怎么写?????

安卓端的DRM比较蛋疼,因为对于使用MediaCodec API组的和使用原生的MediaPlayer API的开发者来说,两者的代码完全不相同。因为DRM开发者比较少,谷歌的官方文档也不完整,而且很难被理解,我也是研究了很久文档才整明白。

先说MediaCodec API的DRM:

4.1 MediaCodec API组的DRM处理

在这个官方文档里面已经讲的很详细了,如果使用MediaCodec进行decode的时候,configure()方法需要传进一个MediaCrypto


首先我们需要创建一个MediaDrm对象并且调用其 openSession 方法,该方法会返回一个sessionID,标识该次解码工作。

第二步我们需要创建一个MediaCrypto对象给MediaCodec 对象。 它需要一个UUID和initdata,UUID是Widevine的Scheme ID,在Exoplayer的源码中可以看到,在C.java里面。而initData就是上面说到的sessionID.

public static final UUID WIDEVINE_UUID = new UUID(0xEDEF8BA979D64ACEL, 0xA3C827DCD51D21EDL);

最后我们还需要对license server做license的call,得到的reponse就是我们需要的license了,此时只需要调用MediaDrm的provideKeyResponse()方法,视频就可以自动开始播放了。

所以其实总结一下,MediaCodec负责解码,它需要一个MediaCrypto对象,同时需要一个MediaDrm对象,前者获取后者的sessionId让framework去寻找对应的license,后者负责保存从服务器下载下来的license并且提供一个唯一的sessionId给前者。附上伪代码

public static final UUID WIDEVINE_UUID = new UUID(0xEDEF8BA979D64ACEL, 0xA3C827DCD51D21EDL);
//获取sessionId
MediaDrm mediaDrm = new MediaDrm();
String sessionId = mediaDrm.openSession();
//创建MediaCrypto
/**
sessionId 是串联 MediaDrm和MediaCrypto的关键
**/
MediaCrypto ctypto = new MediaCrypto(WIDEVINE_UUID, sessionId)
//用cypto对象来进行解密
MediaCodec codec = new MediaCodec("xxxx")
codec.configure(..,...,ctypto)
/**
注意的是license并不需要在configure之前获取,可以稍后再进行
**///网络连接
byte[] license = HttpUrlConnection.connect().......
mediaDrm.provideKeyResponse(xxx,license);/**
所有工作结束,视频可以正常播放了。
**/

MediaCodec的drm处理文档比较齐全,所以问题不大,具体源码还是又不懂的可以参考ExoPlayer的代码, StreamingDrmSessionManager.java里面整个流程都有。

4.2 MediaPlayer API组的DRM处理

最后一个难点来了,就是原生的播放器MediaPlayer的DRM处理,这部分着实让我苦恼了很久,因为网上资料很少文档极其不齐全。如果你直接谷歌搜索MediaPlayer DRM,是没有任何结果的。。。你也不会知道MediaCodec和MediaPlayer处理DRM有任何不同。

不过功夫不负有心人,费劲千辛万苦之后我终于找到了一丢丢线索。先看官方文档。

developer.android.com/reference/a…

重点在这个地方:


这么重要的事情。。。。谷歌就这么轻描淡写的在这个文档里面随便一提。


也就是说,其实MediaPlayer播放视频的时候,是不需要传任何类似MediaCrypto之类的对象的,直接用DrmManagerClient进行相关操作,framework层会自动处理解密工作了。

下面附上伪代码:

public static final String WV_DRM_SERVER_KEY = "WVDRMServerKey";public static final String WV_ASSET_URI_KEY = "WVAssetURIKey";public static final String WV_DEVICE_ID_KEY = "WVDeviceIDKey";public static final String WV_PORTAL_KEY = "WVPortalKey";/**
调用该方法进行解密,执行成功就ok了
**/public  void acquireKey(){DrmInfoRequest drmInfoRequest = createDrmInfoRequest(assetUrl, infoHolder.getDrmLisenceUrl());DrmInfo drmInfo = mDrmManager.acquireDrmInfo(drmInfoRequest);int rights = mDrmManager.acquireRights(drmInfoRequest);}/**licenseServerUri 就是 对license server进行http通信的Url**/private DrmInfoRequest createDrmInfoRequest(String assetUri, String licenseServerUri) {DrmInfoRequest rightsAcquisitionInfo;rightsAcquisitionInfo = new DrmInfoRequest(DrmInfoRequest.TYPE_RIGHTS_ACQUISITION_INFO,WIDEVINE_MIME_TYPE);if (licenseServerUri != null) {rightsAcquisitionInfo.put(WV_DRM_SERVER_KEY, licenseServerUri);}rightsAcquisitionInfo.put(WV_ASSET_URI_KEY, assetUri);rightsAcquisitionInfo.put(WV_DEVICE_ID_KEY, mDeviceId);rightsAcquisitionInfo.put(WV_PORTAL_KEY, PORTAL_NAME);return rightsAcquisitionInfo;}

所以说MediaPlayer的DRM处理更加简单暴力。。。。当然DrmManagerClient还有其他的一些操作,比如说callback的注册等等。。。

这期的文章就到这,因为Drm涉及到很多细节的处理,还有和后台沟通的问题,这里我只是大概介绍一下Drm的概念和不同类型的api的用法,就不详细展开我工作中具体遇到的麻烦了,有问题可以直接留言或者私信~

刚刚从国内回新就感冒了,这几天昏昏沉沉的。。。。要再点睡觉好好休息了。

这篇关于Android中两种播放器API对应的两种DRM的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/411645

相关文章

MybatisGenerator文件生成不出对应文件的问题

《MybatisGenerator文件生成不出对应文件的问题》本文介绍了使用MybatisGenerator生成文件时遇到的问题及解决方法,主要步骤包括检查目标表是否存在、是否能连接到数据库、配置生成... 目录MyBATisGenerator 文件生成不出对应文件先在项目结构里引入“targetProje

使用SpringBoot创建一个RESTful API的详细步骤

《使用SpringBoot创建一个RESTfulAPI的详细步骤》使用Java的SpringBoot创建RESTfulAPI可以满足多种开发场景,它提供了快速开发、易于配置、可扩展、可维护的优点,尤... 目录一、创建 Spring Boot 项目二、创建控制器类(Controller Class)三、运行

Android数据库Room的实际使用过程总结

《Android数据库Room的实际使用过程总结》这篇文章主要给大家介绍了关于Android数据库Room的实际使用过程,详细介绍了如何创建实体类、数据访问对象(DAO)和数据库抽象类,需要的朋友可以... 目录前言一、Room的基本使用1.项目配置2.创建实体类(Entity)3.创建数据访问对象(DAO

Python读取TIF文件的两种方法实现

《Python读取TIF文件的两种方法实现》本文主要介绍了Python读取TIF文件的两种方法实现,包括使用tifffile库和Pillow库逐帧读取TIFF文件,具有一定的参考价值,感兴趣的可以了解... 目录方法 1:使用 tifffile 逐帧读取安装 tifffile:逐帧读取代码:方法 2:使用

Android WebView的加载超时处理方案

《AndroidWebView的加载超时处理方案》在Android开发中,WebView是一个常用的组件,用于在应用中嵌入网页,然而,当网络状况不佳或页面加载过慢时,用户可能会遇到加载超时的问题,本... 目录引言一、WebView加载超时的原因二、加载超时处理方案1. 使用Handler和Timer进行超

Android实现任意版本设置默认的锁屏壁纸和桌面壁纸(两张壁纸可不一致)

客户有些需求需要设置默认壁纸和锁屏壁纸  在默认情况下 这两个壁纸是相同的  如果需要默认的锁屏壁纸和桌面壁纸不一样 需要额外修改 Android13实现 替换默认桌面壁纸: 将图片文件替换frameworks/base/core/res/res/drawable-nodpi/default_wallpaper.*  (注意不能是bmp格式) 替换默认锁屏壁纸: 将图片资源放入vendo

Android平台播放RTSP流的几种方案探究(VLC VS ExoPlayer VS SmartPlayer)

技术背景 好多开发者需要遴选Android平台RTSP直播播放器的时候,不知道如何选的好,本文针对常用的方案,做个大概的说明: 1. 使用VLC for Android VLC Media Player(VLC多媒体播放器),最初命名为VideoLAN客户端,是VideoLAN品牌产品,是VideoLAN计划的多媒体播放器。它支持众多音频与视频解码器及文件格式,并支持DVD影音光盘,VCD影

android-opencv-jni

//------------------start opencv--------------------@Override public void onResume(){ super.onResume(); //通过OpenCV引擎服务加载并初始化OpenCV类库,所谓OpenCV引擎服务即是 //OpenCV_2.4.3.2_Manager_2.4_*.apk程序包,存

EasyPlayer.js网页H5 Web js播放器能力合集

最近遇到一个需求,要求做一款播放器,发现能力上跟EasyPlayer.js基本一致,满足要求: 需求 功性能 分类 需求描述 功能 预览 分屏模式 单分屏(单屏/全屏) 多分屏(2*2) 多分屏(3*3) 多分屏(4*4) 播放控制 播放(单个或全部) 暂停(暂停时展示最后一帧画面) 停止(单个或全部) 声音控制(开关/音量调节) 主辅码流切换 辅助功能 屏

从状态管理到性能优化:全面解析 Android Compose

文章目录 引言一、Android Compose基本概念1.1 什么是Android Compose?1.2 Compose的优势1.3 如何在项目中使用Compose 二、Compose中的状态管理2.1 状态管理的重要性2.2 Compose中的状态和数据流2.3 使用State和MutableState处理状态2.4 通过ViewModel进行状态管理 三、Compose中的列表和滚动