Source: lib/media/presentation_timeline.js

  1. /*! @license
  2. * Shaka Player
  3. * Copyright 2016 Google LLC
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. goog.provide('shaka.media.PresentationTimeline');
  7. goog.require('goog.asserts');
  8. goog.require('shaka.log');
  9. goog.require('shaka.media.SegmentReference');
  10. /**
  11. * PresentationTimeline.
  12. * @export
  13. */
  14. shaka.media.PresentationTimeline = class {
  15. /**
  16. * @param {?number} presentationStartTime The wall-clock time, in seconds,
  17. * when the presentation started or will start. Only required for live.
  18. * @param {number} presentationDelay The delay to give the presentation, in
  19. * seconds. Only required for live.
  20. * @param {boolean=} autoCorrectDrift Whether to account for drift when
  21. * determining the availability window.
  22. *
  23. * @see {shaka.extern.Manifest}
  24. * @see {@tutorial architecture}
  25. */
  26. constructor(presentationStartTime, presentationDelay,
  27. autoCorrectDrift = true) {
  28. /** @private {?number} */
  29. this.presentationStartTime_ = presentationStartTime;
  30. /** @private {number} */
  31. this.presentationDelay_ = presentationDelay;
  32. /** @private {number} */
  33. this.duration_ = Infinity;
  34. /** @private {number} */
  35. this.segmentAvailabilityDuration_ = Infinity;
  36. /**
  37. * The maximum segment duration (in seconds). Can be based on explicitly-
  38. * known segments or on signalling in the manifest.
  39. *
  40. * @private {number}
  41. */
  42. this.maxSegmentDuration_ = 1;
  43. /**
  44. * The minimum segment start time (in seconds, in the presentation timeline)
  45. * for segments we explicitly know about.
  46. *
  47. * This is null if we have no explicit descriptions of segments, such as in
  48. * DASH when using SegmentTemplate w/ duration.
  49. *
  50. * @private {?number}
  51. */
  52. this.minSegmentStartTime_ = null;
  53. /**
  54. * The maximum segment end time (in seconds, in the presentation timeline)
  55. * for segments we explicitly know about.
  56. *
  57. * This is null if we have no explicit descriptions of segments, such as in
  58. * DASH when using SegmentTemplate w/ duration. When this is non-null, the
  59. * presentation start time is calculated from the segment end times.
  60. *
  61. * @private {?number}
  62. */
  63. this.maxSegmentEndTime_ = null;
  64. /** @private {number} */
  65. this.clockOffset_ = 0;
  66. /** @private {boolean} */
  67. this.static_ = true;
  68. /** @private {boolean} */
  69. this.isLive2VodTransition_ = false;
  70. /** @private {number} */
  71. this.userSeekStart_ = 0;
  72. /** @private {boolean} */
  73. this.autoCorrectDrift_ = autoCorrectDrift;
  74. /**
  75. * For low latency Dash, availabilityTimeOffset indicates a segment is
  76. * available for download earlier than its availability start time.
  77. * This field is the minimum availabilityTimeOffset value among the
  78. * segments. We reduce the distance from live edge by this value.
  79. *
  80. * @private {number}
  81. */
  82. this.availabilityTimeOffset_ = 0;
  83. /** @private {boolean} */
  84. this.startTimeLocked_ = false;
  85. /** @private {?number} */
  86. this.initialProgramDateTime_ = presentationStartTime;
  87. }
  88. /**
  89. * @return {number} The presentation's duration in seconds.
  90. * Infinity indicates that the presentation continues indefinitely.
  91. * @export
  92. */
  93. getDuration() {
  94. return this.duration_;
  95. }
  96. /**
  97. * @return {number} The presentation's max segment duration in seconds.
  98. * @export
  99. */
  100. getMaxSegmentDuration() {
  101. return this.maxSegmentDuration_;
  102. }
  103. /**
  104. * Sets the presentation's start time.
  105. *
  106. * @param {number} presentationStartTime The wall-clock time, in seconds,
  107. * when the presentation started or will start. Only required for live.
  108. * @export
  109. */
  110. setPresentationStartTime(presentationStartTime) {
  111. goog.asserts.assert(presentationStartTime >= 0,
  112. 'presentationStartTime must be >= 0');
  113. this.presentationStartTime_ = presentationStartTime;
  114. }
  115. /**
  116. * Sets the presentation's duration.
  117. *
  118. * @param {number} duration The presentation's duration in seconds.
  119. * Infinity indicates that the presentation continues indefinitely.
  120. * @export
  121. */
  122. setDuration(duration) {
  123. goog.asserts.assert(duration > 0, 'duration must be > 0');
  124. this.duration_ = duration;
  125. }
  126. /**
  127. * @return {?number} The presentation's start time in seconds.
  128. * @export
  129. */
  130. getPresentationStartTime() {
  131. return this.presentationStartTime_;
  132. }
  133. /**
  134. * Sets the clock offset, which is the difference between the client's clock
  135. * and the server's clock, in milliseconds (i.e., serverTime = Date.now() +
  136. * clockOffset).
  137. *
  138. * @param {number} offset The clock offset, in ms.
  139. * @export
  140. */
  141. setClockOffset(offset) {
  142. this.clockOffset_ = offset;
  143. }
  144. /**
  145. * Sets the presentation's static flag.
  146. *
  147. * @param {boolean} isStatic If true, the presentation is static, meaning all
  148. * segments are available at once.
  149. * @export
  150. */
  151. setStatic(isStatic) {
  152. // NOTE: the argument name is not "static" because that's a keyword in ES6
  153. if (isStatic && !this.static_) {
  154. this.isLive2VodTransition_ = true;
  155. }
  156. this.static_ = isStatic;
  157. }
  158. /**
  159. * Sets the presentation's segment availability duration. The segment
  160. * availability duration should only be set for live.
  161. *
  162. * @param {number} segmentAvailabilityDuration The presentation's new segment
  163. * availability duration in seconds.
  164. * @export
  165. */
  166. setSegmentAvailabilityDuration(segmentAvailabilityDuration) {
  167. goog.asserts.assert(segmentAvailabilityDuration >= 0,
  168. 'segmentAvailabilityDuration must be >= 0');
  169. this.segmentAvailabilityDuration_ = segmentAvailabilityDuration;
  170. }
  171. /**
  172. * Gets the presentation's segment availability duration.
  173. *
  174. * @return {number}
  175. * @export
  176. */
  177. getSegmentAvailabilityDuration() {
  178. return this.segmentAvailabilityDuration_;
  179. }
  180. /**
  181. * Sets the presentation delay in seconds.
  182. *
  183. * @param {number} delay
  184. * @export
  185. */
  186. setDelay(delay) {
  187. // NOTE: This is no longer used internally, but is exported.
  188. // So we cannot remove it without deprecating it and waiting one release
  189. // cycle, or else we risk breaking custom manifest parsers.
  190. goog.asserts.assert(delay >= 0, 'delay must be >= 0');
  191. this.presentationDelay_ = delay;
  192. }
  193. /**
  194. * Gets the presentation delay in seconds.
  195. * @return {number}
  196. * @export
  197. */
  198. getDelay() {
  199. return this.presentationDelay_;
  200. }
  201. /**
  202. * Gives PresentationTimeline a Stream's timeline so it can size and position
  203. * the segment availability window, and account for missing segment
  204. * information.
  205. *
  206. * @param {!Array<shaka.media.PresentationTimeline.TimeRange>} timeline
  207. * @param {number} startOffset
  208. * @export
  209. */
  210. notifyTimeRange(timeline, startOffset) {
  211. if (timeline.length == 0) {
  212. return;
  213. }
  214. // Date.now() is in milliseconds, from which we compute "now" in seconds.
  215. const now = (Date.now() + this.clockOffset_) / 1000.0;
  216. // Exclude time ranges that are in the "future".
  217. const timelineForStart = timeline.filter((timeRange) =>
  218. timeRange.start + startOffset < now);
  219. if (timelineForStart.length == 0) {
  220. return;
  221. }
  222. const firstStartTime = timelineForStart[0].start + startOffset;
  223. const lastEndTime = timelineForStart[timelineForStart.length - 1].end +
  224. startOffset;
  225. this.notifyMinSegmentStartTime(firstStartTime);
  226. this.maxSegmentDuration_ = timelineForStart.reduce(
  227. (max, r) => { return Math.max(max, r.end - r.start); },
  228. this.maxSegmentDuration_);
  229. this.maxSegmentEndTime_ =
  230. Math.max(this.maxSegmentEndTime_, lastEndTime);
  231. if (this.presentationStartTime_ != null && this.autoCorrectDrift_ &&
  232. !this.startTimeLocked_) {
  233. // Since we have explicit segment end times, calculate a presentation
  234. // start based on them. This start time accounts for drift.
  235. this.presentationStartTime_ =
  236. now - this.maxSegmentEndTime_ - this.maxSegmentDuration_;
  237. }
  238. shaka.log.v1('notifySegments:',
  239. 'maxSegmentDuration=' + this.maxSegmentDuration_);
  240. }
  241. /**
  242. * Gives PresentationTimeline an array of segments so it can size and position
  243. * the segment availability window, and account for missing segment
  244. * information. These segments do not necessarily need to all be from the
  245. * same stream.
  246. *
  247. * @param {!Array<!shaka.media.SegmentReference>} references
  248. * @export
  249. */
  250. notifySegments(references) {
  251. if (references.length == 0) {
  252. return;
  253. }
  254. let firstReferenceStartTime = references[0].startTime;
  255. let lastReferenceEndTime = references[0].endTime;
  256. // Date.now() is in milliseconds, from which we compute "now" in seconds.
  257. const now = (Date.now() + this.clockOffset_) / 1000.0;
  258. for (const reference of references) {
  259. // Exclude segments that are in the "future".
  260. if (now < reference.startTime) {
  261. continue;
  262. }
  263. firstReferenceStartTime = Math.min(
  264. firstReferenceStartTime, reference.startTime);
  265. lastReferenceEndTime = Math.max(lastReferenceEndTime, reference.endTime);
  266. this.maxSegmentDuration_ = Math.max(
  267. this.maxSegmentDuration_, reference.endTime - reference.startTime);
  268. }
  269. this.notifyMinSegmentStartTime(firstReferenceStartTime);
  270. this.maxSegmentEndTime_ =
  271. Math.max(this.maxSegmentEndTime_, lastReferenceEndTime);
  272. if (this.presentationStartTime_ != null && this.autoCorrectDrift_ &&
  273. !this.startTimeLocked_) {
  274. // Since we have explicit segment end times, calculate a presentation
  275. // start based on them. This start time accounts for drift.
  276. this.presentationStartTime_ =
  277. now - this.maxSegmentEndTime_ - this.maxSegmentDuration_;
  278. }
  279. shaka.log.v1('notifySegments:',
  280. 'maxSegmentDuration=' + this.maxSegmentDuration_);
  281. }
  282. /**
  283. * Gives PresentationTimeline an startTime and endTime of the period.
  284. * This should be only set for Dash.
  285. *
  286. * @param {number} startTime
  287. * @param {number} endTime
  288. * @export
  289. */
  290. notifyPeriodDuration(startTime, endTime) {
  291. this.notifyMinSegmentStartTime(startTime);
  292. if (endTime != Infinity && !this.isLive()) {
  293. this.maxSegmentEndTime_ = Math.max(this.maxSegmentEndTime_, endTime);
  294. }
  295. }
  296. /**
  297. * Gets the end time of the last available segment.
  298. *
  299. * @return {?number}
  300. * @export
  301. */
  302. getMaxSegmentEndTime() {
  303. return this.maxSegmentEndTime_;
  304. }
  305. /**
  306. * Lock the presentation timeline's start time. After this is called, no
  307. * further adjustments to presentationStartTime_ will be permitted.
  308. *
  309. * This should be called after all Periods have been parsed, and all calls to
  310. * notifySegments() from the initial manifest parse have been made.
  311. *
  312. * Without this, we can get assertion failures in SegmentIndex for certain
  313. * DAI content. If DAI adds ad segments to the manifest faster than
  314. * real-time, adjustments to presentationStartTime_ can cause availability
  315. * windows to jump around on updates.
  316. *
  317. * @export
  318. */
  319. lockStartTime() {
  320. this.startTimeLocked_ = true;
  321. }
  322. /**
  323. * Returns if the presentation timeline's start time is locked.
  324. *
  325. * @return {boolean}
  326. * @export
  327. */
  328. isStartTimeLocked() {
  329. return this.startTimeLocked_;
  330. }
  331. /**
  332. * Sets the initial program date time.
  333. *
  334. * @param {number} initialProgramDateTime
  335. * @export
  336. */
  337. setInitialProgramDateTime(initialProgramDateTime) {
  338. this.initialProgramDateTime_ = initialProgramDateTime;
  339. }
  340. /**
  341. * @return {?number} The initial program date time in seconds.
  342. * @export
  343. */
  344. getInitialProgramDateTime() {
  345. return this.initialProgramDateTime_;
  346. }
  347. /**
  348. * Gives PresentationTimeline a Stream's minimum segment start time.
  349. *
  350. * @param {number} startTime
  351. * @export
  352. */
  353. notifyMinSegmentStartTime(startTime) {
  354. if (this.minSegmentStartTime_ == null) {
  355. // No data yet, and Math.min(null, startTime) is always 0. So just store
  356. // startTime.
  357. this.minSegmentStartTime_ = startTime;
  358. } else if (!this.isLive2VodTransition_) {
  359. this.minSegmentStartTime_ =
  360. Math.min(this.minSegmentStartTime_, startTime);
  361. }
  362. }
  363. /**
  364. * Gives PresentationTimeline a Stream's maximum segment duration so it can
  365. * size and position the segment availability window. This function should be
  366. * called once for each Stream (no more, no less), but does not have to be
  367. * called if notifySegments() is called instead for a particular stream.
  368. *
  369. * @param {number} maxSegmentDuration The maximum segment duration for a
  370. * particular stream.
  371. * @export
  372. */
  373. notifyMaxSegmentDuration(maxSegmentDuration) {
  374. this.maxSegmentDuration_ = Math.max(
  375. this.maxSegmentDuration_, maxSegmentDuration);
  376. shaka.log.v1('notifyNewSegmentDuration:',
  377. 'maxSegmentDuration=' + this.maxSegmentDuration_);
  378. }
  379. /**
  380. * Offsets the segment times by the given amount.
  381. *
  382. * @param {number} offset The number of seconds to offset by. A positive
  383. * number adjusts the segment times forward.
  384. * @export
  385. */
  386. offset(offset) {
  387. if (this.minSegmentStartTime_ != null) {
  388. this.minSegmentStartTime_ += offset;
  389. }
  390. if (this.maxSegmentEndTime_ != null) {
  391. this.maxSegmentEndTime_ += offset;
  392. }
  393. }
  394. /**
  395. * @return {boolean} True if the presentation is live; otherwise, return
  396. * false.
  397. * @export
  398. */
  399. isLive() {
  400. return this.duration_ == Infinity &&
  401. !this.static_;
  402. }
  403. /**
  404. * @return {boolean} True if the presentation is in progress (meaning not
  405. * live, but also not completely available); otherwise, return false.
  406. * @export
  407. */
  408. isInProgress() {
  409. return this.duration_ != Infinity &&
  410. !this.static_;
  411. }
  412. /**
  413. * Gets the presentation's current segment availability start time. Segments
  414. * ending at or before this time should be assumed to be unavailable.
  415. *
  416. * @return {number} The current segment availability start time, in seconds,
  417. * relative to the start of the presentation.
  418. * @export
  419. */
  420. getSegmentAvailabilityStart() {
  421. goog.asserts.assert(this.segmentAvailabilityDuration_ >= 0,
  422. 'The availability duration should be positive');
  423. const end = this.getSegmentAvailabilityEnd();
  424. const start = end - this.segmentAvailabilityDuration_;
  425. return Math.max(this.userSeekStart_, start);
  426. }
  427. /**
  428. * Sets the start time of the user-defined seek range. This is only used for
  429. * VOD content.
  430. *
  431. * @param {number} time
  432. * @export
  433. */
  434. setUserSeekStart(time) {
  435. this.userSeekStart_ = time;
  436. }
  437. /**
  438. * Gets the presentation's current segment availability end time. Segments
  439. * starting after this time should be assumed to be unavailable.
  440. *
  441. * @return {number} The current segment availability end time, in seconds,
  442. * relative to the start of the presentation. For VOD, the availability
  443. * end time is the content's duration. If the Player's playRangeEnd
  444. * configuration is used, this can override the duration.
  445. * @export
  446. */
  447. getSegmentAvailabilityEnd() {
  448. if (!this.isLive() && !this.isInProgress()) {
  449. // It's a static manifest (can also be a dynamic->static conversion)
  450. if (this.maxSegmentEndTime_) {
  451. // If we know segment times, use the min of that and duration.
  452. // Note that the playRangeEnd configuration changes this.duration_.
  453. // See https://github.com/shaka-project/shaka-player/issues/4026
  454. return Math.min(this.maxSegmentEndTime_, this.duration_);
  455. } else {
  456. // If we don't have segment times, use duration.
  457. return this.duration_;
  458. }
  459. }
  460. // Can be either live or "in-progress recording" (live with known duration)
  461. return Math.min(this.getLiveEdge_() + this.availabilityTimeOffset_,
  462. this.duration_);
  463. }
  464. /**
  465. * Gets the seek range start time, offset by the given amount. This is used
  466. * to ensure that we don't "fall" back out of the seek window while we are
  467. * buffering.
  468. *
  469. * @param {number} offset The offset to add to the start time for live
  470. * streams.
  471. * @return {number} The current seek start time, in seconds, relative to the
  472. * start of the presentation.
  473. * @export
  474. */
  475. getSafeSeekRangeStart(offset) {
  476. // The earliest known segment time, ignoring segment availability duration.
  477. const earliestSegmentTime =
  478. Math.max(this.minSegmentStartTime_, this.userSeekStart_);
  479. // For VOD, the offset and end time are ignored, and we just return the
  480. // earliest segment time. All segments are "safe" in VOD. However, we
  481. // should round up to the nearest millisecond to avoid issues like
  482. // https://github.com/shaka-project/shaka-player/issues/2831, in which we
  483. // tried to seek repeatedly to catch up to the seek range, and never
  484. // actually "arrived" within it. The video's currentTime is not as
  485. // accurate as the JS number representing the earliest segment time for
  486. // some content.
  487. if (this.segmentAvailabilityDuration_ == Infinity) {
  488. return Math.ceil(earliestSegmentTime * 1e3) / 1e3;
  489. }
  490. // AKA the live edge for live streams.
  491. const availabilityEnd = this.getSegmentAvailabilityEnd();
  492. // The ideal availability start, not considering known segments.
  493. const availabilityStart =
  494. availabilityEnd - this.segmentAvailabilityDuration_;
  495. // Add the offset to the availability start to ensure that we don't fall
  496. // outside the availability window while we buffer; we don't need to add the
  497. // offset to earliestSegmentTime since that won't change over time.
  498. // Also see: https://github.com/shaka-project/shaka-player/issues/692
  499. const desiredStart =
  500. Math.min(availabilityStart + offset, this.getSeekRangeEnd());
  501. return Math.max(earliestSegmentTime, desiredStart);
  502. }
  503. /**
  504. * Gets the seek range start time.
  505. *
  506. * @return {number}
  507. * @export
  508. */
  509. getSeekRangeStart() {
  510. return this.getSafeSeekRangeStart(/* offset= */ 0);
  511. }
  512. /**
  513. * Gets the seek range end.
  514. *
  515. * @return {number}
  516. * @export
  517. */
  518. getSeekRangeEnd() {
  519. const useDelay = this.isLive() || this.isInProgress();
  520. const delay = useDelay ? this.presentationDelay_ : 0;
  521. return Math.max(0, this.getSegmentAvailabilityEnd() - delay);
  522. }
  523. /**
  524. * True if the presentation start time is being used to calculate the live
  525. * edge.
  526. * Using the presentation start time means that the stream may be subject to
  527. * encoder drift. At runtime, we will avoid using the presentation start time
  528. * whenever possible.
  529. *
  530. * @return {boolean}
  531. * @export
  532. */
  533. usingPresentationStartTime() {
  534. // If it's VOD, IPR, or an HLS "event", we are not using the presentation
  535. // start time.
  536. if (this.presentationStartTime_ == null) {
  537. return false;
  538. }
  539. // If we have explicit segment times, we're not using the presentation
  540. // start time.
  541. if (this.maxSegmentEndTime_ != null && this.autoCorrectDrift_) {
  542. return false;
  543. }
  544. return true;
  545. }
  546. /**
  547. * @return {number} The current presentation time in seconds.
  548. * @private
  549. */
  550. getLiveEdge_() {
  551. goog.asserts.assert(this.presentationStartTime_ != null,
  552. 'Cannot compute timeline live edge without start time');
  553. // Date.now() is in milliseconds, from which we compute "now" in seconds.
  554. const now = (Date.now() + this.clockOffset_) / 1000.0;
  555. return Math.max(
  556. 0, now - this.maxSegmentDuration_ - this.presentationStartTime_);
  557. }
  558. /**
  559. * Sets the presentation's segment availability time offset. This should be
  560. * only set for Low Latency Dash.
  561. * The segments are available earlier for download than the availability start
  562. * time, so we can move closer to the live edge.
  563. *
  564. * @param {number} offset
  565. * @export
  566. */
  567. setAvailabilityTimeOffset(offset) {
  568. this.availabilityTimeOffset_ = offset;
  569. }
  570. /**
  571. * Gets the presentation's segment availability time offset. This should be
  572. * only configured for Low Latency Dash.
  573. *
  574. * @return {number} availabilityTimeOffset parameter
  575. * @export
  576. */
  577. getAvailabilityTimeOffset() {
  578. return this.availabilityTimeOffset_;
  579. }
  580. /**
  581. * Debug only: assert that the timeline parameters make sense for the type
  582. * of presentation (VOD, IPR, live).
  583. */
  584. assertIsValid() {
  585. if (goog.DEBUG) {
  586. if (this.isLive()) {
  587. // Implied by isLive(): infinite and dynamic.
  588. // Live streams should have a start time.
  589. goog.asserts.assert(this.presentationStartTime_ != null,
  590. 'Detected as live stream, but does not match our model of live!');
  591. } else if (this.isInProgress()) {
  592. // Implied by isInProgress(): finite and dynamic.
  593. // IPR streams should have a start time, and segments should not expire.
  594. goog.asserts.assert(this.presentationStartTime_ != null &&
  595. this.segmentAvailabilityDuration_ == Infinity,
  596. 'Detected as IPR stream, but does not match our model of IPR!');
  597. } else { // VOD
  598. // VOD segments should not expire and the presentation should be finite
  599. // and static.
  600. goog.asserts.assert(this.segmentAvailabilityDuration_ == Infinity &&
  601. this.duration_ != Infinity &&
  602. this.static_,
  603. 'Detected as VOD stream, but does not match our model of VOD!');
  604. }
  605. }
  606. }
  607. };
  608. /**
  609. * @typedef {{
  610. * start: number,
  611. * unscaledStart: number,
  612. * end: number,
  613. * partialSegments: number,
  614. * segmentPosition: number
  615. * }}
  616. *
  617. * @description
  618. * Defines a time range of a media segment. Times are in seconds.
  619. *
  620. * @property {number} start
  621. * The start time of the range.
  622. * @property {number} unscaledStart
  623. * The start time of the range in representation timescale units.
  624. * @property {number} end
  625. * The end time (exclusive) of the range.
  626. * @property {number} partialSegments
  627. * The number of partial segments
  628. * @property {number} segmentPosition
  629. * The segment position of the timeline entry as it appears in the manifest
  630. *
  631. * @export
  632. */
  633. shaka.media.PresentationTimeline.TimeRange;