/*! @license * Shaka Player * Copyright 2016 Google LLC * SPDX-License-Identifier: Apache-2.0 */ describe('DashParser SegmentList', () => { const Dash = shaka.test.Dash; const ManifestParser = shaka.test.ManifestParser; const baseUri = 'http://example.com/'; shaka.test.Dash.makeTimelineTests('SegmentList', '', [ '', '', '', '', '', ]); it('truncates segments when lengths don\'t match', async () => { const source = Dash.makeSimpleManifestText([ '', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', '', ], /* duration= */ 65); const references = [ ManifestParser.makeReference('s1.mp4', 50, 60, baseUri), ManifestParser.makeReference('s2.mp4', 60, 65, baseUri), ]; await Dash.testSegmentIndex(source, references); }); it('supports single segment', async () => { const source = Dash.makeSimpleManifestText([ '', ' ', '', ], /* duration= */ 30); const references = [ ManifestParser.makeReference('s1.mp4', 0, 30, baseUri), ]; await Dash.testSegmentIndex(source, references); }); describe('duration', () => { it('basic support', async () => { const source = Dash.makeSimpleManifestText([ '', ' ', ' ', ' ', ' ', '', ]); const references = [ ManifestParser.makeReference('s1.mp4', 0, 10, baseUri), ManifestParser.makeReference('s2.mp4', 10, 20, baseUri), ManifestParser.makeReference('s3.mp4', 20, 30, baseUri), ManifestParser.makeReference('s4.mp4', 30, 40, baseUri), ]; await Dash.testSegmentIndex(source, references); }); it('uses @startNumber correctly', async () => { const source = Dash.makeSimpleManifestText([ '', ' ', ' ', ' ', ' ', '', ]); const references = [ ManifestParser.makeReference('s1.mp4', 40, 50, baseUri), ManifestParser.makeReference('s2.mp4', 50, 60, baseUri), ManifestParser.makeReference('s3.mp4', 60, 70, baseUri), ManifestParser.makeReference('s4.mp4', 70, 80, baseUri), ]; await Dash.testSegmentIndex(source, references); }); it('supports @startNumber=0', async () => { const source = Dash.makeSimpleManifestText([ '', ' ', ' ', '', ]); const references = [ ManifestParser.makeReference('s1.mp4', 0, 10, baseUri), ManifestParser.makeReference('s2.mp4', 10, 20, baseUri), ]; await Dash.testSegmentIndex(source, references); }); it('supports @timescale', async () => { const source = Dash.makeSimpleManifestText([ '', ' ', ' ', ' ', ' ', '', ]); const references = [ ManifestParser.makeReference('s1.mp4', 0, 2, baseUri), ManifestParser.makeReference('s2.mp4', 2, 4, baseUri), ManifestParser.makeReference('s3.mp4', 4, 6, baseUri), ManifestParser.makeReference('s4.mp4', 6, 8, baseUri), ]; await Dash.testSegmentIndex(source, references); }); }); describe('rejects streams with', () => { it('no @duration or SegmentTimeline', async () => { const source = Dash.makeSimpleManifestText([ '', ' ', ' ', ' ', '', ]); const error = new shaka.util.Error( shaka.util.Error.Severity.CRITICAL, shaka.util.Error.Category.MANIFEST, shaka.util.Error.Code.DASH_NO_SEGMENT_INFO); await Dash.testFails(source, error); }); it('one segment and no durations', async () => { const source = Dash.makeSimpleManifestText([ '', ' ', '', ]); const error = new shaka.util.Error( shaka.util.Error.Severity.CRITICAL, shaka.util.Error.Category.MANIFEST, shaka.util.Error.Code.DASH_NO_SEGMENT_INFO); await Dash.testFails(source, error); }); it('empty SegmentTimeline', async () => { const source = Dash.makeSimpleManifestText([ '', ' ', ' ', ' ', '', ]); const error = new shaka.util.Error( shaka.util.Error.Severity.CRITICAL, shaka.util.Error.Category.MANIFEST, shaka.util.Error.Code.DASH_NO_SEGMENT_INFO); await Dash.testFails(source, error); }); }); describe('inherits', () => { it('attributes', async () => { const source = [ '', ' ', ' ', ' ', ' ', ' ', ' http://example.com', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', '', ].join('\n'); const references = [ ManifestParser.makeReference('s1.mp4', 0, 50, baseUri), ManifestParser.makeReference('s2.mp4', 50, 100, baseUri), ManifestParser.makeReference('s3.mp4', 100, 150, baseUri), ManifestParser.makeReference('s4.mp4', 150, 200, baseUri), ]; await Dash.testSegmentIndex(source, references); }); it('SegmentTimeline', async () => { const source = [ '', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' http://example.com', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', '', ].join('\n'); const references = [ ManifestParser.makeReference('s1.mp4', 50, 60, baseUri), ManifestParser.makeReference('s2.mp4', 60, 65, baseUri), ManifestParser.makeReference('s3.mp4', 65, 73, baseUri), ]; await Dash.testSegmentIndex(source, references); }); it('SegmentURL', async () => { const source = [ '', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' http://example.com', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', '', ].join('\n'); const references = [ ManifestParser.makeReference('s1.mp4', 50, 60, baseUri), ManifestParser.makeReference('s2.mp4', 60, 65, baseUri), ManifestParser.makeReference('s3.mp4', 65, 73, baseUri), ]; await Dash.testSegmentIndex(source, references); }); }); describe('Segment start', () => { it('should be adjusted with presentationTimeOffset', async () => { const source = [ '', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' http://example.com', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', '', ].join('\n'); const references = [ ManifestParser.makeReference('s1.mp4', 40, 50, baseUri), ManifestParser.makeReference('s2.mp4', 50, 55, baseUri), ManifestParser.makeReference('s3.mp4', 55, 63, baseUri), ManifestParser.makeReference('s4.mp4', 63, 70, baseUri), ]; for (const ref of references) { ref.timestampOffset = -10; } await Dash.testSegmentIndex(source, references); }); }); // https://github.com/shaka-project/shaka-player/issues/3230 it('works with multi-Period with eviction', async () => { const setFormat = [ ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ].join('\n'); const source = [ '', ` ${baseUri}`, ' ', `${sprintf(setFormat, [0])}`, ' ', ' ', `${sprintf(setFormat, [20])}`, ' ', '', ].join('\n'); const networkingEngine = new shaka.test.FakeNetworkingEngine() .setResponseText('dummy://foo', source); const dashParser = shaka.test.Dash.makeDashParser(); const config = shaka.util.PlayerConfiguration.createDefault(); const playerInterface = { networkingEngine: networkingEngine, modifyManifestRequest: (request, manifestInfo) => {}, modifySegmentRequest: (request, segmentInfo) => {}, filter: () => {}, makeTextStreamsForClosedCaptions: (manifest) => {}, onTimelineRegionAdded: fail, // Should not have any EventStream elements. onEvent: fail, onError: fail, isLowLatencyMode: () => false, updateDuration: () => {}, newDrmInfo: (stream) => {}, onManifestUpdated: () => {}, getBandwidthEstimate: () => 1e6, onMetadata: () => {}, disableStream: (stream) => {}, addFont: (name, url) => {}, getStreamingRetryParameters: () => config.streaming.retryParameters, onSegmentReceived: (deltaTimeMs, numBytes) => {}, }; try { const manifest = await dashParser.start('dummy://foo', playerInterface); const stream = manifest.variants[0].video; await stream.createSegmentIndex(); goog.asserts.assert(stream.segmentIndex, 'Expected index to be created'); // Don't use Dash.testSegmentIndex since it uses SegmentIndex.find, which // doesn't reproduce this issue. We want to use Array.from which uses // the iterator. const expected = [ ManifestParser.makeReference('s1.mp4', 0, 10, baseUri), ManifestParser.makeReference('s2.mp4', 10, 20, baseUri), ManifestParser.makeReference('s3.mp4', 20, 30, baseUri), ManifestParser.makeReference('s3.mp4', 30, 40, baseUri), ManifestParser.makeReference('s4.mp4', 40, 50, baseUri), ManifestParser.makeReference('s5.mp4', 50, 60, baseUri), ]; const actual = Array.from(stream.segmentIndex); expect(actual).toEqual(expected); } finally { dashParser.stop(); } }); });