数字版权保护(ArkTS)
开发者可以调用DRM Kit的ArkTS接口实现DRM证书管理、DRM许可证管理、DRM节目授权、DRM节目解密等数字版权保护功能。
DRM Kit提供MediaKeySystem实现DRM证书管理、DRM许可证管理功能,并管理MediaKeySession实例;MediaKeySession实现DRM节目授权,并可支持Media Kit或Media Kit/AVCodec Kit实现DRM节目解密以实现DRM节目播放。
开发步骤
详细的API说明请参考@ohos.multimedia.drm(数字版权保护)。
-
导入DRM Kit接口。
import { drm } from '@kit.DrmKit'; -
导入BusinessError模块抛出Drm Kit接口的错误码。
import { BusinessError } from '@kit.BasicServicesKit'; -
(可选)调用getMediaKeySystems,获取设备支持的DRM解决方案名称和唯一标识的列表。
let description: drm.MediaKeySystemDescription[] = drm.getMediaKeySystems();如果获取结果数组为空,说明该设备中不存在支持的DRM解决方案。
-
(可选)调用isMediaKeySystemSupported,查询设备是否支持对应DRM解决方案名称、媒体类型、安全保护级别的DRM解决方案。
let isSupported: boolean = drm.isMediaKeySystemSupported("com.wiseplay.drm", "video/mp4", drm.ContentProtectionLevel.CONTENT_PROTECTION_LEVEL_SW_CRYPTO);如果查询结果为false,说明该设备不支持对应的DRM解决方案。
-
调用createMediaKeySystem,创建MediaKeySystem实例。
let mediaKeySystem: drm.MediaKeySystem = drm.createMediaKeySystem("com.wiseplay.drm");如果创建失败则返回undefined,说明该设备不支持该DRM解决方案。
-
(可选)调用on('keySystemRequired'),设置MediaKeySystem状态监听事件。
通过注册keySystemRequired回调函数监听设备DRM证书请求事件。该事件在需要设备DRM证书时触发,此时建议完成设备DRM证书请求与处理流程。
mediaKeySystem.on('keySystemRequired', (eventInfo: drm.EventInfo) => {console.info('keySystemRequired' + 'extra:' + eventInfo.extraInfo + ' data:' + eventInfo.info);// 设备DRM证书请求与处理。}); -
(可选)调用getCertificateStatus,获取设备DRM证书状态。
let certificateStatus: drm.CertificateStatus = mediaKeySystem.getCertificateStatus(); -
(可选)调用generateKeySystemRequest和processKeySystemResponse,生成设备DRM证书请求与处理设备DRM证书响应。
如果设备DRM证书状态不是drm.CertificateStatus.CERT_STATUS_PROVISIONED,可以生成设备DRM证书请求,处理设备DRM证书响应。
if(certificateStatus != drm.CertificateStatus.CERT_STATUS_PROVISIONED) {mediaKeySystem.generateKeySystemRequest().then(async (drmRequest: drm.ProvisionRequest) => {console.info("generateKeySystemRequest success", drmRequest.data, drmRequest.defaultURL);}).catch((err:BusinessError) =>{console.error("generateKeySystemRequest err end", err.code);});} else {console.info("The certificate already exists.");}// 将设备DRM证书请求返回的drmRequest.data通过网络请求发送给DRM证书服务获取设备DRM证书响应,并处理。let provisionResponseByte = new Uint8Array([0x00, 0x00, 0x00, 0x00]); // 设备DRM证书响应。mediaKeySystem.processKeySystemResponse(provisionResponseByte).then(() => {console.info("processKeySystemResponse success");}).catch((err:BusinessError) =>{console.error("processKeySystemResponse err end", err.code);}); -
调用createMediaKeySession,创建MediaKeySession实例。
创建该DRM解决方案默认内容保护级别的MediaKeySession实例。
let mediaKeySession: drm.MediaKeySession = mediaKeySystem.createMediaKeySession(); -
(可选)设置MediaKeySession状态监听事件。
监听MediaKeySession实例的事件,包括媒体密钥请求事件、媒体密钥过期事件、媒体密钥有效期更新事件、媒体密钥变换事件等。
-
使用on('keyRequired')接口监听媒体密钥请求事件,此时建议完成媒体密钥请求与处理流程。
mediaKeySession.on('keyRequired', (eventInfo: drm.EventInfo) => {console.info('keyRequired' + 'info:' + eventInfo.info + ' extraInfo:' + eventInfo.extraInfo);// 媒体密钥请求与处理。}); -
使用on('keyExpired')接口监听媒体密钥过期事件。
mediaKeySession.on('keyExpired', (eventInfo: drm.EventInfo) => {console.info('keyExpired' + 'info:' + eventInfo.info + ' extraInfo:' + eventInfo.extraInfo);}); -
使用on('expirationUpdate')接口监听媒体密钥有效期更新事件。
mediaKeySession.on('expirationUpdate', (eventInfo: drm.EventInfo) => {console.info('expirationUpdate' + 'info:' + eventInfo.info + ' extraInfo:' + eventInfo.extraInfo);}); -
使用on('keysChange')接口监听媒体密钥变换事件。
mediaKeySession.on('keysChange', (keyInfo : drm.KeysInfo[], newKeyAvailable:boolean) => {for(let i = 0; i < keyInfo.length; i++){console.info('keysChange' + 'info:' + keyInfo[i].keyId + ' extraInfo:' + keyInfo[i].value);}});
-
-
(可选)调用requireSecureDecoderModule,查询是否需要安全解码。
try {let status: boolean = mediaKeySession.requireSecureDecoderModule("video/avc");} catch (err) {let error = err as BusinessError;console.error(`requireSecureDecoderModule ERROR: ${error}`);} -
调用generateMediaKeyRequest和processMediaKeyResponse,生成媒体密钥请求与处理媒体密钥响应。
获取到DRM节目中的DRM信息时,可以生成媒体密钥请求,处理媒体密钥响应,以请求许可证完成DRM节目授权。
// 根据DRM解决方案要求,基于DRM信息中的pssh生成initData。let initData = new Uint8Array([0x00, 0x00, 0x00, 0x00]);// 根据DRM解决方案要求设置可选数据的值。let optionalData:drm.OptionsData[] = [{name: "optionalDataName",value: "optionalDataValue"}];// 在线媒体密钥请求和响应。mediaKeySession.generateMediaKeyRequest("video/mp4", initData, drm.MediaKeyType.MEDIA_KEY_TYPE_ONLINE, optionalData).then(async (licenseRequest: drm.MediaKeyRequest) => {console.info("generateMediaKeyRequest success", licenseRequest.mediaKeyRequestType, licenseRequest.data, licenseRequest.defaultURL);// 将媒体密钥请求返回的licenseRequest.data通过网络请求发送给DRM服务获取媒体密钥响应,并处理。let licenseResponse = new Uint8Array([0x00, 0x00, 0x00, 0x00]); // 媒体密钥响应。mediaKeySession.processMediaKeyResponse(licenseResponse).then((mediaKeyId: Uint8Array) => {console.info("processMediaKeyResponse success");}).catch((err:BusinessError) =>{console.error("processMediaKeyResponse err end", err.code);});}).catch((err:BusinessError) =>{console.error("generateMediaKeyRequest err end", err.code);});// 离线媒体密钥请求和响应。let offlineMediaKeyId: Uint8Array;mediaKeySession.generateMediaKeyRequest("video/mp4", initData, drm.MediaKeyType.MEDIA_KEY_TYPE_OFFLINE, optionalData).then((licenseRequest: drm.MediaKeyRequest) => {console.info("generateMediaKeyRequest success", licenseRequest.mediaKeyRequestType, licenseRequest.data, licenseRequest.defaultURL);// 将媒体密钥请求返回的licenseRequest.data通过网络请求发送给DRM服务获取媒体密钥响应,并处理。let licenseResponse = new Uint8Array([0x00, 0x00, 0x00, 0x00]); // 媒体密钥响应。mediaKeySession.processMediaKeyResponse(licenseResponse).then((mediaKeyId: Uint8Array) => {offlineMediaKeyId = new Uint8Array(mediaKeyId);console.info("processMediaKeyResponse success");}).catch((err:BusinessError) =>{console.error("processMediaKeyResponse err end", err.code);});}).catch((err:BusinessError) =>{console.error("generateMediaKeyRequest err end", err.code);}); -
(可选)调用restoreOfflineMediaKey,恢复离线媒体密钥。
mediaKeySession.restoreOfflineMediaKeys(offlineMediaKeyId).then(() => {console.info("restoreOfflineMediaKeys success.");}).catch((err: BusinessError) => {console.error(`restoreOfflineMediaKeys: ERROR: ${err}`);}); -
(可选)调用checkMediaKeyStatus,检查媒体密钥状态。
let mediaKeyStatus: drm.MediaKeyStatus[]try {mediaKeyStatus = mediaKeySession.checkMediaKeyStatus()} catch (err) {let error = err as BusinessError;console.error(`checkMediaKeyStatus: ERROR: ${error}`);} -
(可选)调用getOfflineMediaKeyIds获取离线媒体密钥标识列表,调用getOfflineMediaKeyStatus获取离线媒体密钥状态,调用clearOfflineMediaKeys删除离线媒体密钥。
媒体密钥标识用于对离线媒体密钥的管理。
let offlineMediaKeyIds: Uint8Array[] = mediaKeySystem.getOfflineMediaKeyIds();try {let offlineMediaKeyStatus: drm.OfflineMediaKeyStatus = mediaKeySystem.getOfflineMediaKeyStatus(offlineMediaKeyIds[0]);} catch (err) {let error = err as BusinessError;console.error(`getOfflineMediaKeyStatus ERROR: ${error}`);}try {mediaKeySystem.clearOfflineMediaKeys(offlineMediaKeyIds[0]);} catch (err) {let error = err as BusinessError;console.error(`clearOfflineMediaKeys ERROR: ${error}`);} -
销毁MediaKeySession实例。
完成加密媒体解密,MediaKeySession实例不再使用时,销毁MediaKeySession实例。
// MediaKeySession实例使用完需要进行资源释放。mediaKeySession.destroy(); -
销毁MediaKeySystem实例。
完成DRM功能使用,MediaKeySystem实例不再使用,销毁MediaKeySystem实例。
// MediaKeySystem实例使用完需要进行资源释放。mediaKeySystem.destroy();