Home Reference Source

src/controller/fragment-finders.ts

  1. import BinarySearch from '../utils/binary-search';
  2. import { Fragment } from '../loader/fragment';
  3.  
  4. /**
  5. * Returns first fragment whose endPdt value exceeds the given PDT.
  6. * @param {Array<Fragment>} fragments - The array of candidate fragments
  7. * @param {number|null} [PDTValue = null] - The PDT value which must be exceeded
  8. * @param {number} [maxFragLookUpTolerance = 0] - The amount of time that a fragment's start/end can be within in order to be considered contiguous
  9. * @returns {*|null} fragment - The best matching fragment
  10. */
  11. export function findFragmentByPDT(
  12. fragments: Array<Fragment>,
  13. PDTValue: number | null,
  14. maxFragLookUpTolerance: number
  15. ): Fragment | null {
  16. if (
  17. PDTValue === null ||
  18. !Array.isArray(fragments) ||
  19. !fragments.length ||
  20. !Number.isFinite(PDTValue)
  21. ) {
  22. return null;
  23. }
  24.  
  25. // if less than start
  26. const startPDT = fragments[0].programDateTime;
  27. if (PDTValue < (startPDT || 0)) {
  28. return null;
  29. }
  30.  
  31. const endPDT = fragments[fragments.length - 1].endProgramDateTime;
  32. if (PDTValue >= (endPDT || 0)) {
  33. return null;
  34. }
  35.  
  36. maxFragLookUpTolerance = maxFragLookUpTolerance || 0;
  37. for (let seg = 0; seg < fragments.length; ++seg) {
  38. const frag = fragments[seg];
  39. if (pdtWithinToleranceTest(PDTValue, maxFragLookUpTolerance, frag)) {
  40. return frag;
  41. }
  42. }
  43.  
  44. return null;
  45. }
  46.  
  47. /**
  48. * Finds a fragment based on the SN of the previous fragment; or based on the needs of the current buffer.
  49. * This method compensates for small buffer gaps by applying a tolerance to the start of any candidate fragment, thus
  50. * breaking any traps which would cause the same fragment to be continuously selected within a small range.
  51. * @param {*} fragPrevious - The last frag successfully appended
  52. * @param {Array} fragments - The array of candidate fragments
  53. * @param {number} [bufferEnd = 0] - The end of the contiguous buffered range the playhead is currently within
  54. * @param {number} maxFragLookUpTolerance - The amount of time that a fragment's start/end can be within in order to be considered contiguous
  55. * @returns {*} foundFrag - The best matching fragment
  56. */
  57. export function findFragmentByPTS(
  58. fragPrevious: Fragment | null,
  59. fragments: Array<Fragment>,
  60. bufferEnd: number = 0,
  61. maxFragLookUpTolerance: number = 0
  62. ): Fragment | null {
  63. let fragNext: Fragment | null = null;
  64. if (fragPrevious) {
  65. fragNext =
  66. fragments[
  67. (fragPrevious.sn as number) - (fragments[0].sn as number) + 1
  68. ] || null;
  69. } else if (bufferEnd === 0 && fragments[0].start === 0) {
  70. fragNext = fragments[0];
  71. }
  72. // Prefer the next fragment if it's within tolerance
  73. if (
  74. fragNext &&
  75. fragmentWithinToleranceTest(bufferEnd, maxFragLookUpTolerance, fragNext) ===
  76. 0
  77. ) {
  78. return fragNext;
  79. }
  80. // We might be seeking past the tolerance so find the best match
  81. const foundFragment = BinarySearch.search(
  82. fragments,
  83. fragmentWithinToleranceTest.bind(null, bufferEnd, maxFragLookUpTolerance)
  84. );
  85. if (foundFragment && (foundFragment !== fragPrevious || !fragNext)) {
  86. return foundFragment;
  87. }
  88. // If no match was found return the next fragment after fragPrevious, or null
  89. return fragNext;
  90. }
  91.  
  92. /**
  93. * The test function used by the findFragmentBySn's BinarySearch to look for the best match to the current buffer conditions.
  94. * @param {*} candidate - The fragment to test
  95. * @param {number} [bufferEnd = 0] - The end of the current buffered range the playhead is currently within
  96. * @param {number} [maxFragLookUpTolerance = 0] - The amount of time that a fragment's start can be within in order to be considered contiguous
  97. * @returns {number} - 0 if it matches, 1 if too low, -1 if too high
  98. */
  99. export function fragmentWithinToleranceTest(
  100. bufferEnd = 0,
  101. maxFragLookUpTolerance = 0,
  102. candidate: Fragment
  103. ) {
  104. // eagerly accept an accurate match (no tolerance)
  105. if (
  106. candidate.start <= bufferEnd &&
  107. candidate.start + candidate.duration > bufferEnd
  108. ) {
  109. return 0;
  110. }
  111. // offset should be within fragment boundary - config.maxFragLookUpTolerance
  112. // this is to cope with situations like
  113. // bufferEnd = 9.991
  114. // frag[Ø] : [0,10]
  115. // frag[1] : [10,20]
  116. // bufferEnd is within frag[0] range ... although what we are expecting is to return frag[1] here
  117. // frag start frag start+duration
  118. // |-----------------------------|
  119. // <---> <--->
  120. // ...--------><-----------------------------><---------....
  121. // previous frag matching fragment next frag
  122. // return -1 return 0 return 1
  123. // logger.log(`level/sn/start/end/bufEnd:${level}/${candidate.sn}/${candidate.start}/${(candidate.start+candidate.duration)}/${bufferEnd}`);
  124. // Set the lookup tolerance to be small enough to detect the current segment - ensures we don't skip over very small segments
  125. const candidateLookupTolerance = Math.min(
  126. maxFragLookUpTolerance,
  127. candidate.duration + (candidate.deltaPTS ? candidate.deltaPTS : 0)
  128. );
  129. if (
  130. candidate.start + candidate.duration - candidateLookupTolerance <=
  131. bufferEnd
  132. ) {
  133. return 1;
  134. } else if (
  135. candidate.start - candidateLookupTolerance > bufferEnd &&
  136. candidate.start
  137. ) {
  138. // if maxFragLookUpTolerance will have negative value then don't return -1 for first element
  139. return -1;
  140. }
  141.  
  142. return 0;
  143. }
  144.  
  145. /**
  146. * The test function used by the findFragmentByPdt's BinarySearch to look for the best match to the current buffer conditions.
  147. * This function tests the candidate's program date time values, as represented in Unix time
  148. * @param {*} candidate - The fragment to test
  149. * @param {number} [pdtBufferEnd = 0] - The Unix time representing the end of the current buffered range
  150. * @param {number} [maxFragLookUpTolerance = 0] - The amount of time that a fragment's start can be within in order to be considered contiguous
  151. * @returns {boolean} True if contiguous, false otherwise
  152. */
  153. export function pdtWithinToleranceTest(
  154. pdtBufferEnd: number,
  155. maxFragLookUpTolerance: number,
  156. candidate: Fragment
  157. ): boolean {
  158. const candidateLookupTolerance =
  159. Math.min(
  160. maxFragLookUpTolerance,
  161. candidate.duration + (candidate.deltaPTS ? candidate.deltaPTS : 0)
  162. ) * 1000;
  163.  
  164. // endProgramDateTime can be null, default to zero
  165. const endProgramDateTime = candidate.endProgramDateTime || 0;
  166. return endProgramDateTime - candidateLookupTolerance > pdtBufferEnd;
  167. }
  168.  
  169. export function findFragWithCC(
  170. fragments: Fragment[],
  171. cc: number
  172. ): Fragment | null {
  173. return BinarySearch.search(fragments, (candidate) => {
  174. if (candidate.cc < cc) {
  175. return 1;
  176. } else if (candidate.cc > cc) {
  177. return -1;
  178. } else {
  179. return 0;
  180. }
  181. });
  182. }