Source: lib/media/presentation_timeline.js

  1. /**
  2. * @license
  3. * Copyright 2016 Google Inc.
  4. *
  5. * Licensed under the Apache License, Version 2.0 (the "License");
  6. * you may not use this file except in compliance with the License.
  7. * You may obtain a copy of the License at
  8. *
  9. * http://www.apache.org/licenses/LICENSE-2.0
  10. *
  11. * Unless required by applicable law or agreed to in writing, software
  12. * distributed under the License is distributed on an "AS IS" BASIS,
  13. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14. * See the License for the specific language governing permissions and
  15. * limitations under the License.
  16. */
  17. goog.provide('shaka.media.PresentationTimeline');
  18. goog.require('goog.asserts');
  19. goog.require('shaka.log');
  20. goog.require('shaka.media.SegmentReference');
  21. /**
  22. * Creates a PresentationTimeline.
  23. *
  24. * @param {?number} presentationStartTime The wall-clock time, in seconds,
  25. * when the presentation started or will start. Only required for live.
  26. * @param {number} presentationDelay The delay to give the presentation, in
  27. * seconds. Only required for live.
  28. *
  29. * @see {shakaExtern.Manifest}
  30. * @see {@tutorial architecture}
  31. *
  32. * @constructor
  33. * @struct
  34. * @export
  35. */
  36. shaka.media.PresentationTimeline = function(
  37. presentationStartTime, presentationDelay) {
  38. /** @private {?number} */
  39. this.presentationStartTime_ = presentationStartTime;
  40. /** @private {number} */
  41. this.presentationDelay_ = presentationDelay;
  42. /** @private {number} */
  43. this.duration_ = Infinity;
  44. /** @private {number} */
  45. this.segmentAvailabilityDuration_ = Infinity;
  46. /** @private {number} */
  47. this.maxSegmentDuration_ = 1;
  48. /** @private {number} */
  49. this.maxFirstSegmentStartTime_ = 0;
  50. /** @private {number} */
  51. this.clockOffset_ = 0;
  52. /** @private {boolean} */
  53. this.static_ = true;
  54. /** @private {number} */
  55. this.userSeekStart_ = 0;
  56. };
  57. /**
  58. * @return {number} The presentation's duration in seconds.
  59. * Infinity indicates that the presentation continues indefinitely.
  60. * @export
  61. */
  62. shaka.media.PresentationTimeline.prototype.getDuration = function() {
  63. return this.duration_;
  64. };
  65. /**
  66. * @return {number} The presentation's max segment duration in seconds.
  67. */
  68. shaka.media.PresentationTimeline.prototype.getMaxSegmentDuration = function() {
  69. return this.maxSegmentDuration_;
  70. };
  71. /**
  72. * Sets the presentation's duration.
  73. *
  74. * @param {number} duration The presentation's duration in seconds.
  75. * Infinity indicates that the presentation continues indefinitely.
  76. * @export
  77. */
  78. shaka.media.PresentationTimeline.prototype.setDuration = function(duration) {
  79. goog.asserts.assert(duration > 0, 'duration must be > 0');
  80. this.duration_ = duration;
  81. };
  82. /**
  83. * @return {?number} The presentation's start time in seconds.
  84. * @export
  85. */
  86. shaka.media.PresentationTimeline.prototype.getPresentationStartTime =
  87. function() {
  88. return this.presentationStartTime_;
  89. };
  90. /**
  91. * Sets the clock offset, which is the difference between the client's clock
  92. * and the server's clock, in milliseconds (i.e., serverTime = Date.now() +
  93. * clockOffset).
  94. *
  95. * @param {number} offset The clock offset, in ms.
  96. * @export
  97. */
  98. shaka.media.PresentationTimeline.prototype.setClockOffset = function(offset) {
  99. this.clockOffset_ = offset;
  100. };
  101. /**
  102. * Sets the presentation's static flag.
  103. *
  104. * @param {boolean} isStatic If true, the presentation is static, meaning all
  105. * segments are available at once.
  106. * @export
  107. */
  108. shaka.media.PresentationTimeline.prototype.setStatic = function(isStatic) {
  109. // NOTE: the argument name is not "static" because that's a keyword in ES6
  110. this.static_ = isStatic;
  111. };
  112. /**
  113. * Sets the presentation's segment availability duration. The segment
  114. * availability duration should only be set for live.
  115. *
  116. * @param {number} segmentAvailabilityDuration The presentation's new segment
  117. * availability duration in seconds.
  118. * @export
  119. */
  120. shaka.media.PresentationTimeline.prototype.setSegmentAvailabilityDuration =
  121. function(segmentAvailabilityDuration) {
  122. goog.asserts.assert(segmentAvailabilityDuration >= 0,
  123. 'segmentAvailabilityDuration must be >= 0');
  124. this.segmentAvailabilityDuration_ = segmentAvailabilityDuration;
  125. };
  126. /**
  127. * Sets the presentation delay.
  128. *
  129. * @param {number} delay
  130. * @export
  131. */
  132. shaka.media.PresentationTimeline.prototype.setDelay = function(delay) {
  133. goog.asserts.assert(delay >= 0, 'delay must be >= 0');
  134. this.presentationDelay_ = delay;
  135. };
  136. /**
  137. * Gives PresentationTimeline a Stream's segments so it can size and position
  138. * the segment availability window, and account for missing segment
  139. * information. This function should be called once for each Stream (no more,
  140. * no less).
  141. *
  142. * @param {!Array.<!shaka.media.SegmentReference>} references
  143. * @param {boolean} isFirstPeriod
  144. * @export
  145. */
  146. shaka.media.PresentationTimeline.prototype.notifySegments = function(
  147. references, isFirstPeriod) {
  148. if (references.length == 0) {
  149. return;
  150. }
  151. if (isFirstPeriod) {
  152. this.maxFirstSegmentStartTime_ =
  153. Math.max(this.maxFirstSegmentStartTime_, references[0].startTime);
  154. }
  155. this.maxSegmentDuration_ = references.reduce(
  156. function(max, r) { return Math.max(max, r.endTime - r.startTime); },
  157. this.maxSegmentDuration_);
  158. shaka.log.v1('notifySegments:',
  159. 'maxSegmentDuration=' + this.maxSegmentDuration_);
  160. };
  161. /**
  162. * Gives PresentationTimeline a Stream's maximum segment duration so it can
  163. * size and position the segment availability window. This function should be
  164. * called once for each Stream (no more, no less), but does not have to be
  165. * called if notifySegments() is called instead for a particular stream.
  166. *
  167. * @param {number} maxSegmentDuration The maximum segment duration for a
  168. * particular stream.
  169. * @export
  170. */
  171. shaka.media.PresentationTimeline.prototype.notifyMaxSegmentDuration = function(
  172. maxSegmentDuration) {
  173. this.maxSegmentDuration_ = Math.max(
  174. this.maxSegmentDuration_, maxSegmentDuration);
  175. shaka.log.v1('notifyNewSegmentDuration:',
  176. 'maxSegmentDuration=' + this.maxSegmentDuration_);
  177. };
  178. /**
  179. * Offsets the segment times by the given amount.
  180. *
  181. * @param {number} offset The number of seconds to offset by. A positive number
  182. * adjusts the segment times forward.
  183. * @export
  184. */
  185. shaka.media.PresentationTimeline.prototype.offset = function(offset) {
  186. this.maxFirstSegmentStartTime_ += offset;
  187. };
  188. /**
  189. * @return {boolean} True if the presentation is live; otherwise, return
  190. * false.
  191. * @export
  192. */
  193. shaka.media.PresentationTimeline.prototype.isLive = function() {
  194. return this.duration_ == Infinity &&
  195. !this.static_;
  196. };
  197. /**
  198. * @return {boolean} True if the presentation is in progress (meaning not live,
  199. * but also not completely available); otherwise, return false.
  200. * @export
  201. */
  202. shaka.media.PresentationTimeline.prototype.isInProgress = function() {
  203. return this.duration_ != Infinity &&
  204. !this.static_;
  205. };
  206. /**
  207. * Gets the presentation's current segment availability start time. Segments
  208. * ending at or before this time should be assumed to be unavailable.
  209. *
  210. * @return {number} The current segment availability start time, in seconds,
  211. * relative to the start of the presentation.
  212. * @export
  213. */
  214. shaka.media.PresentationTimeline.prototype.getSegmentAvailabilityStart =
  215. function() {
  216. goog.asserts.assert(this.segmentAvailabilityDuration_ >= 0,
  217. 'The availability duration should be positive');
  218. if (this.segmentAvailabilityDuration_ == Infinity) {
  219. return this.userSeekStart_;
  220. }
  221. let end = this.getSegmentAvailabilityEnd();
  222. let start = end - this.segmentAvailabilityDuration_;
  223. return Math.max(this.userSeekStart_, start);
  224. };
  225. /**
  226. * Sets the start time of the user-defined seek range. This is only used for
  227. * VOD content.
  228. *
  229. * @param {number} time
  230. * @export
  231. */
  232. shaka.media.PresentationTimeline.prototype.setUserSeekStart =
  233. function(time) {
  234. this.userSeekStart_ = time;
  235. };
  236. /**
  237. * Gets the presentation's current segment availability end time. Segments
  238. * starting after this time should be assumed to be unavailable.
  239. *
  240. * @return {number} The current segment availability end time, in seconds,
  241. * relative to the start of the presentation. Always returns the
  242. * presentation's duration for video-on-demand.
  243. * @export
  244. */
  245. shaka.media.PresentationTimeline.prototype.getSegmentAvailabilityEnd =
  246. function() {
  247. if (!this.isLive() && !this.isInProgress()) {
  248. return this.duration_;
  249. }
  250. return Math.min(this.getLiveEdge_(), this.duration_);
  251. };
  252. /**
  253. * Gets the seek range start time, offset by the given amount. This is used to
  254. * ensure that we don't "fall" back out of the seek window while we are
  255. * buffering.
  256. *
  257. * @param {number} offset The offset to add to the start time.
  258. * @return {number} The current seek start time, in seconds, relative to the
  259. * start of the presentation.
  260. * @export
  261. */
  262. shaka.media.PresentationTimeline.prototype.getSafeSeekRangeStart = function(
  263. offset) {
  264. // The start of the seek window, ignoring segment availability duration.
  265. const seekStart =
  266. Math.max(this.maxFirstSegmentStartTime_, this.userSeekStart_);
  267. if (this.segmentAvailabilityDuration_ == Infinity) {
  268. return seekStart;
  269. }
  270. const availabilityEnd = this.getSegmentAvailabilityEnd();
  271. const availabilityStart = availabilityEnd - this.segmentAvailabilityDuration_;
  272. // Add the offset to the availability start to ensure that we don't fall
  273. // outside the availability window while we buffer; we don't need to add the
  274. // offset to seekStart since that won't change over time.
  275. // Also see: https://github.com/google/shaka-player/issues/692
  276. const desiredStart =
  277. Math.min(availabilityStart + offset, this.getSeekRangeEnd());
  278. return Math.max(seekStart, desiredStart);
  279. };
  280. /**
  281. * Gets the seek range start time.
  282. *
  283. * @return {number}
  284. * @export
  285. */
  286. shaka.media.PresentationTimeline.prototype.getSeekRangeStart = function() {
  287. return this.getSafeSeekRangeStart(/* offset */ 0);
  288. };
  289. /**
  290. * Gets the seek range end.
  291. *
  292. * @return {number}
  293. * @export
  294. */
  295. shaka.media.PresentationTimeline.prototype.getSeekRangeEnd = function() {
  296. let useDelay = this.isLive() || this.isInProgress();
  297. let delay = useDelay ? this.presentationDelay_ : 0;
  298. return Math.max(0, this.getSegmentAvailabilityEnd() - delay);
  299. };
  300. /**
  301. * @return {number} The current presentation time in seconds.
  302. * @private
  303. */
  304. shaka.media.PresentationTimeline.prototype.getLiveEdge_ = function() {
  305. goog.asserts.assert(this.presentationStartTime_ != null,
  306. 'Cannot compute timeline live edge without start time');
  307. let now = (Date.now() + this.clockOffset_) / 1000.0;
  308. return Math.max(
  309. 0, now - this.maxSegmentDuration_ - this.presentationStartTime_);
  310. };
  311. if (goog.DEBUG) {
  312. /**
  313. * Debug only: assert that the timeline parameters make sense for the type of
  314. * presentation (VOD, IPR, live).
  315. */
  316. shaka.media.PresentationTimeline.prototype.assertIsValid = function() {
  317. if (this.isLive()) {
  318. // Implied by isLive(): infinite and dynamic.
  319. // Live streams should have a start time.
  320. goog.asserts.assert(this.presentationStartTime_ != null,
  321. 'Detected as live stream, but does not match our model of live!');
  322. } else if (this.isInProgress()) {
  323. // Implied by isInProgress(): finite and dynamic.
  324. // IPR streams should have a start time, and segments should not expire.
  325. goog.asserts.assert(this.presentationStartTime_ != null &&
  326. this.segmentAvailabilityDuration_ == Infinity,
  327. 'Detected as IPR stream, but does not match our model of IPR!');
  328. } else { // VOD
  329. // VOD segments should not expire and the presentation should be finite
  330. // and static.
  331. goog.asserts.assert(this.segmentAvailabilityDuration_ == Infinity &&
  332. this.duration_ != Infinity &&
  333. this.static_,
  334. 'Detected as VOD stream, but does not match our model of VOD!');
  335. }
  336. };
  337. }