diff --git a/src/controller/audio-stream-controller.ts b/src/controller/audio-stream-controller.ts index a546e03efb5..a4fd4453132 100644 --- a/src/controller/audio-stream-controller.ts +++ b/src/controller/audio-stream-controller.ts @@ -909,6 +909,13 @@ class AudioStreamController `Waiting for video PTS in continuity counter ${frag.cc} of live stream before loading audio fragment ${frag.sn} of level ${this.trackId}` ); this.state = State.WAITING_INIT_PTS; + const mainDetails = this.mainDetails; + if ( + mainDetails && + mainDetails.fragments[0].start !== track.details.fragments[0].start + ) { + alignMediaPlaylistByPDT(track.details, mainDetails); + } } else { this.startFragRequested = true; super.loadFragment(frag, track, targetBufferTime); diff --git a/src/utils/discontinuities.ts b/src/utils/discontinuities.ts index 59eb6b6eadc..edfb96e9add 100644 --- a/src/utils/discontinuities.ts +++ b/src/utils/discontinuities.ts @@ -6,18 +6,16 @@ import type { LevelDetails } from '../loader/level-details'; import type { Level } from '../types/level'; import type { RequiredProperties } from '../types/general'; -export function findFirstFragWithCC(fragments: Fragment[], cc: number) { - let firstFrag: Fragment | null = null; - +export function findFirstFragWithCC( + fragments: Fragment[], + cc: number +): Fragment | null { for (let i = 0, len = fragments.length; i < len; i++) { - const currentFrag = fragments[i]; - if (currentFrag && currentFrag.cc === cc) { - firstFrag = currentFrag; - break; + if (fragments[i]?.cc === cc) { + return fragments[i]; } } - - return firstFrag; + return null; } export function shouldAlignOnDiscontinuities( @@ -39,8 +37,7 @@ export function shouldAlignOnDiscontinuities( // Find the first frag in the previous level which matches the CC of the first frag of the new level export function findDiscontinuousReferenceFrag( prevDetails: LevelDetails, - curDetails: LevelDetails, - referenceIndex: number = 0 + curDetails: LevelDetails ) { const prevFrags = prevDetails.fragments; const curFrags = curDetails.fragments; @@ -104,7 +101,7 @@ export function alignStream( // If the PTS wasn't figured out via discontinuity sequence that means there was no CC increase within the level. // Aligning via Program Date Time should therefore be reliable, since PDT should be the same within the same // discontinuity sequence. - alignPDT(details, lastLevel.details); + alignMediaPlaylistByPDT(details, lastLevel.details); } if ( !details.alignedSliding && @@ -145,40 +142,9 @@ function alignDiscontinuities( } /** - * Computes the PTS of a new level's fragments using the difference in Program Date Time from the last level. - * @param details - The details of the new level - * @param lastDetails - The details of the last loaded level - */ -export function alignPDT(details: LevelDetails, lastDetails: LevelDetails) { - // This check protects the unsafe "!" usage below for null program date time access. - if ( - !lastDetails.fragments.length || - !details.hasProgramDateTime || - !lastDetails.hasProgramDateTime - ) { - return; - } - // if last level sliding is 1000 and its first frag PROGRAM-DATE-TIME is 2017-08-20 1:10:00 AM - // and if new details first frag PROGRAM DATE-TIME is 2017-08-20 1:10:08 AM - // then we can deduce that playlist B sliding is 1000+8 = 1008s - const lastPDT = lastDetails.fragments[0].programDateTime!; // hasProgramDateTime check above makes this safe. - const newPDT = details.fragments[0].programDateTime!; - // date diff is in ms. frag.start is in seconds - const sliding = (newPDT - lastPDT) / 1000 + lastDetails.fragments[0].start; - if (sliding && Number.isFinite(sliding)) { - logger.log( - `Adjusting PTS using programDateTime delta ${ - newPDT - lastPDT - }ms, sliding:${sliding.toFixed(3)} ${details.url} ` - ); - adjustSlidingStart(sliding, details); - } -} - -/** - * Ensures appropriate time-alignment between renditions based on PDT. Unlike `alignPDT`, which adjusts - * the timeline based on the delta between PDTs of the 0th fragment of two playlists/`LevelDetails`, - * this function assumes the timelines represented in `refDetails` are accurate, including the PDTs, + * Ensures appropriate time-alignment between renditions based on PDT. + * This function assumes the timelines represented in `refDetails` are accurate, including the PDTs + * for the last discontinuity sequence number shared by both playlists when present, * and uses the "wallclock"/PDT timeline as a cross-reference to `details`, adjusting the presentation * times/timelines of `details` accordingly. * Given the asynchronous nature of fetches and initial loads of live `main` and audio/subtitle tracks, @@ -205,15 +171,22 @@ export function alignMediaPlaylistByPDT( // Calculate a delta to apply to all fragments according to the delta in PDT times and start times // of a fragment in the reference details, and a fragment in the target details of the same discontinuity. // If a fragment of the same discontinuity was not found use the middle fragment of both. - const middleFrag = Math.round(refFragments.length / 2) - 1; - const refFrag = refFragments[middleFrag]; - const frag = - findFirstFragWithCC(fragments, refFrag.cc) || - fragments[Math.round(fragments.length / 2) - 1]; - + let refFrag: Fragment | null | undefined; + let frag: Fragment | null | undefined; + const targetCC = Math.min(refDetails.endCC, details.endCC); + if (refDetails.startCC < targetCC && details.startCC < targetCC) { + refFrag = findFirstFragWithCC(refFragments, targetCC); + frag = findFirstFragWithCC(fragments, targetCC); + } + if (!refFrag || !frag) { + refFrag = refFragments[Math.floor(refFragments.length / 2)]; + frag = + findFirstFragWithCC(fragments, refFrag.cc) || + fragments[Math.floor(fragments.length / 2)]; + } const refPDT = refFrag.programDateTime; const targetPDT = frag.programDateTime; - if (refPDT === null || targetPDT === null) { + if (!refPDT || !targetPDT) { return; } diff --git a/tests/unit/utils/discontinuities.js b/tests/unit/utils/discontinuities.ts similarity index 52% rename from tests/unit/utils/discontinuities.js rename to tests/unit/utils/discontinuities.ts index e6f4b1c4a69..1dbe87068a9 100644 --- a/tests/unit/utils/discontinuities.js +++ b/tests/unit/utils/discontinuities.ts @@ -2,17 +2,27 @@ import { shouldAlignOnDiscontinuities, findDiscontinuousReferenceFrag, adjustSlidingStart, - alignPDT, alignMediaPlaylistByPDT, } from '../../../src/utils/discontinuities'; +import { LevelDetails } from '../../../src/loader/level-details'; +import { Fragment } from '../../../src/loader/fragment'; +import { PlaylistLevelType } from '../../../src/types/loader'; -const mockReferenceFrag = { +import chai from 'chai'; +import sinonChai from 'sinon-chai'; +import { AttrList } from '../../../src/utils/attr-list'; +import { Level } from '../../../src/types/level'; + +chai.use(sinonChai); +const expect = chai.expect; + +const mockReferenceFrag = objToFragment({ start: 20, startPTS: 20, endPTS: 24, duration: 4, cc: 0, -}; +}); const mockFrags = [ { @@ -36,15 +46,15 @@ const mockFrags = [ duration: 8, cc: 1, }, -]; +].map(objToFragment); describe('discontinuities', function () { it('adjusts level fragments with overlapping CC range using a reference fragment', function () { - const details = { + const details = objToLevelDetails({ fragments: mockFrags.slice(0), PTSKnown: false, alignedSliding: false, - }; + }); const expected = [ { start: 20, @@ -67,7 +77,7 @@ describe('discontinuities', function () { duration: 8, cc: 1, }, - ]; + ].map(objToFragment); adjustSlidingStart(mockReferenceFrag.start, details); expect(expected).to.deep.equal(details.fragments); @@ -76,10 +86,11 @@ describe('discontinuities', function () { it('aligns level fragments times based on PDT and start time of reference level details', function () { const lastLevel = { - details: { + details: objToLevelDetails({ PTSKnown: false, alignedSliding: false, - hasProgramDateTime: true, + startCC: 0, + endCC: 0, fragments: [ { start: 18, @@ -102,18 +113,18 @@ describe('discontinuities', function () { duration: 8, programDateTime: 1629821770107, }, - ], - fragmentHint: { + ].map(objToFragment), + fragmentHint: objToFragment({ start: 30, startPTS: 30, endPTS: 32, duration: 2, programDateTime: 1629821778107, - }, - }, + }), + }), }; - const refDetails = { + const refDetails = objToLevelDetails({ fragments: [ { start: 18, @@ -122,13 +133,14 @@ describe('discontinuities', function () { duration: 2, programDateTime: 1629821768107, }, - ], + ].map(objToFragment), PTSKnown: false, alignedSliding: false, - hasProgramDateTime: true, - }; + startCC: 0, + endCC: 0, + }); - const detailsExpected = { + const detailsExpected = objToLevelDetails({ fragments: [ { start: 16, @@ -151,18 +163,19 @@ describe('discontinuities', function () { duration: 8, programDateTime: 1629821770107, }, - ], - fragmentHint: { + ].map(objToFragment), + fragmentHint: objToFragment({ start: 28, startPTS: 28, endPTS: 30, duration: 2, programDateTime: 1629821778107, - }, + }), PTSKnown: false, alignedSliding: true, - hasProgramDateTime: true, - }; + startCC: 0, + endCC: 0, + }); alignMediaPlaylistByPDT(lastLevel.details, refDetails); expect( lastLevel.details, @@ -174,12 +187,142 @@ describe('discontinuities', function () { ).to.deep.equal(detailsExpected); }); + it('aligns level fragments with overlapping CC bounds and discontiguous programDateTime info', function () { + const lastLevel = objToLevel({ + details: objToLevelDetails({ + PTSKnown: true, + alignedSliding: false, + startCC: 1, + endCC: 3, + fragments: [ + { + start: 20, + startPTS: 20, + endPTS: 24, + duration: 4, + cc: 1, + programDateTime: 1503892800000, + }, + { + start: 24, + startPTS: 24, + endPTS: 28, + duration: 4, + cc: 2, + programDateTime: 1503892850000, + }, + { + start: 28, + startPTS: 28, + endPTS: 36, + duration: 8, + cc: 3, + programDateTime: 1501111110000, + }, + { + start: 28, + startPTS: 28, + endPTS: 36, + duration: 8, + cc: 3, + programDateTime: 1501111118000, + }, + ].map(objToFragment), + }), + }); + + const details = objToLevelDetails({ + fragments: [ + { + start: 0, + startPTS: 0, + endPTS: 4, + duration: 4, + cc: 2, + programDateTime: 1503892850000, + }, + { + start: 4, + startPTS: 4, + endPTS: 8, + duration: 4, + cc: 3, + programDateTime: 1501111110000, + }, + { + start: 8, + startPTS: 8, + endPTS: 12, + duration: 4, + cc: 3, + programDateTime: 1501111114000, + }, + { + start: 12, + startPTS: 12, + endPTS: 16, + duration: 4, + cc: 4, + programDateTime: 1503892854000, + }, + ].map(objToFragment), + PTSKnown: false, + alignedSliding: false, + startCC: 2, + endCC: 4, + }); + + const detailsExpected = objToLevelDetails({ + fragments: [ + { + start: 24, + startPTS: 24, + endPTS: 28, + duration: 4, + cc: 2, + programDateTime: 1503892850000, + }, + { + start: 28, + startPTS: 28, + endPTS: 32, + duration: 4, + cc: 3, + programDateTime: 1501111110000, + }, + { + start: 32, + startPTS: 32, + endPTS: 36, + duration: 4, + cc: 3, + programDateTime: 1501111114000, + }, + { + start: 36, + startPTS: 36, + endPTS: 40, + duration: 4, + cc: 4, + programDateTime: 1503892854000, + }, + ].map(objToFragment), + PTSKnown: false, + alignedSliding: true, + startCC: 2, + endCC: 4, + }); + alignMediaPlaylistByPDT(details, lastLevel.details as LevelDetails); + expect(details).to.deep.equal(detailsExpected); + }); + it('adjusts level fragments without overlapping CC range but with programDateTime info', function () { const lastLevel = { - details: { + details: objToLevelDetails({ PTSKnown: true, alignedSliding: false, - hasProgramDateTime: true, + startCC: 0, + endCC: 1, fragments: [ { start: 20, @@ -195,6 +338,7 @@ describe('discontinuities', function () { endPTS: 28, duration: 4, cc: 1, + programDateTime: 1503892804000, }, { start: 28, @@ -202,12 +346,13 @@ describe('discontinuities', function () { endPTS: 36, duration: 8, cc: 1, + programDateTime: 1503892808000, }, - ], - }, + ].map(objToFragment), + }), }; - const details = { + const details = objToLevelDetails({ fragments: [ { start: 0, @@ -223,6 +368,7 @@ describe('discontinuities', function () { endPTS: 8, duration: 4, cc: 2, + programDateTime: 1503892854000, }, { start: 8, @@ -230,16 +376,16 @@ describe('discontinuities', function () { endPTS: 16, duration: 8, cc: 3, + programDateTime: 1503892858000, }, - ], + ].map(objToFragment), PTSKnown: false, alignedSliding: false, startCC: 2, endCC: 3, - hasProgramDateTime: true, - }; + }); - const detailsExpected = { + const detailsExpected = objToLevelDetails({ fragments: [ { start: 70, @@ -255,6 +401,7 @@ describe('discontinuities', function () { endPTS: 78, duration: 4, cc: 2, + programDateTime: 1503892854000, }, { start: 78, @@ -262,35 +409,35 @@ describe('discontinuities', function () { endPTS: 86, duration: 8, cc: 3, + programDateTime: 1503892858000, }, - ], + ].map(objToFragment), PTSKnown: false, alignedSliding: true, startCC: 2, endCC: 3, - hasProgramDateTime: true, - }; - alignPDT(details, lastLevel.details); - expect(detailsExpected).to.deep.equal(details); + }); + alignMediaPlaylistByPDT(details, lastLevel.details); + expect(detailsExpected).to.deep.equal(details, JSON.stringify(details)); }); it('finds the first fragment in an array which matches the CC of the first fragment in another array', function () { - const prevDetails = { - fragments: [mockReferenceFrag, { cc: 1 }], - }; - const curDetails = { + const prevDetails = objToLevelDetails({ + fragments: [mockReferenceFrag, { cc: 1 }].map(objToFragment), + }); + const curDetails = objToLevelDetails({ fragments: mockFrags, - }; + }); const expected = mockReferenceFrag; const actual = findDiscontinuousReferenceFrag(prevDetails, curDetails); - expect(actual).to.equal(expected); + expect(actual).to.deep.equal(expected); }); it('returns undefined if there are no frags in the previous level', function () { const expected = undefined; const actual = findDiscontinuousReferenceFrag( - { fragments: [] }, - { fragments: mockFrags } + objToLevelDetails({ fragments: [] }), + objToLevelDetails({ fragments: mockFrags }) ); expect(actual).to.equal(expected); }); @@ -298,8 +445,8 @@ describe('discontinuities', function () { it('returns undefined if there are no matching frags in the previous level', function () { const expected = undefined; const actual = findDiscontinuousReferenceFrag( - { fragments: [{ cc: 10 }] }, - { fragments: mockFrags } + objToLevelDetails({ fragments: [{ cc: 10 }].map(objToFragment) }), + objToLevelDetails({ fragments: mockFrags }) ); expect(actual).to.equal(expected); }); @@ -307,36 +454,36 @@ describe('discontinuities', function () { it('returns undefined if there are no frags in the current level', function () { const expected = undefined; const actual = findDiscontinuousReferenceFrag( - { fragments: [{ cc: 0 }] }, - { fragments: [] } + objToLevelDetails({ fragments: [{ cc: 0 }].map(objToFragment) }), + objToLevelDetails({ fragments: [] }) ); expect(actual).to.equal(expected); }); it('should align current level when CC increases within the level', function () { - const lastLevel = { - details: {}, - }; - const curDetails = { + const lastLevel = objToLevel({ + details: objToLevelDetails({}), + }); + const curDetails = objToLevelDetails({ startCC: 0, endCC: 1, - }; + }); const actual = shouldAlignOnDiscontinuities(null, lastLevel, curDetails); expect(actual).to.be.true; }); it('should align current level when CC increases from last frag to current level', function () { - const lastLevel = { - details: {}, - }; - const lastFrag = { + const lastLevel = objToLevel({ + details: objToLevelDetails({}), + }); + const lastFrag = objToFragment({ cc: 0, - }; - const curDetails = { + }); + const curDetails = objToLevelDetails({ startCC: 1, endCC: 1, - }; + }); const actual = shouldAlignOnDiscontinuities( lastFrag, @@ -347,16 +494,16 @@ describe('discontinuities', function () { }); it('should not align when there is no CC increase', function () { - const lastLevel = { - details: {}, - }; - const curDetails = { + const lastLevel = objToLevel({ + details: objToLevelDetails({}), + }); + const curDetails = objToLevelDetails({ startCC: 1, endCC: 1, - }; - const lastFrag = { + }); + const lastFrag = objToFragment({ cc: 1, - }; + }); const actual = shouldAlignOnDiscontinuities( lastFrag, @@ -367,14 +514,14 @@ describe('discontinuities', function () { }); it('should not align when there are no previous level details', function () { - const lastLevel = {}; - const curDetails = { + const lastLevel = objToLevel({}); + const curDetails = objToLevelDetails({ startCC: 1, endCC: 1, - }; - const lastFrag = { + }); + const lastFrag = objToFragment({ cc: 1, - }; + }); const actual = shouldAlignOnDiscontinuities( lastFrag, @@ -384,3 +531,32 @@ describe('discontinuities', function () { expect(actual).to.be.false; }); }); + +function objToFragment(object: Partial): Fragment { + const fragment = new Fragment(PlaylistLevelType.MAIN, ''); + for (const prop in object) { + fragment[prop] = object[prop]; + } + return fragment; +} + +function objToLevelDetails(object: Partial): LevelDetails { + const details = new LevelDetails(''); + for (const prop in object) { + details[prop] = object[prop]; + } + return details; +} + +function objToLevel(object: Partial): Level { + const level = new Level({ + name: '', + url: '', + attrs: new AttrList({}), + bitrate: 500000, + }); + for (const prop in object) { + level[prop] = object[prop]; + } + return level; +}