Skip to content

Commit

Permalink
fix(HLS): Fix duplicate hinted segments (#4258)
Browse files Browse the repository at this point in the history
Because zero-duration references cause such chaos, ensure that the HLS
parser never produces these.  Preload-hinted segments should use the
target duration for partial segments, and if that required attribute
is missing from the playlist, then preload-hinted segments should be
skipped.

Closes #4223
  • Loading branch information
joeyparrish committed Jun 2, 2022
1 parent 1694b99 commit 4c8e8e7
Show file tree
Hide file tree
Showing 2 changed files with 74 additions and 10 deletions.
17 changes: 14 additions & 3 deletions lib/hls/hls_parser.js
Original file line number Diff line number Diff line change
Expand Up @@ -1775,9 +1775,20 @@ shaka.hls.HlsParser = class {
const pPreviousReference = i == 0 ?
previousReference : partialSegmentRefs[partialSegmentRefs.length - 1];
const pStartTime = (i == 0) ? startTime : pPreviousReference.endTime;
const pDuration = Number(item.getAttributeValue('DURATION'));
// A preload hinted partial segment doesn't have duration information,
// so its startTime and endTime are the same.

// If DURATION is missing from this partial segment, use the target
// partial duration from the top of the playlist, which is a required
// attribute for content with partial segments.
const pDuration = Number(item.getAttributeValue('DURATION')) ||
this.partialTargetDuration_;

// If for some reason we have neither an explicit duration, nor a target
// partial duration, we should SKIP this partial segment to avoid
// duplicating content in the presentation timeline.
if (!pDuration) {
continue;
}

const pEndTime = pStartTime + pDuration;

let pStartByte = 0;
Expand Down
67 changes: 60 additions & 7 deletions test/hls/hls_live_unit.js
Original file line number Diff line number Diff line change
Expand Up @@ -630,11 +630,13 @@ describe('HlsParser live', () => {
expect(ref.startTime).not.toBeLessThan(rolloverOffset);
});

// Test for https://github.com/shaka-project/shaka-player/issues/4223
it('parses streams with partial and preload hinted segments', async () => {
playerInterface.isLowLatencyMode = () => true;
const mediaWithPartialSegments = [
'#EXTM3U\n',
'#EXT-X-TARGETDURATION:5\n',
'#EXT-X-PART-INF:PART-TARGET=1.5\n',
'#EXT-X-MAP:URI="init.mp4",BYTERANGE="616@0"\n',
'#EXT-X-MEDIA-SEQUENCE:0\n',
// ref includes partialRef, partialRef2
Expand Down Expand Up @@ -668,6 +670,11 @@ describe('HlsParser live', () => {
segmentDataStartTime + 4, /* baseUri= */ '', /* startByte= */ 200,
/* endByte= */ 429);

const ref = ManifestParser.makeReference(
'test:/main.mp4', segmentDataStartTime, segmentDataStartTime + 4,
/* baseUri= */ '', /* startByte= */ 0, /* endByte= */ 429,
/* timestampOffset= */ 0, [partialRef, partialRef2]);

const partialRef3 = ManifestParser.makeReference(
'test:/partial.mp4', segmentDataStartTime + 4,
segmentDataStartTime + 6,
Expand All @@ -677,17 +684,12 @@ describe('HlsParser live', () => {
// so its startTime and endTime are the same.
const preloadRef = ManifestParser.makeReference(
'test:/partial.mp4', segmentDataStartTime + 6,
segmentDataStartTime + 6,
segmentDataStartTime + 7.5,
/* baseUri= */ '', /* startByte= */ 210, /* endByte= */ null);

const ref = ManifestParser.makeReference(
'test:/main.mp4', segmentDataStartTime, segmentDataStartTime + 4,
/* baseUri= */ '', /* startByte= */ 0, /* endByte= */ 429,
/* timestampOffset= */ 0, [partialRef, partialRef2]);

// ref2 is not fully published yet, so it doesn't have a segment uri.
const ref2 = ManifestParser.makeReference(
'', segmentDataStartTime + 4, segmentDataStartTime + 6,
'', segmentDataStartTime + 4, segmentDataStartTime + 7.5,
/* baseUri= */ '', /* startByte= */ 0, /* endByte= */ null,
/* timestampOffset= */ 0, [partialRef3, preloadRef]);

Expand All @@ -697,6 +699,56 @@ describe('HlsParser live', () => {
ManifestParser.verifySegmentIndex(video, [ref, ref2]);
});

// Test for https://github.com/shaka-project/shaka-player/issues/4223
it('ignores preload hinted segments without target duration', async () => {
playerInterface.isLowLatencyMode = () => true;

// Missing PART-TARGET, so preload hints are skipped.
const mediaWithPartialSegments = [
'#EXTM3U\n',
'#EXT-X-TARGETDURATION:5\n',
'#EXT-X-MAP:URI="init.mp4",BYTERANGE="616@0"\n',
'#EXT-X-MEDIA-SEQUENCE:0\n',
'#EXTINF:4,\n',
// ref1
'main.mp4\n',
// ref2 includes partialRef, but not preloadRef
// partialRef
'#EXT-X-PART:DURATION=2,URI="partial.mp4",BYTERANGE=210@0\n',
// preloadRef
'#EXT-X-PRELOAD-HINT:TYPE=PART,URI="partial.mp4",BYTERANGE-START=210\n',
].join('');

fakeNetEngine
.setResponseText('test:/master', master)
.setResponseText('test:/video', mediaWithPartialSegments)
.setResponseValue('test:/init.mp4', initSegmentData)
.setResponseValue('test:/main.mp4', segmentData)
.setResponseValue('test:/partial.mp4', segmentData)
.setResponseValue('test:/partial2.mp4', segmentData);

const ref1 = ManifestParser.makeReference(
'test:/main.mp4', segmentDataStartTime, segmentDataStartTime + 4,
/* baseUri= */ '', /* startByte= */ 0, /* endByte= */ null,
/* timestampOffset= */ 0, []);

const partialRef = ManifestParser.makeReference(
'test:/partial.mp4', segmentDataStartTime + 4,
segmentDataStartTime + 6,
/* baseUri= */ '', /* startByte= */ 0, /* endByte= */ 209);

// ref2 is not fully published yet, so it doesn't have a segment uri.
const ref2 = ManifestParser.makeReference(
'', segmentDataStartTime + 4, segmentDataStartTime + 6,
/* baseUri= */ '', /* startByte= */ 0, /* endByte= */ 209,
/* timestampOffset= */ 0, [partialRef]);

const manifest = await parser.start('test:/master', playerInterface);
const video = manifest.variants[0].video;
await video.createSegmentIndex();
ManifestParser.verifySegmentIndex(video, [ref1, ref2]);
});

// Test for https://github.com/shaka-project/shaka-player/issues/4185
it('does not fail on preload hints with LL mode off', async () => {
// LL mode must be off for this test!
Expand All @@ -706,6 +758,7 @@ describe('HlsParser live', () => {
'#EXTM3U\n',
'#EXT-X-TARGETDURATION:5\n',
'#EXT-X-MAP:URI="init.mp4",BYTERANGE="616@0"\n',
'#EXT-X-PART-INF:PART-TARGET=1.5\n',
'#EXTINF:4,\n',
'main.mp4\n',
'#EXT-X-PART:DURATION=2,URI="partial.mp4",BYTERANGE=210@0\n',
Expand Down

0 comments on commit 4c8e8e7

Please sign in to comment.