1
0
Fork 0
mirror of https://github.com/jellyfin/jellyfin-web synced 2025-03-30 19:56:21 +00:00
jellyfin-web/dashboard-ui/bower_components/hls.js/src/controller/stream-controller.js

1107 lines
42 KiB
JavaScript
Raw Normal View History

2015-12-16 00:30:14 -05:00
/*
2016-02-24 22:15:07 -05:00
* Stream Controller
2015-12-16 00:30:14 -05:00
*/
import Demuxer from '../demux/demuxer';
import Event from '../events';
2016-01-18 14:07:26 -05:00
import EventHandler from '../event-handler';
2015-12-16 00:30:14 -05:00
import {logger} from '../utils/logger';
import BinarySearch from '../utils/binary-search';
2016-03-25 12:35:19 -04:00
import BufferHelper from '../helper/buffer-helper';
2015-12-16 00:30:14 -05:00
import LevelHelper from '../helper/level-helper';
import {ErrorTypes, ErrorDetails} from '../errors';
const State = {
2016-03-17 01:06:37 -04:00
STOPPED : 'STOPPED',
2016-02-24 22:15:07 -05:00
STARTING : 'STARTING',
IDLE : 'IDLE',
PAUSED : 'PAUSED',
KEY_LOADING : 'KEY_LOADING',
FRAG_LOADING : 'FRAG_LOADING',
FRAG_LOADING_WAITING_RETRY : 'FRAG_LOADING_WAITING_RETRY',
WAITING_LEVEL : 'WAITING_LEVEL',
PARSING : 'PARSING',
PARSED : 'PARSED',
2016-03-17 01:06:37 -04:00
ENDED : 'ENDED',
ERROR : 'ERROR'
2015-12-16 00:30:14 -05:00
};
2016-02-24 22:15:07 -05:00
class StreamController extends EventHandler {
2015-12-16 00:30:14 -05:00
constructor(hls) {
2016-02-24 22:15:07 -05:00
super(hls,
Event.MEDIA_ATTACHED,
2016-01-18 14:07:26 -05:00
Event.MEDIA_DETACHING,
2016-03-17 01:06:37 -04:00
Event.MANIFEST_LOADING,
2016-01-18 14:07:26 -05:00
Event.MANIFEST_PARSED,
Event.LEVEL_LOADED,
Event.KEY_LOADED,
Event.FRAG_LOADED,
2016-03-25 12:35:19 -04:00
Event.FRAG_LOAD_EMERGENCY_ABORTED,
2016-01-18 14:07:26 -05:00
Event.FRAG_PARSING_INIT_SEGMENT,
Event.FRAG_PARSING_DATA,
Event.FRAG_PARSED,
2016-02-24 22:15:07 -05:00
Event.ERROR,
Event.BUFFER_APPENDED,
Event.BUFFER_FLUSHED);
2015-12-16 00:30:14 -05:00
this.config = hls.config;
this.audioCodecSwap = false;
2016-01-13 15:58:12 -05:00
this.ticks = 0;
2015-12-16 00:30:14 -05:00
this.ontick = this.tick.bind(this);
}
destroy() {
2016-03-17 01:06:37 -04:00
this.stopLoad();
if (this.timer) {
clearInterval(this.timer);
this.timer = null;
}
2016-01-18 14:07:26 -05:00
EventHandler.prototype.destroy.call(this);
2016-03-17 01:06:37 -04:00
this.state = State.STOPPED;
2015-12-16 00:30:14 -05:00
}
2016-03-17 01:06:37 -04:00
startLoad(startPosition=0) {
2016-02-03 18:00:01 -05:00
if (this.levels) {
var media = this.media, lastCurrentTime = this.lastCurrentTime;
2016-03-17 01:06:37 -04:00
this.stopLoad();
2016-02-24 22:15:07 -05:00
this.demuxer = new Demuxer(this.hls);
2016-03-17 01:06:37 -04:00
if (!this.timer) {
this.timer = setInterval(this.ontick, 100);
}
2016-02-24 22:15:07 -05:00
this.level = -1;
this.fragLoadError = 0;
2016-02-03 18:00:01 -05:00
if (media && lastCurrentTime) {
2016-02-24 22:15:07 -05:00
logger.log(`configure startPosition @${lastCurrentTime}`);
2015-12-16 00:30:14 -05:00
if (!this.lastPaused) {
logger.log('resuming video');
2016-02-03 18:00:01 -05:00
media.play();
2015-12-16 00:30:14 -05:00
}
this.state = State.IDLE;
} else {
2016-03-17 01:06:37 -04:00
this.lastCurrentTime = this.startPosition ? this.startPosition : startPosition;
2015-12-16 00:30:14 -05:00
this.state = State.STARTING;
}
this.nextLoadPosition = this.startPosition = this.lastCurrentTime;
this.tick();
} else {
2016-02-03 18:00:01 -05:00
logger.warn('cannot start loading as manifest not parsed yet');
2016-03-17 01:06:37 -04:00
this.state = State.STOPPED;
2015-12-16 00:30:14 -05:00
}
}
2016-03-17 01:06:37 -04:00
stopLoad() {
2015-12-16 00:30:14 -05:00
var frag = this.fragCurrent;
if (frag) {
if (frag.loader) {
frag.loader.abort();
}
this.fragCurrent = null;
}
this.fragPrevious = null;
if (this.demuxer) {
this.demuxer.destroy();
this.demuxer = null;
}
2016-03-17 01:06:37 -04:00
this.state = State.STOPPED;
2015-12-16 00:30:14 -05:00
}
tick() {
2016-01-13 15:58:12 -05:00
this.ticks++;
if (this.ticks === 1) {
this.doTick();
if (this.ticks > 1) {
setTimeout(this.tick, 1);
}
this.ticks = 0;
}
}
doTick() {
2016-02-24 22:15:07 -05:00
var pos, level, levelDetails, hls = this.hls, config = hls.config;
//logger.log(this.state);
2015-12-16 00:30:14 -05:00
switch(this.state) {
case State.ERROR:
//don't do anything in error state to avoid breaking further ...
2016-02-24 22:15:07 -05:00
case State.PAUSED:
//don't do anything in paused state either ...
2015-12-16 00:30:14 -05:00
break;
case State.STARTING:
// determine load level
this.startLevel = hls.startLevel;
if (this.startLevel === -1) {
// -1 : guess start Level by doing a bitrate test by loading first fragment of lowest quality level
this.startLevel = 0;
this.fragBitrateTest = true;
}
// set new level to playlist loader : this will trigger start level load
this.level = hls.nextLoadLevel = this.startLevel;
this.state = State.WAITING_LEVEL;
this.loadedmetadata = false;
break;
case State.IDLE:
2016-02-24 22:15:07 -05:00
// if video not attached AND
// start fragment already requested OR start frag prefetch disable
// exit loop
// => if media not attached but start frag prefetch is enabled and start frag not requested yet, we will not exit loop
if (!this.media &&
(this.startFragRequested || !config.startFragPrefetch)) {
2015-12-16 00:30:14 -05:00
break;
}
// determine next candidate fragment to be loaded, based on current position and
// end of buffer position
// ensure 60s of buffer upfront
// if we have not yet loaded any fragment, start loading from start position
if (this.loadedmetadata) {
pos = this.media.currentTime;
} else {
pos = this.nextLoadPosition;
}
// determine next load level
2016-02-24 22:15:07 -05:00
if (this.startFragRequested === false) {
2015-12-16 00:30:14 -05:00
level = this.startLevel;
} else {
// we are not at playback start, get next load level from level Controller
level = hls.nextLoadLevel;
}
2016-03-25 12:35:19 -04:00
var bufferInfo = BufferHelper.bufferInfo(this.media,pos,config.maxBufferHole),
2015-12-16 00:30:14 -05:00
bufferLen = bufferInfo.len,
bufferEnd = bufferInfo.end,
fragPrevious = this.fragPrevious,
maxBufLen;
// compute max Buffer Length that we could get from this load level, based on level bitrate. don't buffer more than 60 MB and more than 30s
if ((this.levels[level]).hasOwnProperty('bitrate')) {
2016-02-24 22:15:07 -05:00
maxBufLen = Math.max(8 * config.maxBufferSize / this.levels[level].bitrate, config.maxBufferLength);
maxBufLen = Math.min(maxBufLen, config.maxMaxBufferLength);
2015-12-16 00:30:14 -05:00
} else {
2016-02-24 22:15:07 -05:00
maxBufLen = config.maxBufferLength;
2015-12-16 00:30:14 -05:00
}
// if buffer length is less than maxBufLen try to load a new fragment
if (bufferLen < maxBufLen) {
// set next load level : this will trigger a playlist load if needed
hls.nextLoadLevel = level;
this.level = level;
levelDetails = this.levels[level].details;
// if level info not retrieved yet, switch state and wait for level retrieval
2016-01-18 14:07:26 -05:00
// if live playlist, ensure that new playlist has been refreshed to avoid loading/try to load
// a useless and outdated fragment (that might even introduce load error if it is already out of the live playlist)
if (typeof levelDetails === 'undefined' || levelDetails.live && this.levelLastLoaded !== level) {
2015-12-16 00:30:14 -05:00
this.state = State.WAITING_LEVEL;
break;
}
// find fragment index, contiguous with end of buffer position
let fragments = levelDetails.fragments,
fragLen = fragments.length,
start = fragments[0].start,
end = fragments[fragLen-1].start + fragments[fragLen-1].duration,
frag;
// in case of live playlist we need to ensure that requested position is not located before playlist start
if (levelDetails.live) {
// check if requested position is within seekable boundaries :
//logger.log(`start/pos/bufEnd/seeking:${start.toFixed(3)}/${pos.toFixed(3)}/${bufferEnd.toFixed(3)}/${this.media.seeking}`);
2016-03-09 12:40:22 -05:00
let maxLatency = config.liveMaxLatencyDuration !== undefined ? config.liveMaxLatencyDuration : config.liveMaxLatencyDurationCount*levelDetails.targetduration;
if (bufferEnd < Math.max(start, end - maxLatency)) {
let targetLatency = config.liveSyncDuration !== undefined ? config.liveSyncDuration : config.liveSyncDurationCount * levelDetails.targetduration;
this.seekAfterBuffered = start + Math.max(0, levelDetails.totalduration - targetLatency);
2015-12-16 00:30:14 -05:00
logger.log(`buffer end: ${bufferEnd} is located too far from the end of live sliding playlist, media position will be reseted to: ${this.seekAfterBuffered.toFixed(3)}`);
bufferEnd = this.seekAfterBuffered;
}
2016-04-15 15:20:04 -04:00
// if end of buffer greater than live edge, don't load any fragment
// this could happen if live playlist intermittently slides in the past.
// level 1 loaded [182580161,182580167]
// level 1 loaded [182580162,182580169]
// Loading 182580168 of [182580162 ,182580169],level 1 ..
// Loading 182580169 of [182580162 ,182580169],level 1 ..
// level 1 loaded [182580162,182580168] <============= here we should have bufferEnd > end. in that case break to avoid reloading 182580168
// level 1 loaded [182580164,182580171]
//
if (bufferEnd > end) {
break;
}
2016-02-24 22:15:07 -05:00
if (this.startFragRequested && !levelDetails.PTSKnown) {
2015-12-16 00:30:14 -05:00
/* we are switching level on live playlist, but we don't have any PTS info for that quality level ...
try to load frag matching with next SN.
even if SN are not synchronized between playlists, loading this frag will help us
compute playlist sliding and find the right one after in case it was not the right consecutive one */
if (fragPrevious) {
var targetSN = fragPrevious.sn + 1;
if (targetSN >= levelDetails.startSN && targetSN <= levelDetails.endSN) {
frag = fragments[targetSN - levelDetails.startSN];
logger.log(`live playlist, switching playlist, load frag with next SN: ${frag.sn}`);
}
}
if (!frag) {
/* we have no idea about which fragment should be loaded.
so let's load mid fragment. it will help computing playlist sliding and find the right one
*/
frag = fragments[Math.min(fragLen - 1, Math.round(fragLen / 2))];
logger.log(`live playlist, switching playlist, unknown, load middle frag : ${frag.sn}`);
}
}
} else {
// VoD playlist: if bufferEnd before start of playlist, load first fragment
if (bufferEnd < start) {
frag = fragments[0];
}
}
if (!frag) {
2016-03-16 13:43:01 -04:00
let foundFrag;
let maxFragLookUpTolerance = config.maxFragLookUpTolerance;
2015-12-16 00:30:14 -05:00
if (bufferEnd < end) {
2016-03-16 13:43:01 -04:00
if (bufferEnd > end - maxFragLookUpTolerance) {
maxFragLookUpTolerance = 0;
}
2015-12-16 00:30:14 -05:00
foundFrag = BinarySearch.search(fragments, (candidate) => {
2016-03-16 13:43:01 -04:00
// offset should be within fragment boundary - config.maxFragLookUpTolerance
// this is to cope with situations like
// bufferEnd = 9.991
// frag[Ø] : [0,10]
// frag[1] : [10,20]
// bufferEnd is within frag[0] range ... although what we are expecting is to return frag[1] here
// frag start frag start+duration
// |-----------------------------|
// <---> <--->
// ...--------><-----------------------------><---------....
// previous frag matching fragment next frag
// return -1 return 0 return 1
2015-12-16 00:30:14 -05:00
//logger.log(`level/sn/start/end/bufEnd:${level}/${candidate.sn}/${candidate.start}/${(candidate.start+candidate.duration)}/${bufferEnd}`);
2016-03-16 13:43:01 -04:00
if ((candidate.start + candidate.duration - maxFragLookUpTolerance) <= bufferEnd) {
2015-12-16 00:30:14 -05:00
return 1;
}
2016-03-16 13:43:01 -04:00
else if (candidate.start - maxFragLookUpTolerance > bufferEnd) {
2015-12-16 00:30:14 -05:00
return -1;
}
return 0;
});
} else {
// reach end of playlist
foundFrag = fragments[fragLen-1];
}
if (foundFrag) {
frag = foundFrag;
start = foundFrag.start;
//logger.log('find SN matching with pos:' + bufferEnd + ':' + frag.sn);
if (fragPrevious && frag.level === fragPrevious.level && frag.sn === fragPrevious.sn) {
if (frag.sn < levelDetails.endSN) {
frag = fragments[frag.sn + 1 - levelDetails.startSN];
logger.log(`SN just loaded, load next one: ${frag.sn}`);
} else {
// have we reached end of VOD playlist ?
if (!levelDetails.live) {
2016-02-24 22:15:07 -05:00
this.hls.trigger(Event.BUFFER_EOS);
this.state = State.ENDED;
2015-12-16 00:30:14 -05:00
}
frag = null;
}
}
}
}
if(frag) {
//logger.log(' loading frag ' + i +',pos/bufEnd:' + pos.toFixed(3) + '/' + bufferEnd.toFixed(3));
if ((frag.decryptdata.uri != null) && (frag.decryptdata.key == null)) {
logger.log(`Loading key for ${frag.sn} of [${levelDetails.startSN} ,${levelDetails.endSN}],level ${level}`);
this.state = State.KEY_LOADING;
hls.trigger(Event.KEY_LOADING, {frag: frag});
} else {
logger.log(`Loading ${frag.sn} of [${levelDetails.startSN} ,${levelDetails.endSN}],level ${level}, currentTime:${pos},bufferEnd:${bufferEnd.toFixed(3)}`);
frag.autoLevel = hls.autoLevelEnabled;
if (this.levels.length > 1) {
frag.expectedLen = Math.round(frag.duration * this.levels[level].bitrate / 8);
frag.trequest = performance.now();
}
// ensure that we are not reloading the same fragments in loop ...
if (this.fragLoadIdx !== undefined) {
this.fragLoadIdx++;
} else {
this.fragLoadIdx = 0;
}
if (frag.loadCounter) {
frag.loadCounter++;
2016-02-24 22:15:07 -05:00
let maxThreshold = config.fragLoadingLoopThreshold;
2015-12-16 00:30:14 -05:00
// if this frag has already been loaded 3 times, and if it has been reloaded recently
if (frag.loadCounter > maxThreshold && (Math.abs(this.fragLoadIdx - frag.loadIdx) < maxThreshold)) {
hls.trigger(Event.ERROR, {type: ErrorTypes.MEDIA_ERROR, details: ErrorDetails.FRAG_LOOP_LOADING_ERROR, fatal: false, frag: frag});
return;
}
} else {
frag.loadCounter = 1;
}
frag.loadIdx = this.fragLoadIdx;
this.fragCurrent = frag;
2016-02-24 22:15:07 -05:00
this.startFragRequested = true;
2015-12-16 00:30:14 -05:00
hls.trigger(Event.FRAG_LOADING, {frag: frag});
this.state = State.FRAG_LOADING;
}
}
}
break;
case State.WAITING_LEVEL:
level = this.levels[this.level];
// check if playlist is already loaded
if (level && level.details) {
this.state = State.IDLE;
}
break;
2016-01-13 15:58:12 -05:00
case State.FRAG_LOADING_WAITING_RETRY:
var now = performance.now();
var retryDate = this.retryDate;
var media = this.media;
var isSeeking = media && media.seeking;
// if current time is gt than retryDate, or if media seeking let's switch to IDLE state to retry loading
if(!retryDate || (now >= retryDate) || isSeeking) {
logger.log(`mediaController: retryDate reached, switch back to IDLE state`);
this.state = State.IDLE;
}
break;
2016-03-25 12:35:19 -04:00
case State.STOPPED:
case State.FRAG_LOADING:
2015-12-16 00:30:14 -05:00
case State.PARSING:
case State.PARSED:
2016-01-25 15:28:29 -05:00
case State.ENDED:
break;
2015-12-16 00:30:14 -05:00
default:
break;
}
// check buffer
this._checkBuffer();
2016-01-13 15:58:12 -05:00
// check/update current fragment
this._checkFragmentChanged();
2015-12-16 00:30:14 -05:00
}
getBufferRange(position) {
2016-03-09 12:40:22 -05:00
var i, range,
bufferRange = this.bufferRange;
if (bufferRange) {
for (i = bufferRange.length - 1; i >=0; i--) {
range = bufferRange[i];
if (position >= range.start && position <= range.end) {
return range;
}
2015-12-16 00:30:14 -05:00
}
}
return null;
}
get currentLevel() {
if (this.media) {
var range = this.getBufferRange(this.media.currentTime);
if (range) {
return range.frag.level;
}
}
return -1;
}
get nextBufferRange() {
if (this.media) {
// first get end range of current fragment
return this.followingBufferRange(this.getBufferRange(this.media.currentTime));
} else {
return null;
}
}
followingBufferRange(range) {
if (range) {
// try to get range of next fragment (500ms after this range)
return this.getBufferRange(range.end + 0.5);
}
return null;
}
get nextLevel() {
var range = this.nextBufferRange;
if (range) {
return range.frag.level;
} else {
return -1;
}
}
isBuffered(position) {
var v = this.media, buffered = v.buffered;
for (var i = 0; i < buffered.length; i++) {
if (position >= buffered.start(i) && position <= buffered.end(i)) {
return true;
}
}
return false;
}
_checkFragmentChanged() {
var rangeCurrent, currentTime, video = this.media;
if (video && video.seeking === false) {
currentTime = video.currentTime;
/* if video element is in seeked state, currentTime can only increase.
(assuming that playback rate is positive ...)
As sometimes currentTime jumps back to zero after a
media decode error, check this, to avoid seeking back to
wrong position after a media decode error
*/
if(currentTime > video.playbackRate*this.lastCurrentTime) {
this.lastCurrentTime = currentTime;
}
if (this.isBuffered(currentTime)) {
rangeCurrent = this.getBufferRange(currentTime);
} else if (this.isBuffered(currentTime + 0.1)) {
/* ensure that FRAG_CHANGED event is triggered at startup,
when first video frame is displayed and playback is paused.
add a tolerance of 100ms, in case current position is not buffered,
check if current pos+100ms is buffered and use that buffer range
for FRAG_CHANGED event reporting */
rangeCurrent = this.getBufferRange(currentTime + 0.1);
}
if (rangeCurrent) {
var fragPlaying = rangeCurrent.frag;
if (fragPlaying !== this.fragPlaying) {
this.fragPlaying = fragPlaying;
this.hls.trigger(Event.FRAG_CHANGED, {frag: fragPlaying});
}
}
}
}
/*
on immediate level switch :
- pause playback if playing
- cancel any pending load request
- and trigger a buffer flush
*/
immediateLevelSwitch() {
logger.log('immediateLevelSwitch');
if (!this.immediateSwitch) {
this.immediateSwitch = true;
this.previouslyPaused = this.media.paused;
this.media.pause();
}
var fragCurrent = this.fragCurrent;
if (fragCurrent && fragCurrent.loader) {
fragCurrent.loader.abort();
}
this.fragCurrent = null;
// flush everything
2016-02-24 22:15:07 -05:00
this.hls.trigger(Event.BUFFER_FLUSHING, {startOffset: 0, endOffset: Number.POSITIVE_INFINITY});
this.state = State.PAUSED;
2015-12-16 00:30:14 -05:00
// increase fragment load Index to avoid frag loop loading error after buffer flush
this.fragLoadIdx += 2 * this.config.fragLoadingLoopThreshold;
// speed up switching, trigger timer function
this.tick();
}
/*
on immediate level switch end, after new fragment has been buffered :
- nudge video decoder by slightly adjusting video currentTime
- resume the playback if needed
*/
immediateLevelSwitchEnd() {
this.immediateSwitch = false;
this.media.currentTime -= 0.0001;
if (!this.previouslyPaused) {
this.media.play();
}
}
nextLevelSwitch() {
/* try to switch ASAP without breaking video playback :
in order to ensure smooth but quick level switching,
we need to find the next flushable buffer range
we should take into account new segment fetch time
*/
var fetchdelay, currentRange, nextRange;
currentRange = this.getBufferRange(this.media.currentTime);
2016-02-24 22:15:07 -05:00
if (currentRange && currentRange.start > 1) {
2015-12-16 00:30:14 -05:00
// flush buffer preceding current fragment (flush until current fragment start offset)
// minus 1s to avoid video freezing, that could happen if we flush keyframe of current video ...
2016-02-24 22:15:07 -05:00
this.hls.trigger(Event.BUFFER_FLUSHING, {startOffset: 0, endOffset: currentRange.start - 1});
this.state = State.PAUSED;
2015-12-16 00:30:14 -05:00
}
if (!this.media.paused) {
// add a safety delay of 1s
var nextLevelId = this.hls.nextLoadLevel,nextLevel = this.levels[nextLevelId], fragLastKbps = this.fragLastKbps;
if (fragLastKbps && this.fragCurrent) {
fetchdelay = this.fragCurrent.duration * nextLevel.bitrate / (1000 * fragLastKbps) + 1;
} else {
fetchdelay = 0;
}
} else {
fetchdelay = 0;
}
//logger.log('fetchdelay:'+fetchdelay);
// find buffer range that will be reached once new fragment will be fetched
nextRange = this.getBufferRange(this.media.currentTime + fetchdelay);
if (nextRange) {
// we can flush buffer range following this one without stalling playback
nextRange = this.followingBufferRange(nextRange);
if (nextRange) {
// flush position is the start position of this new buffer
2016-02-24 22:15:07 -05:00
this.hls.trigger(Event.BUFFER_FLUSHING, {startOffset: nextRange.start, endOffset: Number.POSITIVE_INFINITY});
this.state = State.PAUSED;
2015-12-16 00:30:14 -05:00
// if we are here, we can also cancel any loading/demuxing in progress, as they are useless
var fragCurrent = this.fragCurrent;
if (fragCurrent && fragCurrent.loader) {
fragCurrent.loader.abort();
}
this.fragCurrent = null;
2016-02-24 22:15:07 -05:00
// increase fragment load Index to avoid frag loop loading error after buffer flush
this.fragLoadIdx += 2 * this.config.fragLoadingLoopThreshold;
2015-12-16 00:30:14 -05:00
}
}
}
2016-02-24 22:15:07 -05:00
onMediaAttached(data) {
2015-12-16 00:30:14 -05:00
var media = this.media = data.media;
2016-02-24 22:15:07 -05:00
this.onvseeking = this.onMediaSeeking.bind(this);
this.onvseeked = this.onMediaSeeked.bind(this);
this.onvended = this.onMediaEnded.bind(this);
media.addEventListener('seeking', this.onvseeking);
media.addEventListener('seeked', this.onvseeked);
media.addEventListener('ended', this.onvended);
if(this.levels && this.config.autoStartLoad) {
2016-03-18 13:28:45 -04:00
this.hls.startLoad();
2016-02-24 22:15:07 -05:00
}
2015-12-16 00:30:14 -05:00
}
onMediaDetaching() {
var media = this.media;
if (media && media.ended) {
logger.log('MSE detaching and video ended, reset startPosition');
this.startPosition = this.lastCurrentTime = 0;
}
// reset fragment loading counter on MSE detaching to avoid reporting FRAG_LOOP_LOADING_ERROR after error recovery
var levels = this.levels;
if (levels) {
// reset fragment load counter
levels.forEach(level => {
if(level.details) {
level.details.fragments.forEach(fragment => {
fragment.loadCounter = undefined;
});
}
});
}
2016-02-24 22:15:07 -05:00
// remove video listeners
if (media) {
media.removeEventListener('seeking', this.onvseeking);
media.removeEventListener('seeked', this.onvseeked);
media.removeEventListener('ended', this.onvended);
this.onvseeking = this.onvseeked = this.onvended = null;
2015-12-16 00:30:14 -05:00
}
2016-02-24 22:15:07 -05:00
this.media = null;
this.loadedmetadata = false;
2016-03-17 01:06:37 -04:00
this.stopLoad();
2015-12-16 00:30:14 -05:00
}
onMediaSeeking() {
if (this.state === State.FRAG_LOADING) {
// check if currently loaded fragment is inside buffer.
//if outside, cancel fragment loading, otherwise do nothing
2016-03-25 12:35:19 -04:00
if (BufferHelper.bufferInfo(this.media,this.media.currentTime,this.config.maxBufferHole).len === 0) {
2015-12-16 00:30:14 -05:00
logger.log('seeking outside of buffer while fragment load in progress, cancel fragment load');
var fragCurrent = this.fragCurrent;
if (fragCurrent) {
if (fragCurrent.loader) {
fragCurrent.loader.abort();
}
this.fragCurrent = null;
}
this.fragPrevious = null;
// switch to IDLE state to load new fragment
this.state = State.IDLE;
}
2016-01-25 15:28:29 -05:00
} else if (this.state === State.ENDED) {
// switch to IDLE state to check for potential new fragment
this.state = State.IDLE;
2015-12-16 00:30:14 -05:00
}
if (this.media) {
this.lastCurrentTime = this.media.currentTime;
}
// avoid reporting fragment loop loading error in case user is seeking several times on same position
if (this.fragLoadIdx !== undefined) {
this.fragLoadIdx += 2 * this.config.fragLoadingLoopThreshold;
}
// tick to speed up processing
this.tick();
}
onMediaSeeked() {
// tick to speed up FRAGMENT_PLAYING triggering
this.tick();
}
onMediaEnded() {
logger.log('media ended');
// reset startPosition and lastCurrentTime to restart playback @ stream beginning
this.startPosition = this.lastCurrentTime = 0;
}
2016-03-17 01:06:37 -04:00
onManifestLoading() {
// reset buffer on manifest loading
logger.log('trigger BUFFER_RESET');
this.hls.trigger(Event.BUFFER_RESET);
this.bufferRange = [];
this.stalled = false;
}
2016-01-18 14:07:26 -05:00
onManifestParsed(data) {
2016-02-24 22:15:07 -05:00
var aac = false, heaac = false, codec;
2015-12-16 00:30:14 -05:00
data.levels.forEach(level => {
// detect if we have different kind of audio codecs used amongst playlists
2016-02-24 22:15:07 -05:00
codec = level.audioCodec;
if (codec) {
if (codec.indexOf('mp4a.40.2') !== -1) {
2015-12-16 00:30:14 -05:00
aac = true;
}
2016-02-24 22:15:07 -05:00
if (codec.indexOf('mp4a.40.5') !== -1) {
2015-12-16 00:30:14 -05:00
heaac = true;
}
}
});
2016-02-24 22:15:07 -05:00
this.audioCodecSwitch = (aac && heaac);
if (this.audioCodecSwitch) {
logger.log('both AAC/HE-AAC audio found in levels; declaring level codec as HE-AAC');
2015-12-16 00:30:14 -05:00
}
this.levels = data.levels;
this.startLevelLoaded = false;
2016-02-24 22:15:07 -05:00
this.startFragRequested = false;
if (this.config.autoStartLoad) {
2016-03-18 13:28:45 -04:00
this.hls.startLoad();
2015-12-16 00:30:14 -05:00
}
}
2016-01-18 14:07:26 -05:00
onLevelLoaded(data) {
2015-12-16 00:30:14 -05:00
var newDetails = data.details,
newLevelId = data.level,
curLevel = this.levels[newLevelId],
2016-02-24 22:15:07 -05:00
duration = newDetails.totalduration,
sliding = 0;
2015-12-16 00:30:14 -05:00
logger.log(`level ${newLevelId} loaded [${newDetails.startSN},${newDetails.endSN}],duration:${duration}`);
2016-01-18 14:07:26 -05:00
this.levelLastLoaded = newLevelId;
2015-12-16 00:30:14 -05:00
if (newDetails.live) {
var curDetails = curLevel.details;
if (curDetails) {
// we already have details for that level, merge them
LevelHelper.mergeDetails(curDetails,newDetails);
2016-02-24 22:15:07 -05:00
sliding = newDetails.fragments[0].start;
2015-12-16 00:30:14 -05:00
if (newDetails.PTSKnown) {
2016-02-24 22:15:07 -05:00
logger.log(`live playlist sliding:${sliding.toFixed(3)}`);
2015-12-16 00:30:14 -05:00
} else {
logger.log('live playlist - outdated PTS, unknown sliding');
}
} else {
newDetails.PTSKnown = false;
logger.log('live playlist - first load, unknown sliding');
}
} else {
newDetails.PTSKnown = false;
}
// override level info
curLevel.details = newDetails;
this.hls.trigger(Event.LEVEL_UPDATED, { details: newDetails, level: newLevelId });
// compute start position
2016-02-24 22:15:07 -05:00
if (this.startFragRequested === false) {
2015-12-16 00:30:14 -05:00
// if live playlist, set start position to be fragment N-this.config.liveSyncDurationCount (usually 3)
if (newDetails.live) {
2016-03-09 12:40:22 -05:00
let targetLatency = this.config.liveSyncDuration !== undefined ? this.config.liveSyncDuration : this.config.liveSyncDurationCount * newDetails.targetduration;
this.startPosition = Math.max(0, sliding + duration - targetLatency);
2015-12-16 00:30:14 -05:00
}
this.nextLoadPosition = this.startPosition;
}
// only switch batck to IDLE state if we were waiting for level to start downloading a new fragment
if (this.state === State.WAITING_LEVEL) {
this.state = State.IDLE;
}
//trigger handler right now
this.tick();
}
onKeyLoaded() {
if (this.state === State.KEY_LOADING) {
this.state = State.IDLE;
this.tick();
}
}
2016-01-18 14:07:26 -05:00
onFragLoaded(data) {
2015-12-16 00:30:14 -05:00
var fragCurrent = this.fragCurrent;
if (this.state === State.FRAG_LOADING &&
fragCurrent &&
data.frag.level === fragCurrent.level &&
data.frag.sn === fragCurrent.sn) {
if (this.fragBitrateTest === true) {
// switch back to IDLE state ... we just loaded a fragment to determine adequate start bitrate and initialize autoswitch algo
this.state = State.IDLE;
this.fragBitrateTest = false;
data.stats.tparsed = data.stats.tbuffered = performance.now();
this.hls.trigger(Event.FRAG_BUFFERED, {stats: data.stats, frag: fragCurrent});
} else {
this.state = State.PARSING;
// transmux the MPEG-TS data to ISO-BMFF segments
this.stats = data.stats;
var currentLevel = this.levels[this.level],
details = currentLevel.details,
duration = details.totalduration,
start = fragCurrent.start,
level = fragCurrent.level,
sn = fragCurrent.sn,
2016-02-24 22:15:07 -05:00
audioCodec = currentLevel.audioCodec || this.config.defaultAudioCodec;
2016-01-13 15:58:12 -05:00
if(this.audioCodecSwap) {
2015-12-16 00:30:14 -05:00
logger.log('swapping playlist audio codec');
2016-01-13 15:58:12 -05:00
if(audioCodec === undefined) {
audioCodec = this.lastAudioCodec;
}
2016-02-24 22:15:07 -05:00
if(audioCodec) {
if(audioCodec.indexOf('mp4a.40.5') !==-1) {
audioCodec = 'mp4a.40.2';
} else {
audioCodec = 'mp4a.40.5';
}
2015-12-16 00:30:14 -05:00
}
}
2016-02-24 22:15:07 -05:00
this.pendingAppending = 0;
2015-12-16 00:30:14 -05:00
logger.log(`Demuxing ${sn} of [${details.startSN} ,${details.endSN}],level ${level}`);
this.demuxer.push(data.payload, audioCodec, currentLevel.videoCodec, start, fragCurrent.cc, level, sn, duration, fragCurrent.decryptdata);
}
}
2015-12-23 12:46:01 -05:00
this.fragLoadError = 0;
2015-12-16 00:30:14 -05:00
}
2016-01-18 14:07:26 -05:00
onFragParsingInitSegment(data) {
2015-12-16 00:30:14 -05:00
if (this.state === State.PARSING) {
2016-02-24 22:15:07 -05:00
var tracks = data.tracks, trackName, track;
// include levelCodec in audio and video tracks
track = tracks.audio;
if(track) {
2016-03-10 12:59:32 -05:00
var audioCodec = this.levels[this.level].audioCodec,
ua = navigator.userAgent.toLowerCase();
2016-02-24 22:15:07 -05:00
if(audioCodec && this.audioCodecSwap) {
logger.log('swapping playlist audio codec');
if(audioCodec.indexOf('mp4a.40.5') !==-1) {
audioCodec = 'mp4a.40.2';
} else {
audioCodec = 'mp4a.40.5';
}
}
// in case AAC and HE-AAC audio codecs are signalled in manifest
// force HE-AAC , as it seems that most browsers prefers that way,
2016-03-10 12:59:32 -05:00
// except for mono streams OR on FF
2016-02-24 22:15:07 -05:00
// these conditions might need to be reviewed ...
if (this.audioCodecSwitch) {
// don't force HE-AAC if mono stream
if(track.metadata.channelCount !== 1 &&
// don't force HE-AAC if firefox
ua.indexOf('firefox') === -1) {
audioCodec = 'mp4a.40.5';
}
2015-12-16 00:30:14 -05:00
}
2016-03-10 12:59:32 -05:00
// HE-AAC is broken on Android, always signal audio codec as AAC even if variant manifest states otherwise
if(ua.indexOf('android') !== -1) {
audioCodec = 'mp4a.40.2';
logger.log(`Android: force audio codec to` + audioCodec);
}
2016-02-24 22:15:07 -05:00
track.levelCodec = audioCodec;
2015-12-16 00:30:14 -05:00
}
2016-02-24 22:15:07 -05:00
track = tracks.video;
if(track) {
track.levelCodec = this.levels[this.level].videoCodec;
2015-12-16 00:30:14 -05:00
}
2016-02-24 22:15:07 -05:00
// if remuxer specify that a unique track needs to generated,
// let's merge all tracks together
if (data.unique) {
var mergedTrack = {
codec : '',
levelCodec : ''
};
for (trackName in data.tracks) {
track = tracks[trackName];
mergedTrack.container = track.container;
if (mergedTrack.codec) {
mergedTrack.codec += ',';
mergedTrack.levelCodec += ',';
}
if(track.codec) {
mergedTrack.codec += track.codec;
}
if (track.levelCodec) {
mergedTrack.levelCodec += track.levelCodec;
}
2015-12-16 00:30:14 -05:00
}
2016-02-24 22:15:07 -05:00
tracks = { audiovideo : mergedTrack };
2015-12-16 00:30:14 -05:00
}
2016-02-24 22:15:07 -05:00
this.hls.trigger(Event.BUFFER_CODECS,tracks);
// loop through tracks that are going to be provided to bufferController
for (trackName in tracks) {
track = tracks[trackName];
logger.log(`track:${trackName},container:${track.container},codecs[level/parsed]=[${track.levelCodec}/${track.codec}]`);
var initSegment = track.initSegment;
if (initSegment) {
this.pendingAppending++;
this.hls.trigger(Event.BUFFER_APPENDING, {type: trackName, data: initSegment});
}
2015-12-16 00:30:14 -05:00
}
//trigger handler right now
this.tick();
}
}
2016-01-18 14:07:26 -05:00
onFragParsingData(data) {
2015-12-16 00:30:14 -05:00
if (this.state === State.PARSING) {
this.tparse2 = Date.now();
var level = this.levels[this.level],
frag = this.fragCurrent;
2016-02-24 22:15:07 -05:00
2016-01-13 15:58:12 -05:00
logger.log(`parsed ${data.type},PTS:[${data.startPTS.toFixed(3)},${data.endPTS.toFixed(3)}],DTS:[${data.startDTS.toFixed(3)}/${data.endDTS.toFixed(3)}],nb:${data.nb}`);
2015-12-16 00:30:14 -05:00
2016-02-24 22:15:07 -05:00
var drift = LevelHelper.updateFragPTS(level.details,frag.sn,data.startPTS,data.endPTS),
hls = this.hls;
hls.trigger(Event.LEVEL_PTS_UPDATED, {details: level.details, level: this.level, drift: drift});
[data.data1, data.data2].forEach(buffer => {
if (buffer) {
this.pendingAppending++;
hls.trigger(Event.BUFFER_APPENDING, {type: data.type, data: buffer});
}
});
2015-12-16 00:30:14 -05:00
this.nextLoadPosition = data.endPTS;
this.bufferRange.push({type: data.type, start: data.startPTS, end: data.endPTS, frag: frag});
//trigger handler right now
this.tick();
} else {
2016-02-24 22:15:07 -05:00
logger.warn(`not in PARSING state but ${this.state}, ignoring FRAG_PARSING_DATA event`);
2015-12-16 00:30:14 -05:00
}
}
onFragParsed() {
if (this.state === State.PARSING) {
this.stats.tparsed = performance.now();
2016-02-24 22:15:07 -05:00
this.state = State.PARSED;
this._checkAppendedParsed();
}
}
2016-02-04 13:19:10 -05:00
2016-02-24 22:15:07 -05:00
onBufferAppended() {
switch (this.state) {
case State.PARSING:
case State.PARSED:
this.pendingAppending--;
this._checkAppendedParsed();
break;
default:
break;
}
}
2016-02-04 13:19:10 -05:00
2016-02-24 22:15:07 -05:00
_checkAppendedParsed() {
//trigger handler right now
if (this.state === State.PARSED && this.pendingAppending === 0) {
var frag = this.fragCurrent, stats = this.stats;
if (frag) {
this.fragPrevious = frag;
stats.tbuffered = performance.now();
this.fragLastKbps = Math.round(8 * stats.length / (stats.tbuffered - stats.tfirst));
this.hls.trigger(Event.FRAG_BUFFERED, {stats: stats, frag: frag});
logger.log(`media buffered : ${this.timeRangesToString(this.media.buffered)}`);
2016-02-04 13:19:10 -05:00
this.state = State.IDLE;
}
2015-12-16 00:30:14 -05:00
this.tick();
}
}
2016-01-18 14:07:26 -05:00
onError(data) {
2015-12-16 00:30:14 -05:00
switch(data.details) {
case ErrorDetails.FRAG_LOAD_ERROR:
case ErrorDetails.FRAG_LOAD_TIMEOUT:
2016-01-13 15:58:12 -05:00
if(!data.fatal) {
var loadError = this.fragLoadError;
if(loadError) {
loadError++;
} else {
loadError=1;
}
if (loadError <= this.config.fragLoadingMaxRetry) {
this.fragLoadError = loadError;
// reset load counter to avoid frag loop loading error
data.frag.loadCounter = 0;
// exponential backoff capped to 64s
var delay = Math.min(Math.pow(2,loadError-1)*this.config.fragLoadingRetryDelay,64000);
logger.warn(`mediaController: frag loading failed, retry in ${delay} ms`);
this.retryDate = performance.now() + delay;
// retry loading state
this.state = State.FRAG_LOADING_WAITING_RETRY;
} else {
logger.error(`mediaController: ${data.details} reaches max retry, redispatch as fatal ...`);
// redispatch same error but with fatal set to true
data.fatal = true;
2016-01-18 14:07:26 -05:00
this.hls.trigger(Event.ERROR, data);
2016-01-13 15:58:12 -05:00
this.state = State.ERROR;
}
2015-12-23 12:46:01 -05:00
}
break;
2015-12-16 00:30:14 -05:00
case ErrorDetails.FRAG_LOOP_LOADING_ERROR:
case ErrorDetails.LEVEL_LOAD_ERROR:
case ErrorDetails.LEVEL_LOAD_TIMEOUT:
case ErrorDetails.KEY_LOAD_ERROR:
case ErrorDetails.KEY_LOAD_TIMEOUT:
2016-04-08 23:08:50 -04:00
// when in ERROR state, don't switch back to IDLE state in case a non-fatal error is received
if(this.state !== State.ERROR) {
// if fatal error, stop processing, otherwise move to IDLE to retry loading
this.state = data.fatal ? State.ERROR : State.IDLE;
logger.warn(`mediaController: ${data.details} while loading frag,switch to ${this.state} state ...`);
}
2015-12-16 00:30:14 -05:00
break;
2016-03-09 12:40:22 -05:00
case ErrorDetails.BUFFER_FULL_ERROR:
2016-02-24 22:15:07 -05:00
// trigger a smooth level switch to empty buffers
// also reduce max buffer length as it might be too high. we do this to avoid loop flushing ...
this.config.maxMaxBufferLength/=2;
logger.warn(`reduce max buffer length to ${this.config.maxMaxBufferLength}s and trigger a nextLevelSwitch to flush old buffer and fix QuotaExceededError`);
this.nextLevelSwitch();
break;
2015-12-16 00:30:14 -05:00
default:
break;
}
}
_checkBuffer() {
var media = this.media;
if(media) {
// compare readyState
var readyState = media.readyState;
// if ready state different from HAVE_NOTHING (numeric value 0), we are allowed to seek
if(readyState) {
2016-02-24 22:15:07 -05:00
var targetSeekPosition, currentTime;
2015-12-16 00:30:14 -05:00
// if seek after buffered defined, let's seek if within acceptable range
var seekAfterBuffered = this.seekAfterBuffered;
if(seekAfterBuffered) {
if(media.duration >= seekAfterBuffered) {
2016-02-24 22:15:07 -05:00
targetSeekPosition = seekAfterBuffered;
2015-12-16 00:30:14 -05:00
this.seekAfterBuffered = undefined;
}
2016-01-13 15:58:12 -05:00
} else {
2016-02-24 22:15:07 -05:00
currentTime = media.currentTime;
var loadedmetadata = this.loadedmetadata;
2016-01-25 15:28:29 -05:00
2016-02-24 22:15:07 -05:00
// adjust currentTime to start position on loaded metadata
if(!loadedmetadata && media.buffered.length) {
this.loadedmetadata = true;
// only adjust currentTime if not equal to 0
if (!currentTime && currentTime !== this.startPosition) {
targetSeekPosition = this.startPosition;
}
2016-01-25 15:28:29 -05:00
}
2016-02-24 22:15:07 -05:00
}
if (targetSeekPosition) {
currentTime = targetSeekPosition;
logger.log(`target seek position:${targetSeekPosition}`);
}
2016-03-25 12:35:19 -04:00
var bufferInfo = BufferHelper.bufferInfo(media,currentTime,0),
2016-02-24 22:15:07 -05:00
expectedPlaying = !(media.paused || media.ended || media.seeking || readyState < 2),
2016-03-09 12:40:22 -05:00
jumpThreshold = 0.4, // tolerance needed as some browsers stalls playback before reaching buffered range end
2016-02-24 22:15:07 -05:00
playheadMoving = currentTime > media.playbackRate*this.lastCurrentTime;
2016-01-13 15:58:12 -05:00
2016-02-24 22:15:07 -05:00
if (this.stalled && playheadMoving) {
this.stalled = false;
2016-03-17 01:06:37 -04:00
logger.log(`playback not stuck anymore @${currentTime}`);
2016-02-24 22:15:07 -05:00
}
// check buffer upfront
2016-04-15 15:20:04 -04:00
// if less than jumpThreshold second is buffered, and media is expected to play but playhead is not moving,
2016-02-24 22:15:07 -05:00
// and we have a new buffer range available upfront, let's seek to that one
2016-04-15 15:20:04 -04:00
if(expectedPlaying && bufferInfo.len <= jumpThreshold) {
if(playheadMoving) {
// playhead moving
2016-02-24 22:15:07 -05:00
jumpThreshold = 0;
2016-03-27 19:22:53 -04:00
this.seekHoleNudgeDuration = 0;
2016-02-24 22:15:07 -05:00
} else {
// playhead not moving AND media expected to play
if(!this.stalled) {
2016-03-27 19:22:53 -04:00
this.seekHoleNudgeDuration = 0;
2016-03-17 01:06:37 -04:00
logger.log(`playback seems stuck @${currentTime}`);
2016-02-24 22:15:07 -05:00
this.hls.trigger(Event.ERROR, {type: ErrorTypes.MEDIA_ERROR, details: ErrorDetails.BUFFER_STALLED_ERROR, fatal: false});
this.stalled = true;
2016-03-27 19:22:53 -04:00
} else {
this.seekHoleNudgeDuration += this.config.seekHoleNudgeDuration;
2016-01-13 15:58:12 -05:00
}
2016-02-24 22:15:07 -05:00
}
// if we are below threshold, try to jump if next buffer range is close
if(bufferInfo.len <= jumpThreshold) {
2016-03-09 12:40:22 -05:00
// no buffer available @ currentTime, check if next buffer is close (within a config.maxSeekHole second range)
2016-02-24 22:15:07 -05:00
var nextBufferStart = bufferInfo.nextStart, delta = nextBufferStart-currentTime;
if(nextBufferStart &&
(delta < this.config.maxSeekHole) &&
2016-03-09 12:40:22 -05:00
(delta > 0) &&
2016-02-24 22:15:07 -05:00
!media.seeking) {
// next buffer is close ! adjust currentTime to nextBufferStart
// this will ensure effective video decoding
2016-03-27 19:22:53 -04:00
logger.log(`adjust currentTime from ${media.currentTime} to next buffered @ ${nextBufferStart} + nudge ${this.seekHoleNudgeDuration}`);
media.currentTime = nextBufferStart + this.seekHoleNudgeDuration;
2016-03-09 12:40:22 -05:00
this.hls.trigger(Event.ERROR, {type: ErrorTypes.MEDIA_ERROR, details: ErrorDetails.BUFFER_SEEK_OVER_HOLE, fatal: false});
2015-12-16 00:30:14 -05:00
}
}
2016-02-24 22:15:07 -05:00
} else {
if (targetSeekPosition && media.currentTime !== targetSeekPosition) {
logger.log(`adjust currentTime from ${media.currentTime} to ${targetSeekPosition}`);
media.currentTime = targetSeekPosition;
}
2015-12-16 00:30:14 -05:00
}
}
}
}
2016-03-25 12:35:19 -04:00
onFragLoadEmergencyAborted() {
this.state = State.IDLE;
this.tick();
}
2016-02-24 22:15:07 -05:00
onBufferFlushed() {
/* after successful buffer flushing, rebuild buffer Range array
loop through existing buffer range and check if
corresponding range is still buffered. only push to new array already buffered range
*/
var newRange = [],range,i;
for (i = 0; i < this.bufferRange.length; i++) {
range = this.bufferRange[i];
if (this.isBuffered((range.start + range.end) / 2)) {
newRange.push(range);
}
}
this.bufferRange = newRange;
// handle end of immediate switching if needed
if (this.immediateSwitch) {
this.immediateLevelSwitchEnd();
}
// move to IDLE once flush complete. this should trigger new fragment loading
this.state = State.IDLE;
// reset reference to frag
this.fragPrevious = null;
2015-12-16 00:30:14 -05:00
}
2016-02-24 22:15:07 -05:00
swapAudioCodec() {
this.audioCodecSwap = !this.audioCodecSwap;
2015-12-16 00:30:14 -05:00
}
timeRangesToString(r) {
var log = '', len = r.length;
for (var i=0; i<len; i++) {
log += '[' + r.start(i) + ',' + r.end(i) + ']';
}
return log;
}
}
2016-02-24 22:15:07 -05:00
export default StreamController;
2015-12-16 00:30:14 -05:00