Home Reference Source

src/loader/fragment.ts

  1. import { buildAbsoluteURL } from 'url-toolkit';
  2. import { LevelKey } from './level-key';
  3. import { LoadStats } from './load-stats';
  4. import { AttrList } from '../utils/attr-list';
  5. import type {
  6. FragmentLoaderContext,
  7. KeyLoaderContext,
  8. Loader,
  9. PlaylistLevelType,
  10. } from '../types/loader';
  11. import type { KeySystemFormats } from '../utils/mediakeys-helper';
  12.  
  13. export enum ElementaryStreamTypes {
  14. AUDIO = 'audio',
  15. VIDEO = 'video',
  16. AUDIOVIDEO = 'audiovideo',
  17. }
  18.  
  19. export interface ElementaryStreamInfo {
  20. startPTS: number;
  21. endPTS: number;
  22. startDTS: number;
  23. endDTS: number;
  24. partial?: boolean;
  25. }
  26.  
  27. export type ElementaryStreams = Record<
  28. ElementaryStreamTypes,
  29. ElementaryStreamInfo | null
  30. >;
  31.  
  32. export class BaseSegment {
  33. private _byteRange: number[] | null = null;
  34. private _url: string | null = null;
  35.  
  36. // baseurl is the URL to the playlist
  37. public readonly baseurl: string;
  38. // relurl is the portion of the URL that comes from inside the playlist.
  39. public relurl?: string;
  40. // Holds the types of data this fragment supports
  41. public elementaryStreams: ElementaryStreams = {
  42. [ElementaryStreamTypes.AUDIO]: null,
  43. [ElementaryStreamTypes.VIDEO]: null,
  44. [ElementaryStreamTypes.AUDIOVIDEO]: null,
  45. };
  46.  
  47. constructor(baseurl: string) {
  48. this.baseurl = baseurl;
  49. }
  50.  
  51. // setByteRange converts a EXT-X-BYTERANGE attribute into a two element array
  52. setByteRange(value: string, previous?: BaseSegment) {
  53. const params = value.split('@', 2);
  54. const byteRange: number[] = [];
  55. if (params.length === 1) {
  56. byteRange[0] = previous ? previous.byteRangeEndOffset : 0;
  57. } else {
  58. byteRange[0] = parseInt(params[1]);
  59. }
  60. byteRange[1] = parseInt(params[0]) + byteRange[0];
  61. this._byteRange = byteRange;
  62. }
  63.  
  64. get byteRange(): number[] {
  65. if (!this._byteRange) {
  66. return [];
  67. }
  68.  
  69. return this._byteRange;
  70. }
  71.  
  72. get byteRangeStartOffset(): number {
  73. return this.byteRange[0];
  74. }
  75.  
  76. get byteRangeEndOffset(): number {
  77. return this.byteRange[1];
  78. }
  79.  
  80. get url(): string {
  81. if (!this._url && this.baseurl && this.relurl) {
  82. this._url = buildAbsoluteURL(this.baseurl, this.relurl, {
  83. alwaysNormalize: true,
  84. });
  85. }
  86. return this._url || '';
  87. }
  88.  
  89. set url(value: string) {
  90. this._url = value;
  91. }
  92. }
  93.  
  94. export class Fragment extends BaseSegment {
  95. private _decryptdata: LevelKey | null = null;
  96.  
  97. public rawProgramDateTime: string | null = null;
  98. public programDateTime: number | null = null;
  99. public tagList: Array<string[]> = [];
  100.  
  101. // EXTINF has to be present for a m3u8 to be considered valid
  102. public duration: number = 0;
  103. // sn notates the sequence number for a segment, and if set to a string can be 'initSegment'
  104. public sn: number | 'initSegment' = 0;
  105. // levelkeys are the EXT-X-KEY tags that apply to this segment for decryption
  106. // core difference from the private field _decryptdata is the lack of the initialized IV
  107. // _decryptdata will set the IV for this segment based on the segment number in the fragment
  108. public levelkeys?: { [key: string]: LevelKey };
  109. // A string representing the fragment type
  110. public readonly type: PlaylistLevelType;
  111. // A reference to the loader. Set while the fragment is loading, and removed afterwards. Used to abort fragment loading
  112. public loader: Loader<FragmentLoaderContext> | null = null;
  113. // A reference to the key loader. Set while the key is loading, and removed afterwards. Used to abort key loading
  114. public keyLoader: Loader<KeyLoaderContext> | null = null;
  115. // The level/track index to which the fragment belongs
  116. public level: number = -1;
  117. // The continuity counter of the fragment
  118. public cc: number = 0;
  119. // The starting Presentation Time Stamp (PTS) of the fragment. Set after transmux complete.
  120. public startPTS?: number;
  121. // The ending Presentation Time Stamp (PTS) of the fragment. Set after transmux complete.
  122. public endPTS?: number;
  123. // The latest Presentation Time Stamp (PTS) appended to the buffer.
  124. public appendedPTS?: number;
  125. // The starting Decode Time Stamp (DTS) of the fragment. Set after transmux complete.
  126. public startDTS!: number;
  127. // The ending Decode Time Stamp (DTS) of the fragment. Set after transmux complete.
  128. public endDTS!: number;
  129. // The start time of the fragment, as listed in the manifest. Updated after transmux complete.
  130. public start: number = 0;
  131. // Set by `updateFragPTSDTS` in level-helper
  132. public deltaPTS?: number;
  133. // The maximum starting Presentation Time Stamp (audio/video PTS) of the fragment. Set after transmux complete.
  134. public maxStartPTS?: number;
  135. // The minimum ending Presentation Time Stamp (audio/video PTS) of the fragment. Set after transmux complete.
  136. public minEndPTS?: number;
  137. // Load/parse timing information
  138. public stats: LoadStats = new LoadStats();
  139. public urlId: number = 0;
  140. public data?: Uint8Array;
  141. // A flag indicating whether the segment was downloaded in order to test bitrate, and was not buffered
  142. public bitrateTest: boolean = false;
  143. // #EXTINF segment title
  144. public title: string | null = null;
  145. // The Media Initialization Section for this segment
  146. public initSegment: Fragment | null = null;
  147. // Fragment is the last fragment in the media playlist
  148. public endList?: boolean;
  149.  
  150. constructor(type: PlaylistLevelType, baseurl: string) {
  151. super(baseurl);
  152. this.type = type;
  153. }
  154.  
  155. get decryptdata(): LevelKey | null {
  156. const { levelkeys } = this;
  157. if (!levelkeys && !this._decryptdata) {
  158. return null;
  159. }
  160.  
  161. if (!this._decryptdata && this.levelkeys && !this.levelkeys.NONE) {
  162. const key = this.levelkeys.identity;
  163. if (key) {
  164. this._decryptdata = key.getDecryptData(this.sn);
  165. } else {
  166. const keyFormats = Object.keys(this.levelkeys);
  167. if (keyFormats.length === 1) {
  168. return (this._decryptdata = this.levelkeys[
  169. keyFormats[0]
  170. ].getDecryptData(this.sn));
  171. } else {
  172. // Multiple keys. key-loader to call Fragment.setKeyFormat based on selected key-system.
  173. }
  174. }
  175. }
  176.  
  177. return this._decryptdata;
  178. }
  179.  
  180. get end(): number {
  181. return this.start + this.duration;
  182. }
  183.  
  184. get endProgramDateTime() {
  185. if (this.programDateTime === null) {
  186. return null;
  187. }
  188.  
  189. if (!Number.isFinite(this.programDateTime)) {
  190. return null;
  191. }
  192.  
  193. const duration = !Number.isFinite(this.duration) ? 0 : this.duration;
  194.  
  195. return this.programDateTime + duration * 1000;
  196. }
  197.  
  198. get encrypted() {
  199. // At the m3u8-parser level we need to add support for manifest signalled keyformats
  200. // when we want the fragment to start reporting that it is encrypted.
  201. // Currently, keyFormat will only be set for identity keys
  202. if (this._decryptdata?.encrypted) {
  203. return true;
  204. } else if (this.levelkeys) {
  205. const keyFormats = Object.keys(this.levelkeys);
  206. const len = keyFormats.length;
  207. if (len > 1 || (len === 1 && this.levelkeys[keyFormats[0]].encrypted)) {
  208. return true;
  209. }
  210. }
  211.  
  212. return false;
  213. }
  214.  
  215. setKeyFormat(keyFormat: KeySystemFormats) {
  216. if (this.levelkeys) {
  217. const key = this.levelkeys[keyFormat];
  218. if (key && !this._decryptdata) {
  219. this._decryptdata = key.getDecryptData(this.sn);
  220. }
  221. }
  222. }
  223.  
  224. abortRequests(): void {
  225. this.loader?.abort();
  226. this.keyLoader?.abort();
  227. }
  228.  
  229. setElementaryStreamInfo(
  230. type: ElementaryStreamTypes,
  231. startPTS: number,
  232. endPTS: number,
  233. startDTS: number,
  234. endDTS: number,
  235. partial: boolean = false
  236. ) {
  237. const { elementaryStreams } = this;
  238. const info = elementaryStreams[type];
  239. if (!info) {
  240. elementaryStreams[type] = {
  241. startPTS,
  242. endPTS,
  243. startDTS,
  244. endDTS,
  245. partial,
  246. };
  247. return;
  248. }
  249.  
  250. info.startPTS = Math.min(info.startPTS, startPTS);
  251. info.endPTS = Math.max(info.endPTS, endPTS);
  252. info.startDTS = Math.min(info.startDTS, startDTS);
  253. info.endDTS = Math.max(info.endDTS, endDTS);
  254. }
  255.  
  256. clearElementaryStreamInfo() {
  257. const { elementaryStreams } = this;
  258. elementaryStreams[ElementaryStreamTypes.AUDIO] = null;
  259. elementaryStreams[ElementaryStreamTypes.VIDEO] = null;
  260. elementaryStreams[ElementaryStreamTypes.AUDIOVIDEO] = null;
  261. }
  262. }
  263.  
  264. export class Part extends BaseSegment {
  265. public readonly fragOffset: number = 0;
  266. public readonly duration: number = 0;
  267. public readonly gap: boolean = false;
  268. public readonly independent: boolean = false;
  269. public readonly relurl: string;
  270. public readonly fragment: Fragment;
  271. public readonly index: number;
  272. public stats: LoadStats = new LoadStats();
  273.  
  274. constructor(
  275. partAttrs: AttrList,
  276. frag: Fragment,
  277. baseurl: string,
  278. index: number,
  279. previous?: Part
  280. ) {
  281. super(baseurl);
  282. this.duration = partAttrs.decimalFloatingPoint('DURATION');
  283. this.gap = partAttrs.bool('GAP');
  284. this.independent = partAttrs.bool('INDEPENDENT');
  285. this.relurl = partAttrs.enumeratedString('URI') as string;
  286. this.fragment = frag;
  287. this.index = index;
  288. const byteRange = partAttrs.enumeratedString('BYTERANGE');
  289. if (byteRange) {
  290. this.setByteRange(byteRange, previous);
  291. }
  292. if (previous) {
  293. this.fragOffset = previous.fragOffset + previous.duration;
  294. }
  295. }
  296.  
  297. get start(): number {
  298. return this.fragment.start + this.fragOffset;
  299. }
  300.  
  301. get end(): number {
  302. return this.start + this.duration;
  303. }
  304.  
  305. get loaded(): boolean {
  306. const { elementaryStreams } = this;
  307. return !!(
  308. elementaryStreams.audio ||
  309. elementaryStreams.video ||
  310. elementaryStreams.audiovideo
  311. );
  312. }
  313. }