-
Notifications
You must be signed in to change notification settings - Fork 1.4k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
fix: Fix multi-period DASH with descriptive audio #4629
Changes from 26 commits
90b4527
49d611f
93fdcd3
0b7ac5b
6db0e3c
efaebee
7fed0d0
7a30028
0168a7f
6b64092
a0cdf79
28564dd
5fb344e
1766930
b0cad7c
868b75b
95589c9
f331382
d14c111
b700d3b
0d9ea95
ef5177b
937638e
6f0d4fe
956679e
6338d7e
8d1c69e
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -12,6 +12,14 @@ describe('PeriodCombiner', () => { | |
/** @type {shaka.util.PeriodCombiner} */ | ||
let combiner; | ||
|
||
const makeAudioStreamWithRoles = (id, roles, primary = true) => { | ||
const stream = makeAudioStream('en'); | ||
stream.originalId = id; | ||
stream.roles = roles; | ||
stream.primary = primary; | ||
return stream; | ||
}; | ||
|
||
beforeEach(() => { | ||
combiner = new shaka.util.PeriodCombiner(); | ||
}); | ||
|
@@ -1141,14 +1149,163 @@ describe('PeriodCombiner', () => { | |
expect(audio2.originalId).toBe('stream2,stream4'); | ||
}); | ||
|
||
it('Matches streams with roles in common', async () => { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. This test simulates the stream that originally manifested the issue. It is a multi-period steam where all periods have the same number of video and audio adaptation sets, and the audio adaptations both have roles, one |
||
/** @type {!Array.<shaka.util.PeriodCombiner.Period>} */ | ||
const periods = [ | ||
{ | ||
id: '1', | ||
videoStreams: [ | ||
makeVideoStream(720), | ||
makeVideoStream(1080), | ||
], | ||
audioStreams: [ | ||
makeAudioStreamWithRoles('stream1', ['main']), | ||
makeAudioStreamWithRoles('stream2', ['description'], false), | ||
], | ||
textStreams: [], | ||
imageStreams: [], | ||
}, | ||
{ | ||
id: '2', | ||
videoStreams: [ | ||
makeVideoStream(720), | ||
makeVideoStream(1080), | ||
], | ||
audioStreams: [ | ||
makeAudioStreamWithRoles('stream1', ['main']), | ||
makeAudioStreamWithRoles('stream2', ['description'], false), | ||
], | ||
textStreams: [], | ||
imageStreams: [], | ||
}, | ||
]; | ||
|
||
await combiner.combinePeriods(periods, /* isDynamic= */ false); | ||
const variants = combiner.getVariants(); | ||
|
||
console.log(variants); | ||
|
||
expect(variants.length).toBe(4); | ||
|
||
expect(variants).toEqual(jasmine.arrayWithExactContents([ | ||
makeAVVariant(720, 'en'), | ||
makeAVVariant(1080, 'en'), | ||
makeAVVariant(720, 'en'), | ||
makeAVVariant(1080, 'en'), | ||
])); | ||
|
||
// We can use the originalId field to see what each track is composed of. | ||
const audio1 = variants[0].audio; | ||
expect(audio1.roles).toEqual(['main']); | ||
expect(audio1.originalId).toBe('stream1,stream1'); | ||
|
||
const audio2 = variants[1].audio; | ||
expect(audio2.roles).toEqual(['main']); | ||
expect(audio2.originalId).toBe('stream1,stream1'); | ||
|
||
const audio3 = variants[2].audio; | ||
expect(audio3.roles).toEqual(['description']); | ||
expect(audio3.originalId).toBe('stream2,stream2'); | ||
|
||
const audio4 = variants[3].audio; | ||
expect(audio4.roles).toEqual(['description']); | ||
expect(audio4.originalId).toBe('stream2,stream2'); | ||
}); | ||
|
||
it('Matches streams with mismatched roles', async () => { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. This test simulates a similar stream to the one above, but this time with periods representing pre-roll and mid-roll ads that only have |
||
/** @type {!Array.<shaka.util.PeriodCombiner.Period>} */ | ||
const periods = [ | ||
{ | ||
id: '0', | ||
videoStreams: [ | ||
makeVideoStream(720), | ||
makeVideoStream(1080), | ||
], | ||
audioStreams: [ | ||
makeAudioStreamWithRoles('stream1', ['main']), | ||
], | ||
textStreams: [], | ||
imageStreams: [], | ||
}, | ||
{ | ||
id: '1', | ||
videoStreams: [ | ||
makeVideoStream(720), | ||
makeVideoStream(1080), | ||
], | ||
audioStreams: [ | ||
makeAudioStreamWithRoles('stream1', ['main']), | ||
makeAudioStreamWithRoles('stream2', ['description'], false), | ||
], | ||
textStreams: [], | ||
imageStreams: [], | ||
}, | ||
{ | ||
id: '2', | ||
videoStreams: [ | ||
makeVideoStream(720), | ||
makeVideoStream(1080), | ||
], | ||
audioStreams: [ | ||
makeAudioStreamWithRoles('stream1', ['main']), | ||
], | ||
textStreams: [], | ||
imageStreams: [], | ||
}, | ||
{ | ||
id: '3', | ||
videoStreams: [ | ||
makeVideoStream(720), | ||
makeVideoStream(1080), | ||
], | ||
audioStreams: [ | ||
makeAudioStreamWithRoles('stream1', ['main']), | ||
makeAudioStreamWithRoles('stream2', ['description'], false), | ||
], | ||
textStreams: [], | ||
imageStreams: [], | ||
}, | ||
]; | ||
|
||
await combiner.combinePeriods(periods, /* isDynamic= */ false); | ||
const variants = combiner.getVariants(); | ||
|
||
console.log(variants); | ||
|
||
expect(variants.length).toBe(4); | ||
|
||
expect(variants).toEqual(jasmine.arrayWithExactContents([ | ||
makeAVVariant(720, 'en'), | ||
makeAVVariant(1080, 'en'), | ||
makeAVVariant(720, 'en'), | ||
makeAVVariant(1080, 'en'), | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Could you add an optional roles argument to makeAVVariant? I think this part of the test would be clearer if we could see the expected roles here. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Done. |
||
])); | ||
|
||
// We can use the originalId field to see what each track is composed of. | ||
const audio1 = variants[0].audio; | ||
expect(audio1.roles).toEqual(['main']); | ||
expect(audio1.originalId).toBe('stream1,stream1,stream1,stream1'); | ||
|
||
const audio2 = variants[1].audio; | ||
expect(audio2.roles).toEqual(['main']); | ||
expect(audio2.originalId).toBe('stream1,stream1,stream1,stream1'); | ||
|
||
const audio3 = variants[2].audio; | ||
expect(audio3.roles).toEqual(['description', 'main']); | ||
expect(audio3.originalId).toBe('stream1,stream2,stream1,stream2'); | ||
|
||
const audio4 = variants[3].audio; | ||
expect(audio4.roles).toEqual(['description', 'main']); | ||
expect(audio4.originalId).toBe('stream1,stream2,stream1,stream2'); | ||
}); | ||
|
||
it('The number of variants stays stable after many periods ' + | ||
'when going between similar content and varying ads', async () => { | ||
// This test is based on the content from | ||
// https://github.com/shaka-project/shaka-player/issues/2716 | ||
// that used to cause our period flattening logic to keep | ||
// creating new variants for every new period added. | ||
// It's ok to create a few additional varinats/streams, | ||
// It's ok to create a few additional variants/streams, | ||
// but we should stabilize eventually and keep the number | ||
// of variants from growing indefinitely. | ||
|
||
|
@@ -1334,7 +1491,7 @@ describe('PeriodCombiner', () => { | |
expect(isCandidateBetter).toBe(WORSE); | ||
|
||
// Make sure it works correctly whether it's the candidate or the best | ||
// value that is equel to the output. | ||
// value that is equal to the output. | ||
isCandidateBetter = PeriodCombiner.compareClosestPreferLower( | ||
/* output= */ 5, /* bestValue= */ 3, /* candidateValue= */ 5); | ||
expect(isCandidateBetter).toBe(BETTER); | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@joeyparrish Removing this completely caused a number of issues, but moving it below the
role
check solves the issue.