Source: lib/media/preload_manager.js

  1. /*! @license
  2. * Shaka Player
  3. * Copyright 2023 Google LLC
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. goog.provide('shaka.media.PreloadManager');
  7. goog.require('goog.asserts');
  8. goog.require('shaka.log');
  9. goog.require('shaka.util.Error');
  10. goog.require('shaka.media.ManifestFilterer');
  11. goog.require('shaka.media.ManifestParser');
  12. goog.require('shaka.util.PublicPromise');
  13. goog.require('shaka.media.PreferenceBasedCriteria');
  14. goog.require('shaka.util.Stats');
  15. goog.require('shaka.media.SegmentPrefetch');
  16. goog.require('shaka.util.IDestroyable');
  17. goog.require('shaka.net.NetworkingEngine');
  18. goog.require('shaka.media.AdaptationSetCriteria');
  19. goog.require('shaka.media.DrmEngine');
  20. goog.require('shaka.media.RegionTimeline');
  21. goog.require('shaka.media.QualityObserver');
  22. goog.require('shaka.util.StreamUtils');
  23. goog.require('shaka.media.StreamingEngine');
  24. goog.require('shaka.media.SegmentPrefetch');
  25. goog.require('shaka.util.ConfigUtils');
  26. goog.require('shaka.util.FakeEvent');
  27. goog.require('shaka.util.FakeEventTarget');
  28. goog.require('shaka.util.ObjectUtils');
  29. goog.require('shaka.util.Platform');
  30. goog.require('shaka.util.PlayerConfiguration');
  31. /**
  32. * @implements {shaka.util.IDestroyable}
  33. * @export
  34. */
  35. shaka.media.PreloadManager = class extends shaka.util.FakeEventTarget {
  36. /**
  37. * @param {string} assetUri
  38. * @param {?string} mimeType
  39. * @param {?number} startTime
  40. * @param {*} playerInterface
  41. */
  42. constructor(assetUri, mimeType, startTime, playerInterface) {
  43. super();
  44. // Making the playerInterface a * and casting it to the right type allows
  45. // for the PlayerInterface for this class to not be exported.
  46. // Unfortunately, the constructor is exported by default.
  47. const typedPlayerInterface =
  48. /** @type {!shaka.media.PreloadManager.PlayerInterface} */ (
  49. playerInterface);
  50. /** @private {string} */
  51. this.assetUri_ = assetUri;
  52. /** @private {?string} */
  53. this.mimeType_ = mimeType;
  54. /** @private {!shaka.net.NetworkingEngine} */
  55. this.networkingEngine_ = typedPlayerInterface.networkingEngine;
  56. /** @private {?number} */
  57. this.startTime_ = startTime;
  58. /** @private {?shaka.media.AdaptationSetCriteria} */
  59. this.currentAdaptationSetCriteria_ = null;
  60. /** @private {number} */
  61. this.startTimeOfDrm_ = 0;
  62. /** @private {function():!shaka.media.DrmEngine} */
  63. this.createDrmEngine_ = typedPlayerInterface.createDrmEngine;
  64. /** @private {!shaka.media.ManifestFilterer} */
  65. this.manifestFilterer_ = typedPlayerInterface.manifestFilterer;
  66. /** @private {!shaka.extern.ManifestParser.PlayerInterface} */
  67. this.manifestPlayerInterface_ =
  68. typedPlayerInterface.manifestPlayerInterface;
  69. /** @private {!shaka.extern.PlayerConfiguration} */
  70. this.config_ = typedPlayerInterface.config;
  71. /** @private {?shaka.extern.Manifest} */
  72. this.manifest_ = null;
  73. /** @private {?shaka.extern.ManifestParser.Factory} */
  74. this.parserFactory_ = null;
  75. /** @private {?shaka.extern.ManifestParser} */
  76. this.parser_ = null;
  77. /** @private {boolean} */
  78. this.parserEntrusted_ = false;
  79. /** @private {!shaka.media.RegionTimeline} */
  80. this.regionTimeline_ = typedPlayerInterface.regionTimeline;
  81. /** @private {boolean} */
  82. this.regionTimelineEntrusted_ = false;
  83. /** @private {?shaka.media.DrmEngine} */
  84. this.drmEngine_ = null;
  85. /** @private {boolean} */
  86. this.drmEngineEntrusted_ = false;
  87. /** @private {?shaka.extern.AbrManager.Factory} */
  88. this.abrManagerFactory_ = null;
  89. /** @private {shaka.extern.AbrManager} */
  90. this.abrManager_ = null;
  91. /** @private {boolean} */
  92. this.abrManagerEntrusted_ = false;
  93. /** @private {!Map.<number, shaka.media.SegmentPrefetch>} */
  94. this.segmentPrefetchById_ = new Map();
  95. /** @private {boolean} */
  96. this.segmentPrefetchEntrusted_ = false;
  97. /** @private {?shaka.media.QualityObserver} */
  98. this.qualityObserver_ = typedPlayerInterface.qualityObserver;
  99. /** @private {!shaka.util.Stats} */
  100. this.stats_ = new shaka.util.Stats();
  101. /** @private {!shaka.util.PublicPromise} */
  102. this.manifestPromise_ = new shaka.util.PublicPromise();
  103. /** @private {!shaka.util.PublicPromise} */
  104. this.successPromise_ = new shaka.util.PublicPromise();
  105. /** @private {?shaka.util.FakeEventTarget} */
  106. this.eventHandoffTarget_ = null;
  107. /** @private {boolean} */
  108. this.destroyed_ = false;
  109. /** @private {boolean} */
  110. this.allowPrefetch_ = typedPlayerInterface.allowPrefetch;
  111. /** @private {?shaka.extern.Variant} */
  112. this.prefetchedVariant_ = null;
  113. /** @private {boolean} */
  114. this.allowMakeAbrManager_ = typedPlayerInterface.allowMakeAbrManager;
  115. /** @private {boolean} */
  116. this.hasBeenAttached_ = false;
  117. /** @private {?Array.<function()>} */
  118. this.queuedOperations_ = [];
  119. /** @private {?Array.<function()>} */
  120. this.latePhaseQueuedOperations_ = [];
  121. /** @private {boolean} */
  122. this.isPreload_ = true;
  123. }
  124. /**
  125. * Makes it so that net requests launched from this load will no longer be
  126. * marked as "isPreload"
  127. */
  128. markIsLoad() {
  129. this.isPreload_ = false;
  130. }
  131. /**
  132. * @param {boolean} latePhase
  133. * @param {function()} callback
  134. */
  135. addQueuedOperation(latePhase, callback) {
  136. const queue =
  137. latePhase ? this.latePhaseQueuedOperations_ : this.queuedOperations_;
  138. if (queue) {
  139. queue.push(callback);
  140. } else {
  141. callback();
  142. }
  143. }
  144. /** Calls all late phase queued operations, and stops queueing them. */
  145. stopQueuingLatePhaseQueuedOperations() {
  146. if (this.latePhaseQueuedOperations_) {
  147. for (const callback of this.latePhaseQueuedOperations_) {
  148. callback();
  149. }
  150. }
  151. this.latePhaseQueuedOperations_ = null;
  152. }
  153. /** @param {!shaka.util.FakeEventTarget} eventHandoffTarget */
  154. setEventHandoffTarget(eventHandoffTarget) {
  155. this.eventHandoffTarget_ = eventHandoffTarget;
  156. this.hasBeenAttached_ = true;
  157. // Also call all queued operations, and stop queuing them in the future.
  158. if (this.queuedOperations_) {
  159. for (const callback of this.queuedOperations_) {
  160. callback();
  161. }
  162. }
  163. this.queuedOperations_ = null;
  164. }
  165. /** @param {number} offset */
  166. setOffsetToStartTime(offset) {
  167. if (this.startTime_ && offset) {
  168. this.startTime_ += offset;
  169. }
  170. }
  171. /** @return {?number} */
  172. getStartTime() {
  173. return this.startTime_;
  174. }
  175. /** @return {number} */
  176. getStartTimeOfDRM() {
  177. return this.startTimeOfDrm_;
  178. }
  179. /** @return {?string} */
  180. getMimeType() {
  181. return this.mimeType_;
  182. }
  183. /** @return {string} */
  184. getAssetUri() {
  185. return this.assetUri_;
  186. }
  187. /** @return {?shaka.extern.Manifest} */
  188. getManifest() {
  189. return this.manifest_;
  190. }
  191. /** @return {?shaka.extern.ManifestParser.Factory} */
  192. getParserFactory() {
  193. return this.parserFactory_;
  194. }
  195. /** @return {?shaka.media.AdaptationSetCriteria} */
  196. getCurrentAdaptationSetCriteria() {
  197. return this.currentAdaptationSetCriteria_;
  198. }
  199. /** @return {?shaka.extern.AbrManager.Factory} */
  200. getAbrManagerFactory() {
  201. return this.abrManagerFactory_;
  202. }
  203. /**
  204. * Gets the abr manager, if it exists. Also marks that the abr manager should
  205. * not be stopped if this manager is destroyed.
  206. * @return {?shaka.extern.AbrManager}
  207. */
  208. receiveAbrManager() {
  209. this.abrManagerEntrusted_ = true;
  210. return this.abrManager_;
  211. }
  212. /**
  213. * @return {?shaka.extern.AbrManager}
  214. */
  215. getAbrManager() {
  216. return this.abrManager_;
  217. }
  218. /**
  219. * Gets the parser, if it exists. Also marks that the parser should not be
  220. * stopped if this manager is destroyed.
  221. * @return {?shaka.extern.ManifestParser}
  222. */
  223. receiveParser() {
  224. this.parserEntrusted_ = true;
  225. return this.parser_;
  226. }
  227. /**
  228. * @return {?shaka.extern.ManifestParser}
  229. */
  230. getParser() {
  231. return this.parser_;
  232. }
  233. /**
  234. * Gets the region timeline, if it exists. Also marks that the timeline should
  235. * not be released if this manager is destroyed.
  236. * @return {?shaka.media.RegionTimeline}
  237. */
  238. receiveRegionTimeline() {
  239. this.regionTimelineEntrusted_ = true;
  240. return this.regionTimeline_;
  241. }
  242. /**
  243. * @return {?shaka.media.RegionTimeline}
  244. */
  245. getRegionTimeline() {
  246. return this.regionTimeline_;
  247. }
  248. /** @return {?shaka.media.QualityObserver} */
  249. getQualityObserver() {
  250. return this.qualityObserver_;
  251. }
  252. /** @return {!shaka.util.Stats} */
  253. getStats() {
  254. return this.stats_;
  255. }
  256. /** @return {!shaka.media.ManifestFilterer} */
  257. getManifestFilterer() {
  258. return this.manifestFilterer_;
  259. }
  260. /**
  261. * Gets the drm engine, if it exists. Also marks that the drm engine should
  262. * not be destroyed if this manager is destroyed.
  263. * @return {?shaka.media.DrmEngine}
  264. */
  265. receiveDrmEngine() {
  266. this.drmEngineEntrusted_ = true;
  267. return this.drmEngine_;
  268. }
  269. /**
  270. * @return {?shaka.media.DrmEngine}
  271. */
  272. getDrmEngine() {
  273. return this.drmEngine_;
  274. }
  275. /**
  276. * @return {?shaka.extern.Variant}
  277. */
  278. getPrefetchedVariant() {
  279. return this.prefetchedVariant_;
  280. }
  281. /**
  282. * Gets the SegmentPrefetch objects for the initial stream ids. Also marks
  283. * that those objects should not be aborted if this manager is destroyed.
  284. * @return {!Map.<number, shaka.media.SegmentPrefetch>}
  285. */
  286. receiveSegmentPrefetchesById() {
  287. this.segmentPrefetchEntrusted_ = true;
  288. return this.segmentPrefetchById_;
  289. }
  290. /**
  291. * @param {?shaka.extern.AbrManager} abrManager
  292. * @param {?shaka.extern.AbrManager.Factory} abrFactory
  293. */
  294. attachAbrManager(abrManager, abrFactory) {
  295. this.abrManager_ = abrManager;
  296. this.abrManagerFactory_ = abrFactory;
  297. }
  298. /**
  299. * @param {?shaka.media.AdaptationSetCriteria} adaptationSetCriteria
  300. */
  301. attachAdaptationSetCriteria(adaptationSetCriteria) {
  302. this.currentAdaptationSetCriteria_ = adaptationSetCriteria;
  303. }
  304. /**
  305. * @param {!shaka.extern.Manifest} manifest
  306. * @param {!shaka.extern.ManifestParser} parser
  307. * @param {!shaka.extern.ManifestParser.Factory} parserFactory
  308. */
  309. attachManifest(manifest, parser, parserFactory) {
  310. this.manifest_ = manifest;
  311. this.parser_ = parser;
  312. this.parserFactory_ = parserFactory;
  313. }
  314. /**
  315. * Starts the process of loading the asset.
  316. * Success or failure will be measured through waitForFinish()
  317. */
  318. start() {
  319. (async () => {
  320. // Force a context switch, to give the player a chance to hook up events
  321. // immediately if desired.
  322. await Promise.resolve();
  323. // Perform the preloading process.
  324. try {
  325. await this.parseManifestInner_();
  326. this.throwIfDestroyed_();
  327. if (!shaka.util.Platform.isMediaKeysPolyfilled('webkit')) {
  328. await this.initializeDrm();
  329. this.throwIfDestroyed_();
  330. }
  331. await this.chooseInitialVariantAndPrefetchInner_();
  332. this.throwIfDestroyed_();
  333. // We don't need the drm keys to load completely for the initial variant
  334. // to be chosen, but we won't mark the load as a success until it has
  335. // been loaded. So wait for it here, not inside initializeDrmInner_.
  336. if (this.drmEngine_) {
  337. await this.drmEngine_.waitForActiveRequests();
  338. this.throwIfDestroyed_();
  339. }
  340. this.successPromise_.resolve();
  341. } catch (error) {
  342. // Ignore OPERATION_ABORTED and OBJECT_DESTROYED errors.
  343. if (!(error instanceof shaka.util.Error) ||
  344. (error.code != shaka.util.Error.Code.OPERATION_ABORTED &&
  345. error.code != shaka.util.Error.Code.OBJECT_DESTROYED)) {
  346. this.successPromise_.reject(error);
  347. }
  348. }
  349. })();
  350. }
  351. /**
  352. * @param {!Event} event
  353. * @return {boolean}
  354. * @override
  355. */
  356. dispatchEvent(event) {
  357. if (this.eventHandoffTarget_) {
  358. return this.eventHandoffTarget_.dispatchEvent(event);
  359. } else {
  360. return super.dispatchEvent(event);
  361. }
  362. }
  363. /**
  364. * @param {!shaka.util.Error} error
  365. */
  366. onError(error) {
  367. if (error.severity === shaka.util.Error.Severity.CRITICAL) {
  368. // Cancel the loading process.
  369. this.successPromise_.reject(error);
  370. this.destroy();
  371. }
  372. const eventName = shaka.util.FakeEvent.EventName.Error;
  373. const event = this.makeEvent_(eventName, (new Map()).set('detail', error));
  374. this.dispatchEvent(event);
  375. if (event.defaultPrevented) {
  376. error.handled = true;
  377. }
  378. }
  379. /**
  380. * Throw if destroyed, to interrupt processes with a recognizable error.
  381. *
  382. * @private
  383. */
  384. throwIfDestroyed_() {
  385. if (this.isDestroyed()) {
  386. throw new shaka.util.Error(
  387. shaka.util.Error.Severity.CRITICAL,
  388. shaka.util.Error.Category.PLAYER,
  389. shaka.util.Error.Code.OBJECT_DESTROYED);
  390. }
  391. }
  392. /**
  393. * Makes a fires an event corresponding to entering a state of the loading
  394. * process.
  395. * @param {string} nodeName
  396. * @private
  397. */
  398. makeStateChangeEvent_(nodeName) {
  399. this.dispatchEvent(new shaka.util.FakeEvent(
  400. /* name= */ shaka.util.FakeEvent.EventName.OnStateChange,
  401. /* data= */ (new Map()).set('state', nodeName)));
  402. }
  403. /**
  404. * @param {!shaka.util.FakeEvent.EventName} name
  405. * @param {Map.<string, Object>=} data
  406. * @return {!shaka.util.FakeEvent}
  407. * @private
  408. */
  409. makeEvent_(name, data) {
  410. return new shaka.util.FakeEvent(name, data);
  411. }
  412. /**
  413. * Pick and initialize a manifest parser, then have it download and parse the
  414. * manifest.
  415. *
  416. * @return {!Promise}
  417. * @private
  418. */
  419. async parseManifestInner_() {
  420. this.makeStateChangeEvent_('manifest-parser');
  421. if (!this.parser_) {
  422. // Create the parser that we will use to parse the manifest.
  423. this.parserFactory_ = shaka.media.ManifestParser.getFactory(
  424. this.assetUri_, this.mimeType_);
  425. goog.asserts.assert(this.parserFactory_, 'Must have manifest parser');
  426. this.parser_ = this.parserFactory_();
  427. this.parser_.configure(this.config_.manifest, () => this.isPreload_);
  428. }
  429. const startTime = Date.now() / 1000;
  430. this.makeStateChangeEvent_('manifest');
  431. if (!this.manifest_) {
  432. this.manifest_ = await this.parser_.start(
  433. this.assetUri_, this.manifestPlayerInterface_);
  434. }
  435. this.manifestPromise_.resolve();
  436. // This event is fired after the manifest is parsed, but before any
  437. // filtering takes place.
  438. const event =
  439. this.makeEvent_(shaka.util.FakeEvent.EventName.ManifestParsed);
  440. // Delay event to ensure manifest has been properly propagated
  441. // to the player.
  442. await Promise.resolve();
  443. this.dispatchEvent(event);
  444. // We require all manifests to have at least one variant.
  445. if (this.manifest_.variants.length == 0) {
  446. throw new shaka.util.Error(
  447. shaka.util.Error.Severity.CRITICAL,
  448. shaka.util.Error.Category.MANIFEST,
  449. shaka.util.Error.Code.NO_VARIANTS);
  450. }
  451. // Make sure that all variants are either: audio-only, video-only, or
  452. // audio-video.
  453. shaka.media.PreloadManager.filterForAVVariants_(this.manifest_);
  454. const now = Date.now() / 1000;
  455. const delta = now - startTime;
  456. this.stats_.setManifestTime(delta);
  457. }
  458. /**
  459. * Initializes the DRM engine.
  460. * @param {?HTMLMediaElement=} media
  461. * @return {!Promise}
  462. */
  463. async initializeDrm(media) {
  464. if (!this.manifest_ || this.drmEngine_) {
  465. return;
  466. }
  467. this.makeStateChangeEvent_('drm-engine');
  468. this.startTimeOfDrm_ = Date.now() / 1000;
  469. this.drmEngine_ = this.createDrmEngine_();
  470. this.manifestFilterer_.setDrmEngine(this.drmEngine_);
  471. this.drmEngine_.configure(this.config_.drm, () => this.isPreload_);
  472. const tracksChangedInitial = this.manifestFilterer_.applyRestrictions(
  473. this.manifest_);
  474. if (tracksChangedInitial) {
  475. const event = this.makeEvent_(
  476. shaka.util.FakeEvent.EventName.TracksChanged);
  477. await Promise.resolve();
  478. this.throwIfDestroyed_();
  479. this.dispatchEvent(event);
  480. }
  481. const playableVariants = shaka.util.StreamUtils.getPlayableVariants(
  482. this.manifest_.variants);
  483. await this.drmEngine_.initForPlayback(
  484. playableVariants,
  485. this.manifest_.offlineSessionIds);
  486. this.throwIfDestroyed_();
  487. if (media) {
  488. await this.drmEngine_.attach(media);
  489. this.throwIfDestroyed_();
  490. }
  491. // Now that we have drm information, filter the manifest (again) so that
  492. // we can ensure we only use variants with the selected key system.
  493. const tracksChangedAfter = await this.manifestFilterer_.filterManifest(
  494. this.manifest_);
  495. if (tracksChangedAfter) {
  496. const event = this.makeEvent_(
  497. shaka.util.FakeEvent.EventName.TracksChanged);
  498. await Promise.resolve();
  499. this.dispatchEvent(event);
  500. }
  501. }
  502. /** @param {!shaka.extern.PlayerConfiguration} config */
  503. reconfigure(config) {
  504. this.config_ = config;
  505. }
  506. /**
  507. * @param {string} name
  508. * @param {*=} value
  509. */
  510. configure(name, value) {
  511. const config = shaka.util.ConfigUtils.convertToConfigObject(name, value);
  512. shaka.util.PlayerConfiguration.mergeConfigObjects(this.config_, config);
  513. }
  514. /**
  515. * Return a copy of the current configuration.
  516. *
  517. * @return {shaka.extern.PlayerConfiguration}
  518. */
  519. getConfiguration() {
  520. return shaka.util.ObjectUtils.cloneObject(this.config_);
  521. }
  522. /**
  523. * Performs a final filtering of the manifest, and chooses the initial
  524. * variant. Also prefetches segments.
  525. *
  526. * @return {!Promise}
  527. * @private
  528. */
  529. async chooseInitialVariantAndPrefetchInner_() {
  530. goog.asserts.assert(
  531. this.manifest_, 'The manifest should already be parsed.');
  532. // This step does not have any associated events, as it is only part of the
  533. // "load" state in the old state graph.
  534. if (!this.currentAdaptationSetCriteria_) {
  535. // Copy preferred languages from the config again, in case the config was
  536. // changed between construction and playback.
  537. this.currentAdaptationSetCriteria_ =
  538. new shaka.media.PreferenceBasedCriteria(
  539. this.config_.preferredAudioLanguage,
  540. this.config_.preferredVariantRole,
  541. this.config_.preferredAudioChannelCount,
  542. this.config_.preferredVideoHdrLevel,
  543. this.config_.preferSpatialAudio,
  544. this.config_.preferredVideoLayout,
  545. this.config_.preferredAudioLabel,
  546. this.config_.preferredVideoLabel,
  547. this.config_.mediaSource.codecSwitchingStrategy,
  548. /* audioCodec= */ '');
  549. }
  550. // Make the ABR manager.
  551. if (this.allowMakeAbrManager_) {
  552. const abrFactory = this.config_.abrFactory;
  553. this.abrManagerFactory_ = abrFactory;
  554. this.abrManager_ = abrFactory();
  555. this.abrManager_.configure(this.config_.abr);
  556. }
  557. if (this.allowPrefetch_) {
  558. const isLive = this.manifest_.presentationTimeline.isLive();
  559. // Prefetch segments for the predicted first variant.
  560. // We start these here, but don't wait for them; it's okay to start the
  561. // full load process while the segments are being prefetched.
  562. const playableVariants = shaka.util.StreamUtils.getPlayableVariants(
  563. this.manifest_.variants);
  564. const adaptationSet = this.currentAdaptationSetCriteria_.create(
  565. playableVariants);
  566. // Guess what the first variant will be, based on a SimpleAbrManager.
  567. this.abrManager_.configure(this.config_.abr);
  568. this.abrManager_.setVariants(Array.from(adaptationSet.values()));
  569. const variant = this.abrManager_.chooseVariant();
  570. if (variant) {
  571. const promises = [];
  572. this.prefetchedVariant_ = variant;
  573. if (variant.video) {
  574. promises.push(this.prefetchStream_(variant.video, isLive));
  575. }
  576. if (variant.audio) {
  577. promises.push(this.prefetchStream_(variant.audio, isLive));
  578. }
  579. await Promise.all(promises);
  580. }
  581. }
  582. }
  583. /**
  584. * @param {!shaka.extern.Stream} stream
  585. * @param {boolean} isLive
  586. * @return {!Promise}
  587. * @private
  588. */
  589. async prefetchStream_(stream, isLive) {
  590. // Use the prefetch limit from the config if this is set, otherwise use 2.
  591. const prefetchLimit = this.config_.streaming.segmentPrefetchLimit || 2;
  592. const prefetch = new shaka.media.SegmentPrefetch(
  593. prefetchLimit, stream, (reference, stream, streamDataCallback) => {
  594. return shaka.media.StreamingEngine.dispatchFetch(
  595. reference, stream, streamDataCallback || null,
  596. this.config_.streaming.retryParameters, this.networkingEngine_,
  597. this.isPreload_);
  598. }, /* reverse= */ false);
  599. this.segmentPrefetchById_.set(stream.id, prefetch);
  600. // Start prefetching a bit.
  601. await stream.createSegmentIndex();
  602. const startTime = this.startTime_ || 0;
  603. const prefetchSegmentIterator =
  604. stream.segmentIndex.getIteratorForTime(startTime);
  605. let prefetchSegment =
  606. prefetchSegmentIterator ? prefetchSegmentIterator.current() : null;
  607. if (!prefetchSegment) {
  608. // If we can't get a segment at the desired spot, at least get a segment,
  609. // so we can get the init segment.
  610. prefetchSegment = stream.segmentIndex.earliestReference();
  611. }
  612. if (prefetchSegment) {
  613. if (isLive) {
  614. // Preload only the init segment for Live
  615. if (prefetchSegment.initSegmentReference) {
  616. await prefetch.prefetchInitSegment(
  617. prefetchSegment.initSegmentReference);
  618. }
  619. } else {
  620. // Preload a segment, too... either the first segment, or the segment
  621. // that corresponds with this.startTime_, as appropriate.
  622. // Note: this method also preload the init segment
  623. await prefetch.prefetchSegmentsByTime(prefetchSegment.startTime);
  624. }
  625. }
  626. }
  627. /**
  628. * Waits for the loading to be finished (or to fail with an error).
  629. * @return {!Promise}
  630. * @export
  631. */
  632. waitForFinish() {
  633. return this.successPromise_;
  634. }
  635. /**
  636. * Waits for the manifest to be loaded (or to fail with an error).
  637. * @return {!Promise}
  638. */
  639. waitForManifest() {
  640. const promises = [
  641. this.manifestPromise_,
  642. this.successPromise_,
  643. ];
  644. return Promise.race(promises);
  645. }
  646. /**
  647. * Releases or stops all non-entrusted resources.
  648. *
  649. * @override
  650. * @export
  651. */
  652. async destroy() {
  653. this.destroyed_ = true;
  654. if (this.parser_ && !this.parserEntrusted_) {
  655. await this.parser_.stop();
  656. }
  657. if (this.abrManager_ && !this.abrManagerEntrusted_) {
  658. await this.abrManager_.stop();
  659. }
  660. if (this.regionTimeline_ && !this.regionTimelineEntrusted_) {
  661. this.regionTimeline_.release();
  662. }
  663. if (this.drmEngine_ && !this.drmEngineEntrusted_) {
  664. await this.drmEngine_.destroy();
  665. }
  666. if (this.segmentPrefetchById_.size > 0 && !this.segmentPrefetchEntrusted_) {
  667. for (const segmentPrefetch of this.segmentPrefetchById_.values()) {
  668. segmentPrefetch.clearAll();
  669. }
  670. }
  671. // this.eventHandoffTarget_ is not unset, so that events and errors fired
  672. // after the preload manager is destroyed will still be routed to the
  673. // player, if it was once linked up.
  674. }
  675. /** @return {boolean} */
  676. isDestroyed() {
  677. return this.destroyed_;
  678. }
  679. /** @return {boolean} */
  680. hasBeenAttached() {
  681. return this.hasBeenAttached_;
  682. }
  683. /**
  684. * Take a series of variants and ensure that they only contain one type of
  685. * variant. The different options are:
  686. * 1. Audio-Video
  687. * 2. Audio-Only
  688. * 3. Video-Only
  689. *
  690. * A manifest can only contain a single type because once we initialize media
  691. * source to expect specific streams, it must always have content for those
  692. * streams. If we were to start with audio+video and switch to an audio-only
  693. * variant, media source would block waiting for video content.
  694. *
  695. * @param {shaka.extern.Manifest} manifest
  696. * @private
  697. */
  698. static filterForAVVariants_(manifest) {
  699. const isAVVariant = (variant) => {
  700. // Audio-video variants may include both streams separately or may be
  701. // single multiplexed streams with multiple codecs.
  702. return (variant.video && variant.audio) ||
  703. (variant.video && variant.video.codecs.includes(','));
  704. };
  705. if (manifest.variants.some(isAVVariant)) {
  706. shaka.log.debug('Found variant with audio and video content, ' +
  707. 'so filtering out audio-only content.');
  708. manifest.variants = manifest.variants.filter(isAVVariant);
  709. }
  710. }
  711. };
  712. /**
  713. * @typedef {{
  714. * config: !shaka.extern.PlayerConfiguration,
  715. * manifestPlayerInterface: !shaka.extern.ManifestParser.PlayerInterface,
  716. * regionTimeline: !shaka.media.RegionTimeline,
  717. * qualityObserver: ?shaka.media.QualityObserver,
  718. * createDrmEngine: function():!shaka.media.DrmEngine,
  719. * networkingEngine: !shaka.net.NetworkingEngine,
  720. * manifestFilterer: !shaka.media.ManifestFilterer,
  721. * allowPrefetch: boolean,
  722. * allowMakeAbrManager: boolean
  723. * }}
  724. *
  725. * @property {!shaka.extern.PlayerConfiguration} config
  726. * @property {!shaka.extern.ManifestParser.PlayerInterface}
  727. * manifestPlayerInterface
  728. * @property {!shaka.media.RegionTimeline} regionTimeline
  729. * @property {?shaka.media.QualityObserver} qualityObserver
  730. * @property {function():!shaka.media.DrmEngine} createDrmEngine
  731. * @property {!shaka.net.NetworkingEngine} networkingEngine
  732. * @property {!shaka.media.ManifestFilterer} manifestFilterer
  733. * @property {boolean} allowPrefetch
  734. * @property {boolean} allowMakeAbrManager
  735. */
  736. shaka.media.PreloadManager.PlayerInterface;