Source: lib/polyfill/patchedmediakeys_apple.js

  1. /*! @license
  2. * Shaka Player
  3. * Copyright 2016 Google LLC
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. goog.provide('shaka.polyfill.PatchedMediaKeysApple');
  7. goog.require('goog.asserts');
  8. goog.require('shaka.log');
  9. goog.require('shaka.polyfill');
  10. goog.require('shaka.util.BufferUtils');
  11. goog.require('shaka.util.DrmUtils');
  12. goog.require('shaka.util.EventManager');
  13. goog.require('shaka.util.FakeEvent');
  14. goog.require('shaka.util.FakeEventTarget');
  15. goog.require('shaka.util.MediaReadyState');
  16. goog.require('shaka.util.PublicPromise');
  17. goog.require('shaka.util.StringUtils');
  18. /**
  19. * @summary A polyfill to implement modern, standardized EME on top of Apple's
  20. * prefixed EME in Safari.
  21. * @export
  22. */
  23. shaka.polyfill.PatchedMediaKeysApple = class {
  24. /**
  25. * Installs the polyfill if needed.
  26. */
  27. static defaultInstall() {
  28. if (!window.HTMLVideoElement || !window.WebKitMediaKeys) {
  29. // No HTML5 video or no prefixed EME.
  30. return;
  31. }
  32. if (navigator.requestMediaKeySystemAccess &&
  33. // eslint-disable-next-line no-restricted-syntax
  34. MediaKeySystemAccess.prototype.getConfiguration) {
  35. // Unprefixed EME available
  36. return;
  37. }
  38. // If there is no unprefixed EME and prefixed EME exists, apply installation
  39. // by default. Eg: older versions of Safari.
  40. shaka.polyfill.PatchedMediaKeysApple.install();
  41. }
  42. /**
  43. * Installs the polyfill if needed.
  44. * @param {boolean=} enableUninstall enables uninstalling the polyfill
  45. * @export
  46. */
  47. static install(enableUninstall = false) {
  48. // Alias
  49. const PatchedMediaKeysApple = shaka.polyfill.PatchedMediaKeysApple;
  50. if (!window.HTMLVideoElement || !window.WebKitMediaKeys) {
  51. // No HTML5 video or no prefixed EME.
  52. return;
  53. }
  54. if (enableUninstall) {
  55. PatchedMediaKeysApple.enableUninstall = true;
  56. PatchedMediaKeysApple.originalHTMLMediaElementPrototypeMediaKeys =
  57. /** @type {!Object} */ (
  58. Object.getOwnPropertyDescriptor(
  59. // eslint-disable-next-line no-restricted-syntax
  60. HTMLMediaElement.prototype, 'mediaKeys',
  61. )
  62. );
  63. PatchedMediaKeysApple.originalHTMLMediaElementPrototypeSetMediaKeys =
  64. // eslint-disable-next-line no-restricted-syntax
  65. HTMLMediaElement.prototype.setMediaKeys;
  66. PatchedMediaKeysApple.originalWindowMediaKeys = window.MediaKeys;
  67. PatchedMediaKeysApple.originalWindowMediaKeySystemAccess =
  68. window.MediaKeySystemAccess;
  69. PatchedMediaKeysApple.originalNavigatorRequestMediaKeySystemAccess =
  70. navigator.requestMediaKeySystemAccess;
  71. }
  72. shaka.log.info('Using Apple-prefixed EME');
  73. // Delete mediaKeys to work around strict mode compatibility issues.
  74. // eslint-disable-next-line no-restricted-syntax
  75. delete HTMLMediaElement.prototype['mediaKeys'];
  76. // Work around read-only declaration for mediaKeys by using a string.
  77. // eslint-disable-next-line no-restricted-syntax
  78. HTMLMediaElement.prototype['mediaKeys'] = null;
  79. // eslint-disable-next-line no-restricted-syntax
  80. HTMLMediaElement.prototype.setMediaKeys =
  81. PatchedMediaKeysApple.setMediaKeys;
  82. // Install patches
  83. window.MediaKeys = PatchedMediaKeysApple.MediaKeys;
  84. window.MediaKeySystemAccess = PatchedMediaKeysApple.MediaKeySystemAccess;
  85. navigator.requestMediaKeySystemAccess =
  86. PatchedMediaKeysApple.requestMediaKeySystemAccess;
  87. window.shakaMediaKeysPolyfill = PatchedMediaKeysApple.apiName_;
  88. }
  89. /**
  90. * Uninstalls the polyfill if needed and enabled.
  91. * @export
  92. */
  93. static uninstall() {
  94. // Alias
  95. const PatchedMediaKeysApple = shaka.polyfill.PatchedMediaKeysApple;
  96. if (!PatchedMediaKeysApple.enableUninstall) {
  97. return;
  98. }
  99. shaka.log.info('Un-installing Apple-prefixed EME');
  100. PatchedMediaKeysApple.enableUninstall = false;
  101. Object.defineProperty(
  102. // eslint-disable-next-line no-restricted-syntax
  103. HTMLMediaElement.prototype,
  104. 'mediaKeys',
  105. PatchedMediaKeysApple.originalHTMLMediaElementPrototypeMediaKeys,
  106. );
  107. // eslint-disable-next-line no-restricted-syntax
  108. HTMLMediaElement.prototype.setMediaKeys =
  109. PatchedMediaKeysApple.originalHTMLMediaElementPrototypeSetMediaKeys;
  110. window.MediaKeys = PatchedMediaKeysApple.originalWindowMediaKeys;
  111. window.MediaKeySystemAccess =
  112. PatchedMediaKeysApple.originalWindowMediaKeySystemAccess;
  113. navigator.requestMediaKeySystemAccess =
  114. PatchedMediaKeysApple.originalNavigatorRequestMediaKeySystemAccess;
  115. PatchedMediaKeysApple.originalWindowMediaKeys = null;
  116. PatchedMediaKeysApple.originalWindowMediaKeySystemAccess = null;
  117. PatchedMediaKeysApple.originalHTMLMediaElementPrototypeSetMediaKeys = null;
  118. PatchedMediaKeysApple.originalNavigatorRequestMediaKeySystemAccess = null;
  119. PatchedMediaKeysApple.originalHTMLMediaElementPrototypeMediaKeys = null;
  120. window.shakaMediaKeysPolyfill = '';
  121. }
  122. /**
  123. * An implementation of navigator.requestMediaKeySystemAccess.
  124. * Retrieves a MediaKeySystemAccess object.
  125. *
  126. * @this {!Navigator}
  127. * @param {string} keySystem
  128. * @param {!Array.<!MediaKeySystemConfiguration>} supportedConfigurations
  129. * @return {!Promise.<!MediaKeySystemAccess>}
  130. */
  131. static requestMediaKeySystemAccess(keySystem, supportedConfigurations) {
  132. shaka.log.debug('PatchedMediaKeysApple.requestMediaKeySystemAccess');
  133. goog.asserts.assert(this == navigator,
  134. 'bad "this" for requestMediaKeySystemAccess');
  135. // Alias.
  136. const PatchedMediaKeysApple = shaka.polyfill.PatchedMediaKeysApple;
  137. try {
  138. const access = new PatchedMediaKeysApple.MediaKeySystemAccess(
  139. keySystem, supportedConfigurations);
  140. return Promise.resolve(/** @type {!MediaKeySystemAccess} */ (access));
  141. } catch (exception) {
  142. return Promise.reject(exception);
  143. }
  144. }
  145. /**
  146. * An implementation of HTMLMediaElement.prototype.setMediaKeys.
  147. * Attaches a MediaKeys object to the media element.
  148. *
  149. * @this {!HTMLMediaElement}
  150. * @param {MediaKeys} mediaKeys
  151. * @return {!Promise}
  152. */
  153. static setMediaKeys(mediaKeys) {
  154. shaka.log.debug('PatchedMediaKeysApple.setMediaKeys');
  155. goog.asserts.assert(this instanceof HTMLMediaElement,
  156. 'bad "this" for setMediaKeys');
  157. // Alias
  158. const PatchedMediaKeysApple = shaka.polyfill.PatchedMediaKeysApple;
  159. const newMediaKeys =
  160. /** @type {shaka.polyfill.PatchedMediaKeysApple.MediaKeys} */ (
  161. mediaKeys);
  162. const oldMediaKeys =
  163. /** @type {shaka.polyfill.PatchedMediaKeysApple.MediaKeys} */ (
  164. this.mediaKeys);
  165. if (oldMediaKeys && oldMediaKeys != newMediaKeys) {
  166. goog.asserts.assert(
  167. oldMediaKeys instanceof PatchedMediaKeysApple.MediaKeys,
  168. 'non-polyfill instance of oldMediaKeys');
  169. // Have the old MediaKeys stop listening to events on the video tag.
  170. oldMediaKeys.setMedia(null);
  171. }
  172. delete this['mediaKeys']; // in case there is an existing getter
  173. this['mediaKeys'] = mediaKeys; // work around read-only declaration
  174. if (newMediaKeys) {
  175. goog.asserts.assert(
  176. newMediaKeys instanceof PatchedMediaKeysApple.MediaKeys,
  177. 'non-polyfill instance of newMediaKeys');
  178. return newMediaKeys.setMedia(this);
  179. }
  180. return Promise.resolve();
  181. }
  182. /**
  183. * Handler for the native media elements webkitneedkey event.
  184. *
  185. * @this {!HTMLMediaElement}
  186. * @param {!MediaKeyEvent} event
  187. * @suppress {constantProperty} We reassign what would be const on a real
  188. * MediaEncryptedEvent, but in our look-alike event.
  189. * @private
  190. */
  191. static onWebkitNeedKey_(event) {
  192. shaka.log.debug('PatchedMediaKeysApple.onWebkitNeedKey_', event);
  193. const PatchedMediaKeysApple = shaka.polyfill.PatchedMediaKeysApple;
  194. const mediaKeys =
  195. /** @type {shaka.polyfill.PatchedMediaKeysApple.MediaKeys} */(
  196. this.mediaKeys);
  197. goog.asserts.assert(mediaKeys instanceof PatchedMediaKeysApple.MediaKeys,
  198. 'non-polyfill instance of newMediaKeys');
  199. goog.asserts.assert(event.initData != null, 'missing init data!');
  200. // Convert the prefixed init data to match the native 'encrypted' event.
  201. const uint8 = shaka.util.BufferUtils.toUint8(event.initData);
  202. const dataview = shaka.util.BufferUtils.toDataView(uint8);
  203. // The first part is a 4 byte little-endian int, which is the length of
  204. // the second part.
  205. const length = dataview.getUint32(
  206. /* position= */ 0, /* littleEndian= */ true);
  207. if (length + 4 != uint8.byteLength) {
  208. throw new RangeError('Malformed FairPlay init data');
  209. }
  210. // The remainder is a UTF-16 skd URL. Convert this to UTF-8 and pass on.
  211. const str = shaka.util.StringUtils.fromUTF16(
  212. uint8.subarray(4), /* littleEndian= */ true);
  213. const initData = shaka.util.StringUtils.toUTF8(str);
  214. // NOTE: Because "this" is a real EventTarget, the event we dispatch here
  215. // must also be a real Event.
  216. const event2 = new Event('encrypted');
  217. const encryptedEvent =
  218. /** @type {!MediaEncryptedEvent} */(/** @type {?} */(event2));
  219. encryptedEvent.initDataType = 'skd';
  220. encryptedEvent.initData = shaka.util.BufferUtils.toArrayBuffer(initData);
  221. this.dispatchEvent(event2);
  222. }
  223. };
  224. /**
  225. * An implementation of MediaKeySystemAccess.
  226. *
  227. * @implements {MediaKeySystemAccess}
  228. */
  229. shaka.polyfill.PatchedMediaKeysApple.MediaKeySystemAccess = class {
  230. /**
  231. * @param {string} keySystem
  232. * @param {!Array.<!MediaKeySystemConfiguration>} supportedConfigurations
  233. */
  234. constructor(keySystem, supportedConfigurations) {
  235. shaka.log.debug('PatchedMediaKeysApple.MediaKeySystemAccess');
  236. /** @type {string} */
  237. this.keySystem = keySystem;
  238. /** @private {!MediaKeySystemConfiguration} */
  239. this.configuration_;
  240. // Optimization: WebKitMediaKeys.isTypeSupported delays responses by a
  241. // significant amount of time, possibly to discourage fingerprinting.
  242. // Since we know only FairPlay is supported here, let's skip queries for
  243. // anything else to speed up the process.
  244. if (keySystem.startsWith('com.apple.fps')) {
  245. for (const cfg of supportedConfigurations) {
  246. const newCfg = this.checkConfig_(cfg);
  247. if (newCfg) {
  248. this.configuration_ = newCfg;
  249. return;
  250. }
  251. }
  252. }
  253. // According to the spec, this should be a DOMException, but there is not a
  254. // public constructor for that. So we make this look-alike instead.
  255. const unsupportedKeySystemError = new Error('Unsupported keySystem');
  256. unsupportedKeySystemError.name = 'NotSupportedError';
  257. unsupportedKeySystemError['code'] = DOMException.NOT_SUPPORTED_ERR;
  258. throw unsupportedKeySystemError;
  259. }
  260. /**
  261. * Check a single config for MediaKeySystemAccess.
  262. *
  263. * @param {MediaKeySystemConfiguration} cfg The requested config.
  264. * @return {?MediaKeySystemConfiguration} A matching config we can support, or
  265. * null if the input is not supportable.
  266. * @private
  267. */
  268. checkConfig_(cfg) {
  269. if (cfg.persistentState == 'required') {
  270. // Not supported by the prefixed API.
  271. return null;
  272. }
  273. // Create a new config object and start adding in the pieces which we find
  274. // support for. We will return this from getConfiguration() later if
  275. // asked.
  276. /** @type {!MediaKeySystemConfiguration} */
  277. const newCfg = {
  278. 'audioCapabilities': [],
  279. 'videoCapabilities': [],
  280. // It is technically against spec to return these as optional, but we
  281. // don't truly know their values from the prefixed API:
  282. 'persistentState': 'optional',
  283. 'distinctiveIdentifier': 'optional',
  284. // Pretend the requested init data types are supported, since we don't
  285. // really know that either:
  286. 'initDataTypes': cfg.initDataTypes,
  287. 'sessionTypes': ['temporary'],
  288. 'label': cfg.label,
  289. };
  290. // PatchedMediaKeysApple tests for key system availability through
  291. // WebKitMediaKeys.isTypeSupported.
  292. let ranAnyTests = false;
  293. let success = false;
  294. if (cfg.audioCapabilities) {
  295. for (const cap of cfg.audioCapabilities) {
  296. if (cap.contentType) {
  297. ranAnyTests = true;
  298. const contentType = cap.contentType.split(';')[0];
  299. if (WebKitMediaKeys.isTypeSupported(this.keySystem, contentType)) {
  300. newCfg.audioCapabilities.push(cap);
  301. success = true;
  302. }
  303. }
  304. }
  305. }
  306. if (cfg.videoCapabilities) {
  307. for (const cap of cfg.videoCapabilities) {
  308. if (cap.contentType) {
  309. ranAnyTests = true;
  310. const contentType = cap.contentType.split(';')[0];
  311. if (WebKitMediaKeys.isTypeSupported(this.keySystem, contentType)) {
  312. newCfg.videoCapabilities.push(cap);
  313. success = true;
  314. }
  315. }
  316. }
  317. }
  318. if (!ranAnyTests) {
  319. // If no specific types were requested, we check all common types to
  320. // find out if the key system is present at all.
  321. success = WebKitMediaKeys.isTypeSupported(this.keySystem, 'video/mp4');
  322. }
  323. if (success) {
  324. return newCfg;
  325. }
  326. return null;
  327. }
  328. /** @override */
  329. createMediaKeys() {
  330. shaka.log.debug(
  331. 'PatchedMediaKeysApple.MediaKeySystemAccess.createMediaKeys');
  332. // Alias
  333. const PatchedMediaKeysApple = shaka.polyfill.PatchedMediaKeysApple;
  334. const mediaKeys = new PatchedMediaKeysApple.MediaKeys(this.keySystem);
  335. return Promise.resolve(/** @type {!MediaKeys} */ (mediaKeys));
  336. }
  337. /** @override */
  338. getConfiguration() {
  339. shaka.log.debug(
  340. 'PatchedMediaKeysApple.MediaKeySystemAccess.getConfiguration');
  341. return this.configuration_;
  342. }
  343. };
  344. /**
  345. * An implementation of MediaKeys.
  346. *
  347. * @implements {MediaKeys}
  348. */
  349. shaka.polyfill.PatchedMediaKeysApple.MediaKeys = class {
  350. /** @param {string} keySystem */
  351. constructor(keySystem) {
  352. shaka.log.debug('PatchedMediaKeysApple.MediaKeys');
  353. /** @private {!WebKitMediaKeys} */
  354. this.nativeMediaKeys_ = new WebKitMediaKeys(keySystem);
  355. /** @private {!shaka.util.EventManager} */
  356. this.eventManager_ = new shaka.util.EventManager();
  357. }
  358. /** @override */
  359. createSession(sessionType) {
  360. shaka.log.debug('PatchedMediaKeysApple.MediaKeys.createSession');
  361. sessionType = sessionType || 'temporary';
  362. // For now, only the 'temporary' type is supported.
  363. if (sessionType != 'temporary') {
  364. throw new TypeError('Session type ' + sessionType +
  365. ' is unsupported on this platform.');
  366. }
  367. // Alias
  368. const PatchedMediaKeysApple = shaka.polyfill.PatchedMediaKeysApple;
  369. return new PatchedMediaKeysApple.MediaKeySession(
  370. this.nativeMediaKeys_, sessionType);
  371. }
  372. /** @override */
  373. setServerCertificate(serverCertificate) {
  374. shaka.log.debug('PatchedMediaKeysApple.MediaKeys.setServerCertificate');
  375. return Promise.resolve(false);
  376. }
  377. /**
  378. * @param {HTMLMediaElement} media
  379. * @protected
  380. * @return {!Promise}
  381. */
  382. setMedia(media) {
  383. // Alias
  384. const PatchedMediaKeysApple = shaka.polyfill.PatchedMediaKeysApple;
  385. // Remove any old listeners.
  386. this.eventManager_.removeAll();
  387. // It is valid for media to be null; null is used to flag that event
  388. // handlers need to be cleaned up.
  389. if (!media) {
  390. return Promise.resolve();
  391. }
  392. // Intercept and translate these prefixed EME events.
  393. this.eventManager_.listen(media, 'webkitneedkey',
  394. /** @type {shaka.util.EventManager.ListenerType} */
  395. (PatchedMediaKeysApple.onWebkitNeedKey_));
  396. // Wrap native HTMLMediaElement.webkitSetMediaKeys with a Promise.
  397. try {
  398. // Some browsers require that readyState >=1 before mediaKeys can be
  399. // set, so check this and wait for loadedmetadata if we are not in the
  400. // correct state
  401. shaka.util.MediaReadyState.waitForReadyState(media,
  402. HTMLMediaElement.HAVE_METADATA,
  403. this.eventManager_, () => {
  404. media.webkitSetMediaKeys(this.nativeMediaKeys_);
  405. });
  406. return Promise.resolve();
  407. } catch (exception) {
  408. return Promise.reject(exception);
  409. }
  410. }
  411. /** @override */
  412. getStatusForPolicy(policy) {
  413. return Promise.resolve('usable');
  414. }
  415. };
  416. /**
  417. * An implementation of MediaKeySession.
  418. *
  419. * @implements {MediaKeySession}
  420. */
  421. shaka.polyfill.PatchedMediaKeysApple.MediaKeySession =
  422. class extends shaka.util.FakeEventTarget {
  423. /**
  424. * @param {WebKitMediaKeys} nativeMediaKeys
  425. * @param {string} sessionType
  426. */
  427. constructor(nativeMediaKeys, sessionType) {
  428. shaka.log.debug('PatchedMediaKeysApple.MediaKeySession');
  429. super();
  430. /**
  431. * The native MediaKeySession, which will be created in generateRequest.
  432. * @private {WebKitMediaKeySession}
  433. */
  434. this.nativeMediaKeySession_ = null;
  435. /** @private {WebKitMediaKeys} */
  436. this.nativeMediaKeys_ = nativeMediaKeys;
  437. // Promises that are resolved later
  438. /** @private {shaka.util.PublicPromise} */
  439. this.generateRequestPromise_ = null;
  440. /** @private {shaka.util.PublicPromise} */
  441. this.updatePromise_ = null;
  442. /** @private {!shaka.util.EventManager} */
  443. this.eventManager_ = new shaka.util.EventManager();
  444. /** @type {string} */
  445. this.sessionId = '';
  446. /** @type {number} */
  447. this.expiration = NaN;
  448. /** @type {!shaka.util.PublicPromise} */
  449. this.closed = new shaka.util.PublicPromise();
  450. /** @type {!shaka.polyfill.PatchedMediaKeysApple.MediaKeyStatusMap} */
  451. this.keyStatuses =
  452. new shaka.polyfill.PatchedMediaKeysApple.MediaKeyStatusMap();
  453. }
  454. /** @override */
  455. generateRequest(initDataType, initData) {
  456. shaka.log.debug(
  457. 'PatchedMediaKeysApple.MediaKeySession.generateRequest');
  458. this.generateRequestPromise_ = new shaka.util.PublicPromise();
  459. try {
  460. // This EME spec version requires a MIME content type as the 1st param to
  461. // createSession, but doesn't seem to matter what the value is.
  462. // It also only accepts Uint8Array, not ArrayBuffer, so explicitly make
  463. // initData into a Uint8Array.
  464. const session = this.nativeMediaKeys_.createSession(
  465. 'video/mp4', shaka.util.BufferUtils.toUint8(initData));
  466. this.nativeMediaKeySession_ = session;
  467. this.sessionId = session.sessionId || '';
  468. // Attach session event handlers here.
  469. this.eventManager_.listen(
  470. this.nativeMediaKeySession_, 'webkitkeymessage',
  471. /** @type {shaka.util.EventManager.ListenerType} */
  472. ((event) => this.onWebkitKeyMessage_(event)));
  473. this.eventManager_.listen(session, 'webkitkeyadded',
  474. /** @type {shaka.util.EventManager.ListenerType} */
  475. ((event) => this.onWebkitKeyAdded_(event)));
  476. this.eventManager_.listen(session, 'webkitkeyerror',
  477. /** @type {shaka.util.EventManager.ListenerType} */
  478. ((event) => this.onWebkitKeyError_(event)));
  479. this.updateKeyStatus_('status-pending');
  480. } catch (exception) {
  481. this.generateRequestPromise_.reject(exception);
  482. }
  483. return this.generateRequestPromise_;
  484. }
  485. /** @override */
  486. load() {
  487. shaka.log.debug('PatchedMediaKeysApple.MediaKeySession.load');
  488. return Promise.reject(new Error('MediaKeySession.load not yet supported'));
  489. }
  490. /** @override */
  491. update(response) {
  492. shaka.log.debug('PatchedMediaKeysApple.MediaKeySession.update');
  493. this.updatePromise_ = new shaka.util.PublicPromise();
  494. try {
  495. // Pass through to the native session.
  496. this.nativeMediaKeySession_.update(
  497. shaka.util.BufferUtils.toUint8(response));
  498. } catch (exception) {
  499. this.updatePromise_.reject(exception);
  500. }
  501. return this.updatePromise_;
  502. }
  503. /** @override */
  504. close() {
  505. shaka.log.debug('PatchedMediaKeysApple.MediaKeySession.close');
  506. try {
  507. // Pass through to the native session.
  508. this.nativeMediaKeySession_.close();
  509. this.closed.resolve();
  510. this.eventManager_.removeAll();
  511. } catch (exception) {
  512. this.closed.reject(exception);
  513. }
  514. return this.closed;
  515. }
  516. /** @override */
  517. remove() {
  518. shaka.log.debug('PatchedMediaKeysApple.MediaKeySession.remove');
  519. return Promise.reject(new Error(
  520. 'MediaKeySession.remove is only applicable for persistent licenses, ' +
  521. 'which are not supported on this platform'));
  522. }
  523. /**
  524. * Handler for the native keymessage event on WebKitMediaKeySession.
  525. *
  526. * @param {!MediaKeyEvent} event
  527. * @private
  528. */
  529. onWebkitKeyMessage_(event) {
  530. shaka.log.debug('PatchedMediaKeysApple.onWebkitKeyMessage_', event);
  531. // We can now resolve this.generateRequestPromise, which should be non-null.
  532. goog.asserts.assert(this.generateRequestPromise_,
  533. 'generateRequestPromise_ should be set before now!');
  534. if (this.generateRequestPromise_) {
  535. this.generateRequestPromise_.resolve();
  536. this.generateRequestPromise_ = null;
  537. }
  538. const isNew = this.keyStatuses.getStatus() == undefined;
  539. const data = new Map()
  540. .set('messageType', isNew ? 'license-request' : 'license-renewal')
  541. .set('message', shaka.util.BufferUtils.toArrayBuffer(event.message));
  542. const event2 = new shaka.util.FakeEvent('message', data);
  543. this.dispatchEvent(event2);
  544. }
  545. /**
  546. * Handler for the native keyadded event on WebKitMediaKeySession.
  547. *
  548. * @param {!MediaKeyEvent} event
  549. * @private
  550. */
  551. onWebkitKeyAdded_(event) {
  552. shaka.log.debug('PatchedMediaKeysApple.onWebkitKeyAdded_', event);
  553. // This shouldn't fire while we're in the middle of generateRequest,
  554. // but if it does, we will need to change the logic to account for it.
  555. goog.asserts.assert(!this.generateRequestPromise_,
  556. 'Key added during generate!');
  557. // We can now resolve this.updatePromise, which should be non-null.
  558. goog.asserts.assert(this.updatePromise_,
  559. 'updatePromise_ should be set before now!');
  560. if (this.updatePromise_) {
  561. this.updateKeyStatus_('usable');
  562. this.updatePromise_.resolve();
  563. this.updatePromise_ = null;
  564. }
  565. }
  566. /**
  567. * Handler for the native keyerror event on WebKitMediaKeySession.
  568. *
  569. * @param {!MediaKeyEvent} event
  570. * @private
  571. */
  572. onWebkitKeyError_(event) {
  573. shaka.log.debug('PatchedMediaKeysApple.onWebkitKeyError_', event);
  574. const error = new Error('EME PatchedMediaKeysApple key error');
  575. error['errorCode'] = this.nativeMediaKeySession_.error;
  576. if (this.generateRequestPromise_ != null) {
  577. this.generateRequestPromise_.reject(error);
  578. this.generateRequestPromise_ = null;
  579. } else if (this.updatePromise_ != null) {
  580. this.updatePromise_.reject(error);
  581. this.updatePromise_ = null;
  582. } else {
  583. // Unexpected error - map native codes to standardised key statuses.
  584. // Possible values of this.nativeMediaKeySession_.error.code:
  585. // MEDIA_KEYERR_UNKNOWN = 1
  586. // MEDIA_KEYERR_CLIENT = 2
  587. // MEDIA_KEYERR_SERVICE = 3
  588. // MEDIA_KEYERR_OUTPUT = 4
  589. // MEDIA_KEYERR_HARDWARECHANGE = 5
  590. // MEDIA_KEYERR_DOMAIN = 6
  591. switch (this.nativeMediaKeySession_.error.code) {
  592. case WebKitMediaKeyError.MEDIA_KEYERR_OUTPUT:
  593. case WebKitMediaKeyError.MEDIA_KEYERR_HARDWARECHANGE:
  594. this.updateKeyStatus_('output-not-allowed');
  595. break;
  596. default:
  597. this.updateKeyStatus_('internal-error');
  598. break;
  599. }
  600. }
  601. }
  602. /**
  603. * Updates key status and dispatch a 'keystatuseschange' event.
  604. *
  605. * @param {string} status
  606. * @private
  607. */
  608. updateKeyStatus_(status) {
  609. this.keyStatuses.setStatus(status);
  610. const event = new shaka.util.FakeEvent('keystatuseschange');
  611. this.dispatchEvent(event);
  612. }
  613. };
  614. /**
  615. * @summary An implementation of MediaKeyStatusMap.
  616. * This fakes a map with a single key ID.
  617. *
  618. * @todo Consolidate the MediaKeyStatusMap types in these polyfills.
  619. * @implements {MediaKeyStatusMap}
  620. */
  621. shaka.polyfill.PatchedMediaKeysApple.MediaKeyStatusMap = class {
  622. /** */
  623. constructor() {
  624. /**
  625. * @type {number}
  626. */
  627. this.size = 0;
  628. /**
  629. * @private {string|undefined}
  630. */
  631. this.status_ = undefined;
  632. }
  633. /**
  634. * An internal method used by the session to set key status.
  635. * @param {string|undefined} status
  636. */
  637. setStatus(status) {
  638. this.size = status == undefined ? 0 : 1;
  639. this.status_ = status;
  640. }
  641. /**
  642. * An internal method used by the session to get key status.
  643. * @return {string|undefined}
  644. */
  645. getStatus() {
  646. return this.status_;
  647. }
  648. /** @override */
  649. forEach(fn) {
  650. if (this.status_) {
  651. fn(this.status_, shaka.util.DrmUtils.DUMMY_KEY_ID.value());
  652. }
  653. }
  654. /** @override */
  655. get(keyId) {
  656. if (this.has(keyId)) {
  657. return this.status_;
  658. }
  659. return undefined;
  660. }
  661. /** @override */
  662. has(keyId) {
  663. const fakeKeyId = shaka.util.DrmUtils.DUMMY_KEY_ID.value();
  664. if (this.status_ && shaka.util.BufferUtils.equal(keyId, fakeKeyId)) {
  665. return true;
  666. }
  667. return false;
  668. }
  669. /**
  670. * @suppress {missingReturn}
  671. * @override
  672. */
  673. entries() {
  674. goog.asserts.assert(false, 'Not used! Provided only for the compiler.');
  675. }
  676. /**
  677. * @suppress {missingReturn}
  678. * @override
  679. */
  680. keys() {
  681. goog.asserts.assert(false, 'Not used! Provided only for the compiler.');
  682. }
  683. /**
  684. * @suppress {missingReturn}
  685. * @override
  686. */
  687. values() {
  688. goog.asserts.assert(false, 'Not used! Provided only for the compiler.');
  689. }
  690. };
  691. /**
  692. * API name.
  693. *
  694. * @private {string}
  695. */
  696. shaka.polyfill.PatchedMediaKeysApple.apiName_ = 'apple';
  697. shaka.polyfill.register(shaka.polyfill.PatchedMediaKeysApple.defaultInstall);