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