Source: dash/container_segment_index_source.js

/**
 * @license
 * Copyright 2015 Google Inc.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

goog.provide('shaka.dash.ContainerSegmentIndexSource');

goog.require('shaka.asserts');
goog.require('shaka.dash.LiveSegmentIndex');
goog.require('shaka.features');
goog.require('shaka.log');
goog.require('shaka.media.ISegmentIndexSource');
goog.require('shaka.media.Mp4SegmentIndexParser');
goog.require('shaka.media.SegmentIndex');
goog.require('shaka.media.SegmentReference');
goog.require('shaka.media.WebmSegmentIndexParser');
goog.require('shaka.util.FailoverUri');
goog.require('shaka.util.TypedBind');



/**
 * Creates an ISegmentIndexSource that constructs a SegmentIndex from an MP4 or
 * WebM container.
 *
 * @param {!shaka.dash.mpd.Mpd} mpd
 * @param {!shaka.dash.mpd.Period} period
 * @param {string} containerType The container type, which must be either
 *     'mp4' or 'webm'.
 * @param {!shaka.util.FailoverUri} indexMetadata The location of the
 *     container's segment index.
 * @param {shaka.util.FailoverUri} initMetadata The location of the container's
 *     headers, which is required for WebM containers and ignored
 *     for MP4 containers.
 * @param {number} manifestCreationTime The time, in seconds, when the manifest
 *     was created.
 * @param {shaka.util.FailoverUri.NetworkCallback} networkCallback
 * @constructor
 * @struct
 * @implements {shaka.media.ISegmentIndexSource}
 */
shaka.dash.ContainerSegmentIndexSource = function(
    mpd, period, containerType, indexMetadata, initMetadata,
    manifestCreationTime, networkCallback) {
  shaka.asserts.assert(containerType != 'webm' || initMetadata);

  /** @private {!shaka.dash.mpd.Mpd} */
  this.mpd_ = mpd;

  /** @private {!shaka.dash.mpd.Period} */
  this.period_ = period;

  /** @private {string} */
  this.containerType_ = containerType;

  /** @private {!shaka.util.FailoverUri} */
  this.indexMetadata_ = indexMetadata;

  /** @private {shaka.util.FailoverUri} */
  this.initMetadata_ = initMetadata;

  /** @private {number} */
  this.manifestCreationTime_ = manifestCreationTime;

  /** @private {Promise.<!shaka.media.SegmentIndex>} */
  this.promise_ = null;

  /** @private {shaka.util.FailoverUri.NetworkCallback} */
  this.networkCallback_ = networkCallback;
};


/**
 * @override
 * @suppress {checkTypes} to set otherwise non-nullable types to null.
 */
shaka.dash.ContainerSegmentIndexSource.prototype.destroy = function() {
  this.mpd_ = null;
  this.period_ = null;
  this.networkCallback_ = null;

  this.indexMetadata_.abortFetch();
  this.indexMetadata_ = null;

  if (this.initMetadata_) {
    this.initMetadata_.abortFetch();
    this.initMetadata_ = null;
  }

  this.promise_ = null;
};


/** @override */
shaka.dash.ContainerSegmentIndexSource.prototype.create = function() {
  if (this.containerType_ == 'webm' && !shaka.features.Webm) {
    var error = new Error('WebM is not supported in this build configuration.');
    error.type = 'app';
    return Promise.reject(error);
  }

  if (this.promise_) {
    return this.promise_;
  }

  var async = [this.indexMetadata_.fetch()];
  if (this.containerType_ == 'webm') {
    async.push(this.initMetadata_.fetch());
  }

  this.promise_ = Promise.all(async).then(shaka.util.TypedBind(this,
      /** @param {!Array} results */
      function(results) {
        var indexData = results[0];
        var initData = results[1] || null;

        var references = null;
        if (this.containerType_ == 'mp4') {
          var parser = new shaka.media.Mp4SegmentIndexParser();
          references = parser.parse(new DataView(indexData),
                                    this.indexMetadata_.startByte,
                                    this.indexMetadata_.urls,
                                    this.networkCallback_);
        } else if (shaka.features.Webm && this.containerType_ == 'webm') {
          shaka.asserts.assert(initData);
          var parser = new shaka.media.WebmSegmentIndexParser();
          references = parser.parse(new DataView(indexData),
                                    new DataView(initData),
                                    this.indexMetadata_.urls,
                                    this.networkCallback_);
        } else {
          shaka.asserts.unreachable();
        }

        if (!references) {
          var error = new Error(
              'Failed to parse SegmentReferences from ' +
              this.indexMetadata_.toString() + ' ' +
              '(or one of its fallbacks).');
          error.type = 'stream';
          return Promise.reject(error);
        }

        var segmentIndex;
        if (shaka.features.Live && this.mpd_.type == 'dynamic') {
          segmentIndex = new shaka.dash.LiveSegmentIndex(
              references,
              this.mpd_,
              this.period_,
              this.manifestCreationTime_);
        } else {
          shaka.asserts.assert(this.mpd_.type == 'static');
          segmentIndex = new shaka.media.SegmentIndex(references);
        }
        return Promise.resolve(segmentIndex);
      }));

  return this.promise_;
};