Source: lib/abr/simple_abr_manager.js

  1. /*! @license
  2. * Shaka Player
  3. * Copyright 2016 Google LLC
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. goog.provide('shaka.abr.SimpleAbrManager');
  7. goog.require('goog.asserts');
  8. goog.require('shaka.abr.EwmaBandwidthEstimator');
  9. goog.require('shaka.log');
  10. goog.require('shaka.util.StreamUtils');
  11. /**
  12. * @summary
  13. * <p>
  14. * This defines the default ABR manager for the Player. An instance of this
  15. * class is used when no ABR manager is given.
  16. * </p>
  17. * <p>
  18. * The behavior of this class is to take throughput samples using
  19. * segmentDownloaded to estimate the current network bandwidth. Then it will
  20. * use that to choose the streams that best fit the current bandwidth. It will
  21. * always pick the highest bandwidth variant it thinks can be played.
  22. * </p>
  23. * <p>
  24. * After initial choices are made, this class will call switchCallback() when
  25. * there is a better choice. switchCallback() will not be called more than once
  26. * per ({@link shaka.abr.SimpleAbrManager.SWITCH_INTERVAL_MS}).
  27. * </p>
  28. *
  29. * @implements {shaka.extern.AbrManager}
  30. * @export
  31. */
  32. shaka.abr.SimpleAbrManager = class {
  33. /** */
  34. constructor() {
  35. /** @private {?shaka.extern.AbrManager.SwitchCallback} */
  36. this.switch_ = null;
  37. /** @private {boolean} */
  38. this.enabled_ = false;
  39. /** @private {shaka.abr.EwmaBandwidthEstimator} */
  40. this.bandwidthEstimator_ = new shaka.abr.EwmaBandwidthEstimator();
  41. // Some browsers implement the Network Information API, which allows
  42. // retrieving information about a user's network connection. We listen
  43. // to the change event to be able to make quick changes in case the type
  44. // of connectivity changes.
  45. if (navigator.connection) {
  46. navigator.connection.addEventListener('change', () => {
  47. if (this.config_.useNetworkInformation && this.enabled_) {
  48. this.bandwidthEstimator_ = new shaka.abr.EwmaBandwidthEstimator();
  49. if (this.config_) {
  50. this.bandwidthEstimator_.configure(this.config_.advanced);
  51. }
  52. const chosenVariant = this.chooseVariant();
  53. if (chosenVariant) {
  54. this.switch_(chosenVariant);
  55. }
  56. }
  57. });
  58. }
  59. /**
  60. * A filtered list of Variants to choose from.
  61. * @private {!Array.<!shaka.extern.Variant>}
  62. */
  63. this.variants_ = [];
  64. /** @private {number} */
  65. this.playbackRate_ = 1;
  66. /** @private {boolean} */
  67. this.startupComplete_ = false;
  68. /**
  69. * The last wall-clock time, in milliseconds, when streams were chosen.
  70. *
  71. * @private {?number}
  72. */
  73. this.lastTimeChosenMs_ = null;
  74. /** @private {?shaka.extern.AbrConfiguration} */
  75. this.config_ = null;
  76. }
  77. /**
  78. * @override
  79. * @export
  80. */
  81. stop() {
  82. this.switch_ = null;
  83. this.enabled_ = false;
  84. this.variants_ = [];
  85. this.playbackRate_ = 1;
  86. this.lastTimeChosenMs_ = null;
  87. // Don't reset |startupComplete_|: if we've left the startup interval, we
  88. // can start using bandwidth estimates right away after init() is called.
  89. }
  90. /**
  91. * @override
  92. * @export
  93. */
  94. init(switchCallback) {
  95. this.switch_ = switchCallback;
  96. }
  97. /**
  98. * @override
  99. * @export
  100. */
  101. chooseVariant() {
  102. const SimpleAbrManager = shaka.abr.SimpleAbrManager;
  103. // Get sorted Variants.
  104. let sortedVariants = SimpleAbrManager.filterAndSortVariants_(
  105. this.config_.restrictions, this.variants_);
  106. const defaultBandwidthEstimate = this.getDefaultBandwidth_();
  107. const currentBandwidth = this.bandwidthEstimator_.getBandwidthEstimate(
  108. defaultBandwidthEstimate);
  109. if (this.variants_.length && !sortedVariants.length) {
  110. // If we couldn't meet the ABR restrictions, we should still play
  111. // something.
  112. // These restrictions are not "hard" restrictions in the way that
  113. // top-level or DRM-based restrictions are. Sort the variants without
  114. // restrictions and keep just the first (lowest-bandwidth) one.
  115. shaka.log.warning('No variants met the ABR restrictions. ' +
  116. 'Choosing a variant by lowest bandwidth.');
  117. sortedVariants = SimpleAbrManager.filterAndSortVariants_(
  118. /* restrictions= */ null, this.variants_);
  119. sortedVariants = [sortedVariants[0]];
  120. }
  121. // Start by assuming that we will use the first Stream.
  122. let chosen = sortedVariants[0] || null;
  123. for (let i = 0; i < sortedVariants.length; i++) {
  124. const item = sortedVariants[i];
  125. const playbackRate =
  126. !isNaN(this.playbackRate_) ? Math.abs(this.playbackRate_) : 1;
  127. const itemBandwidth = playbackRate * item.bandwidth;
  128. const minBandwidth =
  129. itemBandwidth / this.config_.bandwidthDowngradeTarget;
  130. let next = {bandwidth: Infinity};
  131. for (let j = i + 1; j < sortedVariants.length; j++) {
  132. if (item.bandwidth != sortedVariants[j].bandwidth) {
  133. next = sortedVariants[j];
  134. break;
  135. }
  136. }
  137. const nextBandwidth = playbackRate * next.bandwidth;
  138. const maxBandwidth = nextBandwidth / this.config_.bandwidthUpgradeTarget;
  139. shaka.log.v2('Bandwidth ranges:',
  140. (itemBandwidth / 1e6).toFixed(3),
  141. (minBandwidth / 1e6).toFixed(3),
  142. (maxBandwidth / 1e6).toFixed(3));
  143. if (currentBandwidth >= minBandwidth &&
  144. currentBandwidth <= maxBandwidth &&
  145. chosen.bandwidth != item.bandwidth) {
  146. chosen = item;
  147. }
  148. }
  149. this.lastTimeChosenMs_ = Date.now();
  150. return chosen;
  151. }
  152. /**
  153. * @override
  154. * @export
  155. */
  156. enable() {
  157. this.enabled_ = true;
  158. }
  159. /**
  160. * @override
  161. * @export
  162. */
  163. disable() {
  164. this.enabled_ = false;
  165. }
  166. /**
  167. * @override
  168. * @export
  169. */
  170. segmentDownloaded(deltaTimeMs, numBytes) {
  171. shaka.log.v2('Segment downloaded:',
  172. 'deltaTimeMs=' + deltaTimeMs,
  173. 'numBytes=' + numBytes,
  174. 'lastTimeChosenMs=' + this.lastTimeChosenMs_,
  175. 'enabled=' + this.enabled_);
  176. goog.asserts.assert(deltaTimeMs >= 0, 'expected a non-negative duration');
  177. this.bandwidthEstimator_.sample(deltaTimeMs, numBytes);
  178. if ((this.lastTimeChosenMs_ != null) && this.enabled_) {
  179. this.suggestStreams_();
  180. }
  181. }
  182. /**
  183. * @override
  184. * @export
  185. */
  186. getBandwidthEstimate() {
  187. const defaultBandwidthEstimate = this.getDefaultBandwidth_();
  188. return this.bandwidthEstimator_.getBandwidthEstimate(
  189. defaultBandwidthEstimate);
  190. }
  191. /**
  192. * @override
  193. * @export
  194. */
  195. setVariants(variants) {
  196. this.variants_ = variants;
  197. }
  198. /**
  199. * @override
  200. * @export
  201. */
  202. playbackRateChanged(rate) {
  203. this.playbackRate_ = rate;
  204. }
  205. /**
  206. * @override
  207. * @export
  208. */
  209. configure(config) {
  210. this.config_ = config;
  211. if (this.bandwidthEstimator_ && this.config_) {
  212. this.bandwidthEstimator_.configure(this.config_.advanced);
  213. }
  214. }
  215. /**
  216. * Calls switch_() with the variant chosen by chooseVariant().
  217. *
  218. * @private
  219. */
  220. suggestStreams_() {
  221. shaka.log.v2('Suggesting Streams...');
  222. goog.asserts.assert(this.lastTimeChosenMs_ != null,
  223. 'lastTimeChosenMs_ should not be null');
  224. if (!this.startupComplete_) {
  225. // Check if we've got enough data yet.
  226. if (!this.bandwidthEstimator_.hasGoodEstimate()) {
  227. shaka.log.v2('Still waiting for a good estimate...');
  228. return;
  229. }
  230. this.startupComplete_ = true;
  231. } else {
  232. // Check if we've left the switch interval.
  233. const now = Date.now();
  234. const delta = now - this.lastTimeChosenMs_;
  235. if (delta < this.config_.switchInterval * 1000) {
  236. shaka.log.v2('Still within switch interval...');
  237. return;
  238. }
  239. }
  240. const chosenVariant = this.chooseVariant();
  241. const defaultBandwidthEstimate = this.getDefaultBandwidth_();
  242. const bandwidthEstimate = this.bandwidthEstimator_.getBandwidthEstimate(
  243. defaultBandwidthEstimate);
  244. const currentBandwidthKbps = Math.round(bandwidthEstimate / 1000.0);
  245. if (chosenVariant) {
  246. shaka.log.debug(
  247. 'Calling switch_(), bandwidth=' + currentBandwidthKbps + ' kbps');
  248. // If any of these chosen streams are already chosen, Player will filter
  249. // them out before passing the choices on to StreamingEngine.
  250. this.switch_(chosenVariant);
  251. }
  252. }
  253. /**
  254. * @private
  255. */
  256. getDefaultBandwidth_() {
  257. let defaultBandwidthEstimate = this.config_.defaultBandwidthEstimate;
  258. // Some browsers implement the Network Information API, which allows
  259. // retrieving information about a user's network connection. Tizen 3 has
  260. // NetworkInformation, but not the downlink attribute.
  261. if (navigator.connection && navigator.connection.downlink &&
  262. this.config_.useNetworkInformation) {
  263. // If it's available, get the bandwidth estimate from the browser (in
  264. // megabits per second) and use it as defaultBandwidthEstimate.
  265. defaultBandwidthEstimate = navigator.connection.downlink * 1e6;
  266. }
  267. return defaultBandwidthEstimate;
  268. }
  269. /**
  270. * @param {?shaka.extern.Restrictions} restrictions
  271. * @param {!Array.<shaka.extern.Variant>} variants
  272. * @return {!Array.<shaka.extern.Variant>} variants filtered according to
  273. * |restrictions| and sorted in ascending order of bandwidth.
  274. * @private
  275. */
  276. static filterAndSortVariants_(restrictions, variants) {
  277. if (restrictions) {
  278. variants = variants.filter((variant) => {
  279. // This was already checked in another scope, but the compiler doesn't
  280. // seem to understand that.
  281. goog.asserts.assert(restrictions, 'Restrictions should exist!');
  282. return shaka.util.StreamUtils.meetsRestrictions(
  283. variant, restrictions,
  284. /* maxHwRes= */ {width: Infinity, height: Infinity});
  285. });
  286. }
  287. return variants.sort((v1, v2) => {
  288. return v1.bandwidth - v2.bandwidth;
  289. });
  290. }
  291. };