diff --git a/karma.conf.js b/karma.conf.js index 3c99eebff5..20fbd77ace 100644 --- a/karma.conf.js +++ b/karma.conf.js @@ -257,6 +257,7 @@ module.exports = (config) => { {pattern: 'test/test/assets/dash-multi-codec/*', included: false}, {pattern: 'test/test/assets/dash-multi-codec-ec3/*', included: false}, {pattern: 'test/test/assets/3675/*', included: false}, + {pattern: 'test/test/assets/7401/*', included: false}, {pattern: 'test/test/assets/6339/*', included: false}, {pattern: 'test/test/assets/dash-aes-128/*', included: false}, {pattern: 'test/test/assets/dash-clearkey/*', included: false}, diff --git a/lib/dash/dash_parser.js b/lib/dash/dash_parser.js index 74ef240013..9f2721a9ca 100644 --- a/lib/dash/dash_parser.js +++ b/lib/dash/dash_parser.js @@ -1341,11 +1341,13 @@ shaka.dash.DashParser = class { for (let i = 0; i < periodNodes.length; i++) { const elem = periodNodes[i]; const next = periodNodes[i + 1]; - const start = /** @type {number} */ ( + let start = /** @type {number} */ ( TXml.parseAttr(elem, 'start', TXml.parseDuration, prevEnd)); const periodId = elem.attributes['id']; const givenDuration = TXml.parseAttr(elem, 'duration', TXml.parseDuration); + start = (i == 0 && start == 0 && this.isTransitionFromDynamicToStatic_) ? + seekRangeStart : start; let periodDuration = null; if (next) { @@ -1355,7 +1357,7 @@ shaka.dash.DashParser = class { const nextStart = TXml.parseAttr(next, 'start', TXml.parseDuration); if (nextStart != null) { - periodDuration = nextStart - start; + periodDuration = nextStart - start + seekRangeStart; } } else if (presentationDuration != null) { // "The Period extends until the Period.start of the next Period, or diff --git a/test/player_integration.js b/test/player_integration.js index 799f8d4518..0030cb8de7 100644 --- a/test/player_integration.js +++ b/test/player_integration.js @@ -102,6 +102,46 @@ describe('Player', () => { expect(seekRange.start).toBeCloseTo(start); }); }); + describe('Live to VOD', () => { + it('playback transition when current time is in the past', async () => { + const netEngine = player.getNetworkingEngine(); + const startTime = Date.now(); + netEngine.registerRequestFilter((type, request) => { + if (type != shaka.net.NetworkingEngine.RequestType.MANIFEST) { + return; + } + // Simulate a live stream by providing different manifests over time. + const time = (Date.now() - startTime) / 1000; + const manifestNumber = Math.min(5, Math.floor(0.5 + time / 2)); + request.uris = [ + '/base/test/test/assets/7401/dash_' + manifestNumber + '.mpd', + ]; + console.log('getting manifest', request.uris); + }); + player.configure('streaming.bufferBehind', 1); + player.configure('streaming.evictionGoal', 1); + // Play the stream . + await player.load('/base/test/test/assets/7401/dash_0.mpd', 1020); + await video.play(); + video.pause(); + // Wait for the stream to be over. + eventManager.listen(player, 'error', Util.spyFunc(onErrorSpy)); + /** @type {shaka.test.Waiter} */ + const waiter = new shaka.test.Waiter(eventManager) + .setPlayer(player) + .timeoutAfter(40) + .failOnTimeout(true); + // wait for Dynamic to static + await waiter.waitUntilVodTransition(video); + expect(player.isLive()).toBe(false); + // set the playback to 1020 in middle of the second period + video.currentTime = 1020; + await video.play(); + await waiter.waitForEnd(video); + // The stream should have transitioned to VOD by now. + expect(player.isLive()).toBe(false); + }); + }); describe('attach', () => { beforeEach(async () => { diff --git a/test/test/assets/7401/chunk-stream0-00001.m4s b/test/test/assets/7401/chunk-stream0-00001.m4s new file mode 100644 index 0000000000..b423109cbc Binary files /dev/null and b/test/test/assets/7401/chunk-stream0-00001.m4s differ diff --git a/test/test/assets/7401/chunk-stream0-00002.m4s b/test/test/assets/7401/chunk-stream0-00002.m4s new file mode 100644 index 0000000000..6e805b076b Binary files /dev/null and b/test/test/assets/7401/chunk-stream0-00002.m4s differ diff --git a/test/test/assets/7401/chunk-stream0-00003.m4s b/test/test/assets/7401/chunk-stream0-00003.m4s new file mode 100644 index 0000000000..9a357bb2c7 Binary files /dev/null and b/test/test/assets/7401/chunk-stream0-00003.m4s differ diff --git a/test/test/assets/7401/chunk-stream0-00004.m4s b/test/test/assets/7401/chunk-stream0-00004.m4s new file mode 100644 index 0000000000..de1e8191da Binary files /dev/null and b/test/test/assets/7401/chunk-stream0-00004.m4s differ diff --git a/test/test/assets/7401/chunk-stream0-00005.m4s b/test/test/assets/7401/chunk-stream0-00005.m4s new file mode 100644 index 0000000000..a00d931eea Binary files /dev/null and b/test/test/assets/7401/chunk-stream0-00005.m4s differ diff --git a/test/test/assets/7401/chunk-stream0-00006.m4s b/test/test/assets/7401/chunk-stream0-00006.m4s new file mode 100644 index 0000000000..dce9209c3f Binary files /dev/null and b/test/test/assets/7401/chunk-stream0-00006.m4s differ diff --git a/test/test/assets/7401/chunk-stream0-00007.m4s b/test/test/assets/7401/chunk-stream0-00007.m4s new file mode 100644 index 0000000000..0a30cb49ca Binary files /dev/null and b/test/test/assets/7401/chunk-stream0-00007.m4s differ diff --git a/test/test/assets/7401/dash_0.mpd b/test/test/assets/7401/dash_0.mpd new file mode 100644 index 0000000000..d575921096 --- /dev/null +++ b/test/test/assets/7401/dash_0.mpd @@ -0,0 +1,52 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/test/test/assets/7401/dash_1.mpd b/test/test/assets/7401/dash_1.mpd new file mode 100644 index 0000000000..cc415e0b76 --- /dev/null +++ b/test/test/assets/7401/dash_1.mpd @@ -0,0 +1,52 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/test/test/assets/7401/dash_2.mpd b/test/test/assets/7401/dash_2.mpd new file mode 100644 index 0000000000..e8b5252d75 --- /dev/null +++ b/test/test/assets/7401/dash_2.mpd @@ -0,0 +1,48 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/test/test/assets/7401/init-stream0.m4s b/test/test/assets/7401/init-stream0.m4s new file mode 100644 index 0000000000..ba6f961f21 Binary files /dev/null and b/test/test/assets/7401/init-stream0.m4s differ diff --git a/test/test/util/waiter.js b/test/test/util/waiter.js index 42f6447975..4035a9b489 100644 --- a/test/test/util/waiter.js +++ b/test/test/util/waiter.js @@ -143,6 +143,41 @@ shaka.test.Waiter = class { return this.waitUntilGeneric_(goalName, p, cleanup, mediaElement); } + /** + * Wait for the video player to fetch the static manifest. + * Promise is resolved player.isLive is false. + * + * @param {!HTMLMediaElement} mediaElement + * @return {!Promise} + */ + waitUntilVodTransition(mediaElement) { + // The name of what we're waiting for + const goalName = 'manifest to be static'; + + // The cleanup on timeout + let timer = null; + const cleanup = () => { + if (timer) { + timer.stop(); + } + }; + + // The conditions for success + const p = new Promise((resolve) => { + const check = () => { + if (!this.player_.isLive()) { + cleanup(); + resolve(); + } + }; + + timer = new shaka.util.Timer(check); + timer.tickEvery(/* seconds= */ 1); + }); + + return this.waitUntilGeneric_(goalName, p, cleanup, mediaElement); + } + /** * Wait for the video playhead to reach a certain target time. * If the playhead reaches |timeGoal| or the video ends before |timeout|