Source: lib/dash/segment_list.js

  1. /*! @license
  2. * Shaka Player
  3. * Copyright 2016 Google LLC
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. goog.provide('shaka.dash.SegmentList');
  7. goog.require('goog.asserts');
  8. goog.require('shaka.dash.MpdUtils');
  9. goog.require('shaka.dash.SegmentBase');
  10. goog.require('shaka.log');
  11. goog.require('shaka.media.InitSegmentReference');
  12. goog.require('shaka.media.SegmentIndex');
  13. goog.require('shaka.media.SegmentReference');
  14. goog.require('shaka.util.Error');
  15. goog.require('shaka.util.Functional');
  16. goog.require('shaka.util.ManifestParserUtils');
  17. goog.require('shaka.util.XmlUtils');
  18. goog.requireType('shaka.dash.DashParser');
  19. /**
  20. * @summary A set of functions for parsing SegmentList elements.
  21. */
  22. shaka.dash.SegmentList = class {
  23. /**
  24. * Creates a new StreamInfo object.
  25. * Updates the existing SegmentIndex, if any.
  26. *
  27. * @param {shaka.dash.DashParser.Context} context
  28. * @param {!Object.<string, !shaka.extern.Stream>} streamMap
  29. * @return {shaka.dash.DashParser.StreamInfo}
  30. */
  31. static createStreamInfo(context, streamMap) {
  32. goog.asserts.assert(context.representation.segmentList,
  33. 'Should only be called with SegmentList');
  34. const SegmentList = shaka.dash.SegmentList;
  35. const initSegmentReference = shaka.dash.SegmentBase.createInitSegment(
  36. context, SegmentList.fromInheritance_);
  37. const info = SegmentList.parseSegmentListInfo_(context);
  38. SegmentList.checkSegmentListInfo_(context, info);
  39. /** @type {shaka.media.SegmentIndex} */
  40. let segmentIndex = null;
  41. let stream = null;
  42. if (context.period.id && context.representation.id) {
  43. // Only check/store the index if period and representation IDs are set.
  44. const id = context.period.id + ',' + context.representation.id;
  45. stream = streamMap[id];
  46. if (stream) {
  47. segmentIndex = stream.segmentIndex;
  48. }
  49. }
  50. const references = SegmentList.createSegmentReferences_(
  51. context.periodInfo.start, context.periodInfo.duration,
  52. info.startNumber, context.representation.baseUris, info,
  53. initSegmentReference);
  54. const isNew = !segmentIndex;
  55. if (segmentIndex) {
  56. const start = context.presentationTimeline.getSegmentAvailabilityStart();
  57. segmentIndex.mergeAndEvict(references, start);
  58. } else {
  59. segmentIndex = new shaka.media.SegmentIndex(references);
  60. }
  61. context.presentationTimeline.notifySegments(references);
  62. if (!context.dynamic || !context.periodInfo.isLastPeriod) {
  63. const periodStart = context.periodInfo.start;
  64. const periodEnd = context.periodInfo.duration ?
  65. context.periodInfo.start + context.periodInfo.duration : Infinity;
  66. segmentIndex.fit(periodStart, periodEnd, isNew);
  67. }
  68. if (stream) {
  69. stream.segmentIndex = segmentIndex;
  70. }
  71. return {
  72. generateSegmentIndex: () => {
  73. if (!segmentIndex || segmentIndex.isEmpty()) {
  74. segmentIndex.merge(references);
  75. }
  76. return Promise.resolve(segmentIndex);
  77. },
  78. };
  79. }
  80. /**
  81. * @param {?shaka.dash.DashParser.InheritanceFrame} frame
  82. * @return {Element}
  83. * @private
  84. */
  85. static fromInheritance_(frame) {
  86. return frame.segmentList;
  87. }
  88. /**
  89. * Parses the SegmentList items to create an info object.
  90. *
  91. * @param {shaka.dash.DashParser.Context} context
  92. * @return {shaka.dash.SegmentList.SegmentListInfo}
  93. * @private
  94. */
  95. static parseSegmentListInfo_(context) {
  96. const SegmentList = shaka.dash.SegmentList;
  97. const MpdUtils = shaka.dash.MpdUtils;
  98. const mediaSegments = SegmentList.parseMediaSegments_(context);
  99. const segmentInfo =
  100. MpdUtils.parseSegmentInfo(context, SegmentList.fromInheritance_);
  101. let startNumber = segmentInfo.startNumber;
  102. if (startNumber == 0) {
  103. shaka.log.warning('SegmentList@startNumber must be > 0');
  104. startNumber = 1;
  105. }
  106. let startTime = 0;
  107. if (segmentInfo.segmentDuration) {
  108. // See DASH sec. 5.3.9.5.3
  109. // Don't use presentationTimeOffset for @duration.
  110. startTime = segmentInfo.segmentDuration * (startNumber - 1);
  111. } else if (segmentInfo.timeline && segmentInfo.timeline.length > 0) {
  112. // The presentationTimeOffset was considered in timeline creation.
  113. startTime = segmentInfo.timeline[0].start;
  114. }
  115. return {
  116. segmentDuration: segmentInfo.segmentDuration,
  117. startTime: startTime,
  118. startNumber: startNumber,
  119. scaledPresentationTimeOffset: segmentInfo.scaledPresentationTimeOffset,
  120. timeline: segmentInfo.timeline,
  121. mediaSegments: mediaSegments,
  122. };
  123. }
  124. /**
  125. * Checks whether a SegmentListInfo object is valid.
  126. *
  127. * @param {shaka.dash.DashParser.Context} context
  128. * @param {shaka.dash.SegmentList.SegmentListInfo} info
  129. * @private
  130. */
  131. static checkSegmentListInfo_(context, info) {
  132. if (!info.segmentDuration && !info.timeline &&
  133. info.mediaSegments.length > 1) {
  134. shaka.log.warning(
  135. 'SegmentList does not contain sufficient segment information:',
  136. 'the SegmentList specifies multiple segments,',
  137. 'but does not specify a segment duration or timeline.',
  138. context.representation);
  139. throw new shaka.util.Error(
  140. shaka.util.Error.Severity.CRITICAL,
  141. shaka.util.Error.Category.MANIFEST,
  142. shaka.util.Error.Code.DASH_NO_SEGMENT_INFO);
  143. }
  144. if (!info.segmentDuration && !context.periodInfo.duration &&
  145. !info.timeline && info.mediaSegments.length == 1) {
  146. shaka.log.warning(
  147. 'SegmentList does not contain sufficient segment information:',
  148. 'the SegmentList specifies one segment,',
  149. 'but does not specify a segment duration, period duration,',
  150. 'or timeline.',
  151. context.representation);
  152. throw new shaka.util.Error(
  153. shaka.util.Error.Severity.CRITICAL,
  154. shaka.util.Error.Category.MANIFEST,
  155. shaka.util.Error.Code.DASH_NO_SEGMENT_INFO);
  156. }
  157. if (info.timeline && info.timeline.length == 0) {
  158. shaka.log.warning(
  159. 'SegmentList does not contain sufficient segment information:',
  160. 'the SegmentList has an empty timeline.',
  161. context.representation);
  162. throw new shaka.util.Error(
  163. shaka.util.Error.Severity.CRITICAL,
  164. shaka.util.Error.Category.MANIFEST,
  165. shaka.util.Error.Code.DASH_NO_SEGMENT_INFO);
  166. }
  167. }
  168. /**
  169. * Creates an array of segment references for the given data.
  170. *
  171. * @param {number} periodStart in seconds.
  172. * @param {?number} periodDuration in seconds.
  173. * @param {number} startNumber
  174. * @param {!Array.<string>} baseUris
  175. * @param {shaka.dash.SegmentList.SegmentListInfo} info
  176. * @param {shaka.media.InitSegmentReference} initSegmentReference
  177. * @return {!Array.<!shaka.media.SegmentReference>}
  178. * @private
  179. */
  180. static createSegmentReferences_(
  181. periodStart, periodDuration, startNumber, baseUris, info,
  182. initSegmentReference) {
  183. const ManifestParserUtils = shaka.util.ManifestParserUtils;
  184. let max = info.mediaSegments.length;
  185. if (info.timeline && info.timeline.length != info.mediaSegments.length) {
  186. max = Math.min(info.timeline.length, info.mediaSegments.length);
  187. shaka.log.warning(
  188. 'The number of items in the segment timeline and the number of ',
  189. 'segment URLs do not match, truncating', info.mediaSegments.length,
  190. 'to', max);
  191. }
  192. const timestampOffset = periodStart - info.scaledPresentationTimeOffset;
  193. const appendWindowStart = periodStart;
  194. const appendWindowEnd = periodDuration ?
  195. periodStart + periodDuration : Infinity;
  196. /** @type {!Array.<!shaka.media.SegmentReference>} */
  197. const references = [];
  198. let prevEndTime = info.startTime;
  199. for (let i = 0; i < max; i++) {
  200. const segment = info.mediaSegments[i];
  201. const mediaUri = ManifestParserUtils.resolveUris(
  202. baseUris, [segment.mediaUri]);
  203. const startTime = prevEndTime;
  204. let endTime;
  205. if (info.segmentDuration != null) {
  206. endTime = startTime + info.segmentDuration;
  207. } else if (info.timeline) {
  208. // Ignore the timepoint start since they are continuous.
  209. endTime = info.timeline[i].end;
  210. } else {
  211. // If segmentDuration and timeline are null then there must
  212. // be exactly one segment.
  213. goog.asserts.assert(
  214. info.mediaSegments.length == 1 && periodDuration,
  215. 'There should be exactly one segment with a Period duration.');
  216. endTime = startTime + periodDuration;
  217. }
  218. const getUris = () => mediaUri;
  219. references.push(
  220. new shaka.media.SegmentReference(
  221. periodStart + startTime,
  222. periodStart + endTime,
  223. getUris,
  224. segment.start,
  225. segment.end,
  226. initSegmentReference,
  227. timestampOffset,
  228. appendWindowStart, appendWindowEnd));
  229. prevEndTime = endTime;
  230. }
  231. return references;
  232. }
  233. /**
  234. * Parses the media URIs from the context.
  235. *
  236. * @param {shaka.dash.DashParser.Context} context
  237. * @return {!Array.<shaka.dash.SegmentList.MediaSegment>}
  238. * @private
  239. */
  240. static parseMediaSegments_(context) {
  241. const Functional = shaka.util.Functional;
  242. /** @type {!Array.<!Element>} */
  243. const segmentLists = [
  244. context.representation.segmentList,
  245. context.adaptationSet.segmentList,
  246. context.period.segmentList,
  247. ].filter(Functional.isNotNull);
  248. const XmlUtils = shaka.util.XmlUtils;
  249. // Search each SegmentList for one with at least one SegmentURL element,
  250. // select the first one, and convert each SegmentURL element to a tuple.
  251. return segmentLists
  252. .map((node) => { return XmlUtils.findChildren(node, 'SegmentURL'); })
  253. .reduce((all, part) => { return all.length > 0 ? all : part; })
  254. .map((urlNode) => {
  255. if (urlNode.getAttribute('indexRange') &&
  256. !context.indexRangeWarningGiven) {
  257. context.indexRangeWarningGiven = true;
  258. shaka.log.warning(
  259. 'We do not support the SegmentURL@indexRange attribute on ' +
  260. 'SegmentList. We only use the SegmentList@duration ' +
  261. 'attribute or SegmentTimeline, which must be accurate.');
  262. }
  263. const uri = urlNode.getAttribute('media');
  264. const range = XmlUtils.parseAttr(
  265. urlNode, 'mediaRange', XmlUtils.parseRange,
  266. {start: 0, end: null});
  267. return {mediaUri: uri, start: range.start, end: range.end};
  268. });
  269. }
  270. };
  271. /**
  272. * @typedef {{
  273. * mediaUri: string,
  274. * start: number,
  275. * end: ?number
  276. * }}
  277. *
  278. * @property {string} mediaUri
  279. * The URI of the segment.
  280. * @property {number} start
  281. * The start byte of the segment.
  282. * @property {?number} end
  283. * The end byte of the segment, or null.
  284. */
  285. shaka.dash.SegmentList.MediaSegment;
  286. /**
  287. * @typedef {{
  288. * segmentDuration: ?number,
  289. * startTime: number,
  290. * startNumber: number,
  291. * scaledPresentationTimeOffset: number,
  292. * timeline: Array.<shaka.dash.MpdUtils.TimeRange>,
  293. * mediaSegments: !Array.<shaka.dash.SegmentList.MediaSegment>
  294. * }}
  295. * @private
  296. *
  297. * @description
  298. * Contains information about a SegmentList.
  299. *
  300. * @property {?number} segmentDuration
  301. * The duration of the segments, if given.
  302. * @property {number} startTime
  303. * The start time of the first segment, in seconds.
  304. * @property {number} startNumber
  305. * The start number of the segments; 1 or greater.
  306. * @property {number} scaledPresentationTimeOffset
  307. * The scaledPresentationTimeOffset of the representation, in seconds.
  308. * @property {Array.<shaka.dash.MpdUtils.TimeRange>} timeline
  309. * The timeline of the representation, if given. Times in seconds.
  310. * @property {!Array.<shaka.dash.SegmentList.MediaSegment>} mediaSegments
  311. * The URI and byte-ranges of the media segments.
  312. */
  313. shaka.dash.SegmentList.SegmentListInfo;