99580d5c by brandonocasey

browserify-p5: videojs-contrib-hls and bin-utils conversion

removed stub test and stub src files
updated build scripts to continue working
fixed several linting issues that were previously unnoticed
turn the linter back on
remove init function from stream
added ignore for playlist-loader as it will be finished later
1 parent c694b4b7
...@@ -46,10 +46,7 @@ ...@@ -46,10 +46,7 @@
46 </ul> 46 </ul>
47 47
48 <script src="/node_modules/video.js/dist/video.js"></script> 48 <script src="/node_modules/video.js/dist/video.js"></script>
49 <script src="/node_modules/videojs-contrib-media-sources/dist/videojs-media-sources.js"></script>
50 <script src="/src/videojs-contrib-hls.js"></script>
51 <script src="/dist/videojs-contrib-hls.js"></script> 49 <script src="/dist/videojs-contrib-hls.js"></script>
52 <script src="/src/bin-utils.js"></script>
53 <script> 50 <script>
54 (function(window, videojs) { 51 (function(window, videojs) {
55 var player = window.player = videojs('videojs-contrib-hls-player'); 52 var player = window.player = videojs('videojs-contrib-hls-player');
......
...@@ -2,7 +2,7 @@ ...@@ -2,7 +2,7 @@
2 "name": "videojs-contrib-hls", 2 "name": "videojs-contrib-hls",
3 "version": "1.3.5", 3 "version": "1.3.5",
4 "description": "Play back HLS with video.js, even where it's not natively supported", 4 "description": "Play back HLS with video.js, even where it's not natively supported",
5 "main": "es5/stub.js", 5 "main": "es5/videojs-contrib-hls.js",
6 "engines": { 6 "engines": {
7 "node": ">= 0.10.12" 7 "node": ">= 0.10.12"
8 }, 8 },
...@@ -27,7 +27,7 @@ ...@@ -27,7 +27,7 @@
27 "docs": "npm-run-all docs:*", 27 "docs": "npm-run-all docs:*",
28 "docs:api": "jsdoc src -r -d docs/api", 28 "docs:api": "jsdoc src -r -d docs/api",
29 "docs:toc": "doctoc README.md", 29 "docs:toc": "doctoc README.md",
30 "lint": "vjsstandard :", 30 "lint": "vjsstandard",
31 "prestart": "npm-run-all docs build", 31 "prestart": "npm-run-all docs build",
32 "start": "npm-run-all -p start:* watch:*", 32 "start": "npm-run-all -p start:* watch:*",
33 "start:serve": "babel-node scripts/server.js", 33 "start:serve": "babel-node scripts/server.js",
...@@ -40,7 +40,7 @@ ...@@ -40,7 +40,7 @@
40 "preversion": "npm test", 40 "preversion": "npm test",
41 "version": "npm run build", 41 "version": "npm run build",
42 "watch": "npm-run-all -p watch:*", 42 "watch": "npm-run-all -p watch:*",
43 "watch:js": "watchify src/stub.js -t babelify -v -o dist/videojs-contrib-hls.js", 43 "watch:js": "watchify src/videojs-contrib-hls.js -t babelify -v -o dist/videojs-contrib-hls.js",
44 "watch:test": "npm-run-all -p watch:test:*", 44 "watch:test": "npm-run-all -p watch:test:*",
45 "watch:test:js": "node scripts/watch-test.js", 45 "watch:test:js": "node scripts/watch-test.js",
46 "watch:test:manifest": "node -e \"var b=require('./scripts/manifest-data.js'); b.watch();\"", 46 "watch:test:manifest": "node -e \"var b=require('./scripts/manifest-data.js'); b.watch();\"",
...@@ -72,7 +72,8 @@ ...@@ -72,7 +72,8 @@
72 "scripts", 72 "scripts",
73 "utils", 73 "utils",
74 "test/test-manifests.js", 74 "test/test-manifests.js",
75 "test/test-expected.js" 75 "test/test-expected.js",
76 "src/playlist-loader.js"
76 ] 77 ]
77 }, 78 },
78 "files": [ 79 "files": [
......
...@@ -2,7 +2,7 @@ var browserify = require('browserify'); ...@@ -2,7 +2,7 @@ var browserify = require('browserify');
2 var fs = require('fs'); 2 var fs = require('fs');
3 var glob = require('glob'); 3 var glob = require('glob');
4 4
5 glob('test/{playlist*,decryper,m3u8,stub}.test.js', function(err, files) { 5 glob('test/**/*.test.js', function(err, files) {
6 browserify(files) 6 browserify(files)
7 .transform('babelify') 7 .transform('babelify')
8 .bundle() 8 .bundle()
......
...@@ -3,7 +3,7 @@ var fs = require('fs'); ...@@ -3,7 +3,7 @@ var fs = require('fs');
3 var glob = require('glob'); 3 var glob = require('glob');
4 var watchify = require('watchify'); 4 var watchify = require('watchify');
5 5
6 glob('test/{playlist*,decrypter,m3u8,stub}.test.js', function(err, files) { 6 glob('test/**/*.test.js', function(err, files) {
7 var b = browserify(files, { 7 var b = browserify(files, {
8 cache: {}, 8 cache: {},
9 packageCache: {}, 9 packageCache: {},
......
1 (function(window) { 1 const textRange = function(range, i) {
2 var textRange = function(range, i) {
3 return range.start(i) + '-' + range.end(i); 2 return range.start(i) + '-' + range.end(i);
4 }; 3 };
5 var module = { 4
6 hexDump: function(data) { 5 const formatHexString = function(e, i) {
7 var 6 let value = e.toString(16);
8 bytes = Array.prototype.slice.call(data), 7
9 step = 16, 8 return '00'.substring(0, 2 - value.length) + value + (i % 2 ? ' ' : '');
10 formatHexString = function(e, i) { 9 };
11 var value = e.toString(16); 10 const formatAsciiString = function(e) {
12 return "00".substring(0, 2 - value.length) + value + (i % 2 ? ' ' : '');
13 },
14 formatAsciiString = function(e) {
15 if (e >= 0x20 && e < 0x7e) { 11 if (e >= 0x20 && e < 0x7e) {
16 return String.fromCharCode(e); 12 return String.fromCharCode(e);
17 } 13 }
18 return '.'; 14 return '.';
19 }, 15 };
20 result = '', 16
21 hex, 17 const utils = {
22 ascii; 18 hexDump(data) {
23 for (var j = 0; j < bytes.length / step; j++) { 19 let bytes = Array.prototype.slice.call(data);
20 let step = 16;
21 let result = '';
22 let hex;
23 let ascii;
24
25 for (let j = 0; j < bytes.length / step; j++) {
24 hex = bytes.slice(j * step, j * step + step).map(formatHexString).join(''); 26 hex = bytes.slice(j * step, j * step + step).map(formatHexString).join('');
25 ascii = bytes.slice(j * step, j * step + step).map(formatAsciiString).join(''); 27 ascii = bytes.slice(j * step, j * step + step).map(formatAsciiString).join('');
26 result += hex + ' ' + ascii + '\n'; 28 result += hex + ' ' + ascii + '\n';
27 } 29 }
28 return result; 30 return result;
29 }, 31 },
30 tagDump: function(tag) { 32 tagDump(tag) {
31 return module.hexDump(tag.bytes); 33 return utils.hexDump(tag.bytes);
32 }, 34 },
33 textRanges: function(ranges) { 35 textRanges(ranges) {
34 var result = '', i; 36 let result = '';
37 let i;
38
35 for (i = 0; i < ranges.length; i++) { 39 for (i = 0; i < ranges.length; i++) {
36 result += textRange(ranges, i) + ' '; 40 result += textRange(ranges, i) + ' ';
37 } 41 }
38 return result; 42 return result;
39 } 43 }
40 }; 44 };
41 45
42 window.videojs.Hls.utils = module; 46 export default utils;
43 })(this);
......
...@@ -89,10 +89,9 @@ const precompute = function() { ...@@ -89,10 +89,9 @@ const precompute = function() {
89 decTable[i] = decTable[i].slice(0); 89 decTable[i] = decTable[i].slice(0);
90 } 90 }
91 return tables; 91 return tables;
92 } 92 };
93
94
95 let aesTables = null; 93 let aesTables = null;
94
96 /** 95 /**
97 * Schedule out an AES key for both encryption and decryption. This 96 * Schedule out an AES key for both encryption and decryption. This
98 * is a low-level class. Use a cipher mode to do bulk encryption. 97 * is a low-level class. Use a cipher mode to do bulk encryption.
...@@ -116,7 +115,7 @@ export default class AES { ...@@ -116,7 +115,7 @@ export default class AES {
116 */ 115 */
117 // if we have yet to precompute the S-box tables 116 // if we have yet to precompute the S-box tables
118 // do so now 117 // do so now
119 if(!aesTables) { 118 if (!aesTables) {
120 aesTables = precompute(); 119 aesTables = precompute();
121 } 120 }
122 // then make a copy of that object for use 121 // then make a copy of that object for use
......
1 /** 1 /**
2 * A lightweight readable stream implemention that handles event dispatching. 2 * A lightweight readable stream implemention that handles event dispatching.
3 * Objects that inherit from streams should call init in their constructors.
4 */ 3 */
5 export default class Stream { 4 export default class Stream {
6 constructor() { 5 constructor() {
7 this.init();
8 }
9
10 init() {
11 this.listeners = {}; 6 this.listeners = {};
12 } 7 }
13 8
......
1 import m3u8 from './m3u8';
2 import Stream from './stream';
3 import videojs from 'video.js';
4 import {Decrypter, decrypt, AsyncStream} from './decrypter';
5 import Playlist from './playlist';
6 import PlaylistLoader from './playlist-loader';
7 import xhr from './xhr';
8
9
10 if(typeof window.videojs.Hls === 'undefined') {
11 videojs.Hls = {};
12 }
13 videojs.Hls.Stream = Stream;
14 videojs.m3u8 = m3u8;
15 videojs.Hls.decrypt = decrypt;
16 videojs.Hls.Decrypter = Decrypter;
17 videojs.Hls.AsyncStream = AsyncStream;
18 videojs.Hls.xhr = xhr;
19 videojs.Hls.Playlist = Playlist;
20 videojs.Hls.PlaylistLoader = PlaylistLoader;
1 /* 1 /**
2 * videojs-hls 2 * videojs-hls
3 * The main file for the HLS project. 3 * The main file for the HLS project.
4 * License: https://github.com/videojs/videojs-contrib-hls/blob/master/LICENSE 4 * License: https://github.com/videojs/videojs-contrib-hls/blob/master/LICENSE
5 */ 5 */
6 (function(window, videojs, document, undefined) { 6 import PlaylistLoader from './playlist-loader';
7 'use strict'; 7 import Playlist from './playlist';
8 import xhr from './xhr';
9 import {Decrypter, AsyncStream, decrypt} from './decrypter';
10 import utils from './bin-utils';
11 import {MediaSource, URL} from 'videojs-contrib-media-sources';
12 import m3u8 from './m3u8';
13 import videojs from 'video.js';
14 import resolveUrl from './resolve-url';
15
16 const Hls = {
17 PlaylistLoader,
18 Playlist,
19 Decrypter,
20 AsyncStream,
21 decrypt,
22 utils,
23 xhr
24 };
25
26 // the desired length of video to maintain in the buffer, in seconds
27 Hls.GOAL_BUFFER_LENGTH = 30;
28
29 // HLS is a source handler, not a tech. Make sure attempts to use it
30 // as one do not cause exceptions.
31 Hls.canPlaySource = function() {
32 return videojs.log.warn('HLS is no longer a tech. Please remove it from ' +
33 'your player\'s techOrder.');
34 };
35
36 // Search for a likely end time for the segment that was just appened
37 // based on the state of the `buffered` property before and after the
38 // append.
39 // If we found only one such uncommon end-point return it.
40 Hls.findSoleUncommonTimeRangesEnd_ = function(original, update) {
41 let i;
42 let start;
43 let end;
44 let result = [];
45 let edges = [];
46
47 // In order to qualify as a possible candidate, the end point must:
48 // 1) Not have already existed in the `original` ranges
49 // 2) Not result from the shrinking of a range that already existed
50 // in the `original` ranges
51 // 3) Not be contained inside of a range that existed in `original`
52 let overlapsCurrentEnd = function(span) {
53 return (span[0] <= end && span[1] >= end);
54 };
55
56 if (original) {
57 // Save all the edges in the `original` TimeRanges object
58 for (i = 0; i < original.length; i++) {
59 start = original.start(i);
60 end = original.end(i);
61
62 edges.push([start, end]);
63 }
64 }
65
66 if (update) {
67 // Save any end-points in `update` that are not in the `original`
68 // TimeRanges object
69 for (i = 0; i < update.length; i++) {
70 start = update.start(i);
71 end = update.end(i);
72
73 if (edges.some(overlapsCurrentEnd)) {
74 continue;
75 }
76
77 // at this point it must be a unique non-shrinking end edge
78 result.push(end);
79 }
80 }
81
82 // we err on the side of caution and return null if didn't find
83 // exactly *one* differing end edge in the search above
84 if (result.length !== 1) {
85 return null;
86 }
8 87
9 var 88 return result[0];
10 // A fudge factor to apply to advertised playlist bitrates to account for 89 };
11 // temporary flucations in client bandwidth
12 bandwidthVariance = 1.2,
13 blacklistDuration = 5 * 60 * 1000, // 5 minute blacklist
14 TIME_FUDGE_FACTOR = 1 / 30, // Fudge factor to account for TimeRanges rounding
15 Component = videojs.getComponent('Component'),
16 90
17 // The amount of time to wait between checking the state of the buffer 91 /**
18 bufferCheckInterval = 500, 92 * Whether the browser has built-in HLS support.
93 */
94 Hls.supportsNativeHls = function() {
95 let video = document.createElement('video');
96 let xMpegUrl;
97 let vndMpeg;
98
99 // native HLS is definitely not supported if HTML5 video isn't
100 if (!videojs.getComponent('Html5').isSupported()) {
101 return false;
102 }
19 103
20 keyFailed, 104 xMpegUrl = video.canPlayType('application/x-mpegURL');
21 resolveUrl; 105 vndMpeg = video.canPlayType('application/vnd.apple.mpegURL');
106 return (/probably|maybe/).test(xMpegUrl) ||
107 (/probably|maybe/).test(vndMpeg);
108 };
109
110 // HLS is a source handler, not a tech. Make sure attempts to use it
111 // as one do not cause exceptions.
112 Hls.isSupported = function() {
113 return videojs.log.warn('HLS is no longer a tech. Please remove it from ' +
114 'your player\'s techOrder.');
115 };
116
117 /**
118 * A comparator function to sort two playlist object by bandwidth.
119 * @param left {object} a media playlist object
120 * @param right {object} a media playlist object
121 * @return {number} Greater than zero if the bandwidth attribute of
122 * left is greater than the corresponding attribute of right. Less
123 * than zero if the bandwidth of right is greater than left and
124 * exactly zero if the two are equal.
125 */
126 Hls.comparePlaylistBandwidth = function(left, right) {
127 let leftBandwidth;
128 let rightBandwidth;
129
130 if (left.attributes && left.attributes.BANDWIDTH) {
131 leftBandwidth = left.attributes.BANDWIDTH;
132 }
133 leftBandwidth = leftBandwidth || window.Number.MAX_VALUE;
134 if (right.attributes && right.attributes.BANDWIDTH) {
135 rightBandwidth = right.attributes.BANDWIDTH;
136 }
137 rightBandwidth = rightBandwidth || window.Number.MAX_VALUE;
138
139 return leftBandwidth - rightBandwidth;
140 };
141
142 /**
143 * A comparator function to sort two playlist object by resolution (width).
144 * @param left {object} a media playlist object
145 * @param right {object} a media playlist object
146 * @return {number} Greater than zero if the resolution.width attribute of
147 * left is greater than the corresponding attribute of right. Less
148 * than zero if the resolution.width of right is greater than left and
149 * exactly zero if the two are equal.
150 */
151 Hls.comparePlaylistResolution = function(left, right) {
152 let leftWidth;
153 let rightWidth;
154
155 if (left.attributes &&
156 left.attributes.RESOLUTION &&
157 left.attributes.RESOLUTION.width) {
158 leftWidth = left.attributes.RESOLUTION.width;
159 }
160
161 leftWidth = leftWidth || window.Number.MAX_VALUE;
162
163 if (right.attributes &&
164 right.attributes.RESOLUTION &&
165 right.attributes.RESOLUTION.width) {
166 rightWidth = right.attributes.RESOLUTION.width;
167 }
168
169 rightWidth = rightWidth || window.Number.MAX_VALUE;
170
171 // NOTE - Fallback to bandwidth sort as appropriate in cases where multiple renditions
172 // have the same media dimensions/ resolution
173 if (leftWidth === rightWidth &&
174 left.attributes.BANDWIDTH &&
175 right.attributes.BANDWIDTH) {
176 return left.attributes.BANDWIDTH - right.attributes.BANDWIDTH;
177 }
178 return leftWidth - rightWidth;
179 };
180
181 // A fudge factor to apply to advertised playlist bitrates to account for
182 // temporary flucations in client bandwidth
183 const bandwidthVariance = 1.2;
184
185 // 5 minute blacklist
186 const blacklistDuration = 5 * 60 * 1000;
187
188 // Fudge factor to account for TimeRanges rounding
189 const TIME_FUDGE_FACTOR = 1 / 30;
190 const Component = videojs.getComponent('Component');
191
192 // The amount of time to wait between checking the state of the buffer
193 const bufferCheckInterval = 500;
22 194
23 // returns true if a key has failed to download within a certain amount of retries 195 // returns true if a key has failed to download within a certain amount of retries
24 keyFailed = function(key) { 196 const keyFailed = function(key) {
25 return key.retries && key.retries >= 2; 197 return key.retries && key.retries >= 2;
26 }; 198 };
27 199
28 videojs.Hls = {}; 200 const parseCodecs = function(codecs) {
29 videojs.HlsHandler = videojs.extend(Component, { 201 let result = {
30 constructor: function(tech, options) { 202 codecCount: 0,
31 var self = this, _player; 203 videoCodec: null,
204 audioProfile: null
205 };
206
207 result.codecCount = codecs.split(',').length;
208 result.codecCount = result.codecCount || 2;
209
210 // parse the video codec but ignore the version
211 result.videoCodec = (/(^|\s|,)+(avc1)[^ ,]*/i).exec(codecs);
212 result.videoCodec = result.videoCodec && result.videoCodec[2];
213
214 // parse the last field of the audio codec
215 result.audioProfile = (/(^|\s|,)+mp4a.\d+\.(\d+)/i).exec(codecs);
216 result.audioProfile = result.audioProfile && result.audioProfile[2];
217
218 return result;
219 };
220
221 const filterBufferedRanges = function(predicate) {
222 return function(time) {
223 let i;
224 let ranges = [];
225 let tech = this.tech_;
226 // !!The order of the next two assignments is important!!
227 // `currentTime` must be equal-to or greater-than the start of the
228 // buffered range. Flash executes out-of-process so, every value can
229 // change behind the scenes from line-to-line. By reading `currentTime`
230 // after `buffered`, we ensure that it is always a current or later
231 // value during playback.
232 let buffered = tech.buffered();
233
234 if (typeof time === 'undefined') {
235 time = tech.currentTime();
236 }
237
238 if (buffered && buffered.length) {
239 // Search for a range containing the play-head
240 for (i = 0; i < buffered.length; i++) {
241 if (predicate(buffered.start(i), buffered.end(i), time)) {
242 ranges.push([buffered.start(i), buffered.end(i)]);
243 }
244 }
245 }
246
247 return videojs.createTimeRanges(ranges);
248 };
249 };
32 250
33 Component.call(this, tech); 251 export default class HlsHandler extends Component {
252 constructor(tech, options) {
253 super(tech);
254 let _player;
34 255
35 // tech.player() is deprecated but setup a reference to HLS for 256 // tech.player() is deprecated but setup a reference to HLS for
36 // backwards-compatibility 257 // backwards-compatibility
...@@ -38,9 +259,9 @@ videojs.HlsHandler = videojs.extend(Component, { ...@@ -38,9 +259,9 @@ videojs.HlsHandler = videojs.extend(Component, {
38 _player = videojs(tech.options_.playerId); 259 _player = videojs(tech.options_.playerId);
39 if (!_player.hls) { 260 if (!_player.hls) {
40 Object.defineProperty(_player, 'hls', { 261 Object.defineProperty(_player, 'hls', {
41 get: function() { 262 get: () => {
42 videojs.log.warn('player.hls is deprecated. Use player.tech.hls instead.'); 263 videojs.log.warn('player.hls is deprecated. Use player.tech.hls instead.');
43 return self; 264 return this;
44 } 265 }
45 }); 266 });
46 } 267 }
...@@ -54,7 +275,8 @@ videojs.HlsHandler = videojs.extend(Component, { ...@@ -54,7 +275,8 @@ videojs.HlsHandler = videojs.extend(Component, {
54 275
55 // start playlist selection at a reasonable bandwidth for 276 // start playlist selection at a reasonable bandwidth for
56 // broadband internet 277 // broadband internet
57 this.bandwidth = options.bandwidth || 4194304; // 0.5 Mbps 278 // 0.5 Mbps
279 this.bandwidth = options.bandwidth || 4194304;
58 this.bytesReceived = 0; 280 this.bytesReceived = 0;
59 281
60 // loadingState_ tracks how far along the buffering process we 282 // loadingState_ tracks how far along the buffering process we
...@@ -81,71 +303,8 @@ videojs.HlsHandler = videojs.extend(Component, { ...@@ -81,71 +303,8 @@ videojs.HlsHandler = videojs.extend(Component, {
81 303
82 this.on(this.tech_, 'play', this.play); 304 this.on(this.tech_, 'play', this.play);
83 } 305 }
84 }); 306 src(src) {
85 307 let oldMediaPlaylist;
86 // HLS is a source handler, not a tech. Make sure attempts to use it
87 // as one do not cause exceptions.
88 videojs.Hls.canPlaySource = function() {
89 return videojs.log.warn('HLS is no longer a tech. Please remove it from ' +
90 'your player\'s techOrder.');
91 };
92
93 /**
94 * The Source Handler object, which informs video.js what additional
95 * MIME types are supported and sets up playback. It is registered
96 * automatically to the appropriate tech based on the capabilities of
97 * the browser it is running in. It is not necessary to use or modify
98 * this object in normal usage.
99 */
100 videojs.HlsSourceHandler = function(mode) {
101 return {
102 canHandleSource: function(srcObj) {
103 return videojs.HlsSourceHandler.canPlayType(srcObj.type);
104 },
105 handleSource: function(source, tech) {
106 if (mode === 'flash') {
107 // We need to trigger this asynchronously to give others the chance
108 // to bind to the event when a source is set at player creation
109 tech.setTimeout(function() {
110 tech.trigger('loadstart');
111 }, 1);
112 }
113 tech.hls = new videojs.HlsHandler(tech, {
114 source: source,
115 mode: mode
116 });
117 tech.hls.src(source.src);
118 return tech.hls;
119 },
120 canPlayType: function(type) {
121 return videojs.HlsSourceHandler.canPlayType(type);
122 }
123 };
124 };
125
126 videojs.HlsSourceHandler.canPlayType = function(type) {
127 var mpegurlRE = /^application\/(?:x-|vnd\.apple\.)mpegurl/i;
128
129 // favor native HLS support if it's available
130 if (videojs.Hls.supportsNativeHls) {
131 return false;
132 }
133 return mpegurlRE.test(type);
134 };
135
136 // register source handlers with the appropriate techs
137 if (videojs.MediaSource.supportsNativeMediaSources()) {
138 videojs.getComponent('Html5').registerSourceHandler(videojs.HlsSourceHandler('html5'));
139 }
140 if (window.Uint8Array) {
141 videojs.getComponent('Flash').registerSourceHandler(videojs.HlsSourceHandler('flash'));
142 }
143
144 // the desired length of video to maintain in the buffer, in seconds
145 videojs.Hls.GOAL_BUFFER_LENGTH = 30;
146
147 videojs.HlsHandler.prototype.src = function(src) {
148 var oldMediaPlaylist;
149 308
150 // do nothing if the src is falsey 309 // do nothing if the src is falsey
151 if (!src) { 310 if (!src) {
...@@ -158,16 +317,17 @@ videojs.HlsHandler.prototype.src = function(src) { ...@@ -158,16 +317,17 @@ videojs.HlsHandler.prototype.src = function(src) {
158 this.mediaSource.addEventListener('sourceopen', this.handleSourceOpen.bind(this)); 317 this.mediaSource.addEventListener('sourceopen', this.handleSourceOpen.bind(this));
159 318
160 this.options_ = {}; 319 this.options_ = {};
161 if (this.source_.withCredentials !== undefined) { 320 if (typeof this.source_.withCredentials !== 'undefined') {
162 this.options_.withCredentials = this.source_.withCredentials; 321 this.options_.withCredentials = this.source_.withCredentials;
163 } else if (videojs.options.hls) { 322 } else if (videojs.options.hls) {
164 this.options_.withCredentials = videojs.options.hls.withCredentials; 323 this.options_.withCredentials = videojs.options.hls.withCredentials;
165 } 324 }
166 this.playlists = new videojs.Hls.PlaylistLoader(this.source_.src, this.options_.withCredentials); 325 this.playlists = new Hls.PlaylistLoader(this.source_.src,
326 this.options_.withCredentials);
167 327
168 this.tech_.one('canplay', this.setupFirstPlay.bind(this)); 328 this.tech_.one('canplay', this.setupFirstPlay.bind(this));
169 329
170 this.playlists.on('loadedmetadata', function() { 330 this.playlists.on('loadedmetadata', () => {
171 oldMediaPlaylist = this.playlists.media(); 331 oldMediaPlaylist = this.playlists.media();
172 332
173 // if this isn't a live video and preload permits, start 333 // if this isn't a live video and preload permits, start
...@@ -182,14 +342,15 @@ videojs.HlsHandler.prototype.src = function(src) { ...@@ -182,14 +342,15 @@ videojs.HlsHandler.prototype.src = function(src) {
182 this.setupFirstPlay(); 342 this.setupFirstPlay();
183 this.fillBuffer(); 343 this.fillBuffer();
184 this.tech_.trigger('loadedmetadata'); 344 this.tech_.trigger('loadedmetadata');
185 }.bind(this)); 345 });
186 346
187 this.playlists.on('error', function() { 347 this.playlists.on('error', () => {
188 this.blacklistCurrentPlaylist_(this.playlists.error); 348 this.blacklistCurrentPlaylist_(this.playlists.error);
189 }.bind(this)); 349 });
190 350
191 this.playlists.on('loadedplaylist', function() { 351 this.playlists.on('loadedplaylist', () => {
192 var updatedPlaylist = this.playlists.media(), seekable; 352 let updatedPlaylist = this.playlists.media();
353 let seekable;
193 354
194 if (!updatedPlaylist) { 355 if (!updatedPlaylist) {
195 // select the initial variant 356 // select the initial variant
...@@ -207,14 +368,14 @@ videojs.HlsHandler.prototype.src = function(src) { ...@@ -207,14 +368,14 @@ videojs.HlsHandler.prototype.src = function(src) {
207 } 368 }
208 369
209 oldMediaPlaylist = updatedPlaylist; 370 oldMediaPlaylist = updatedPlaylist;
210 }.bind(this)); 371 });
211 372
212 this.playlists.on('mediachange', function() { 373 this.playlists.on('mediachange', () => {
213 this.tech_.trigger({ 374 this.tech_.trigger({
214 type: 'mediachange', 375 type: 'mediachange',
215 bubbles: true 376 bubbles: true
216 }); 377 });
217 }.bind(this)); 378 });
218 379
219 // do nothing if the tech has been disposed already 380 // do nothing if the tech has been disposed already
220 // this can occur if someone sets the src in player.ready(), for instance 381 // this can occur if someone sets the src in player.ready(), for instance
...@@ -223,9 +384,8 @@ videojs.HlsHandler.prototype.src = function(src) { ...@@ -223,9 +384,8 @@ videojs.HlsHandler.prototype.src = function(src) {
223 } 384 }
224 385
225 this.tech_.src(videojs.URL.createObjectURL(this.mediaSource)); 386 this.tech_.src(videojs.URL.createObjectURL(this.mediaSource));
226 }; 387 }
227 388 handleSourceOpen() {
228 videojs.HlsHandler.prototype.handleSourceOpen = function() {
229 // Only attempt to create the source buffer if none already exist. 389 // Only attempt to create the source buffer if none already exist.
230 // handleSourceOpen is also called when we are "re-opening" a source buffer 390 // handleSourceOpen is also called when we are "re-opening" a source buffer
231 // after `endOfStream` has been called (in response to a seek for instance) 391 // after `endOfStream` has been called (in response to a seek for instance)
...@@ -242,83 +402,9 @@ videojs.HlsHandler.prototype.handleSourceOpen = function() { ...@@ -242,83 +402,9 @@ videojs.HlsHandler.prototype.handleSourceOpen = function() {
242 if (this.tech_.autoplay()) { 402 if (this.tech_.autoplay()) {
243 this.play(); 403 this.play();
244 } 404 }
245 };
246
247 // Search for a likely end time for the segment that was just appened
248 // based on the state of the `buffered` property before and after the
249 // append.
250 // If we found only one such uncommon end-point return it.
251 videojs.Hls.findSoleUncommonTimeRangesEnd_ = function(original, update) {
252 var
253 i, start, end,
254 result = [],
255 edges = [],
256 // In order to qualify as a possible candidate, the end point must:
257 // 1) Not have already existed in the `original` ranges
258 // 2) Not result from the shrinking of a range that already existed
259 // in the `original` ranges
260 // 3) Not be contained inside of a range that existed in `original`
261 overlapsCurrentEnd = function(span) {
262 return (span[0] <= end && span[1] >= end);
263 };
264
265 if (original) {
266 // Save all the edges in the `original` TimeRanges object
267 for (i = 0; i < original.length; i++) {
268 start = original.start(i);
269 end = original.end(i);
270
271 edges.push([start, end]);
272 }
273 } 405 }
274 406
275 if (update) { 407 /**
276 // Save any end-points in `update` that are not in the `original`
277 // TimeRanges object
278 for (i = 0; i < update.length; i++) {
279 start = update.start(i);
280 end = update.end(i);
281
282 if (edges.some(overlapsCurrentEnd)) {
283 continue;
284 }
285
286 // at this point it must be a unique non-shrinking end edge
287 result.push(end);
288 }
289 }
290
291 // we err on the side of caution and return null if didn't find
292 // exactly *one* differing end edge in the search above
293 if (result.length !== 1) {
294 return null;
295 }
296
297 return result[0];
298 };
299
300 var parseCodecs = function(codecs) {
301 var result = {
302 codecCount: 0,
303 videoCodec: null,
304 audioProfile: null
305 };
306
307 result.codecCount = codecs.split(',').length;
308 result.codecCount = result.codecCount || 2;
309
310 // parse the video codec but ignore the version
311 result.videoCodec = /(^|\s|,)+(avc1)[^ ,]*/i.exec(codecs);
312 result.videoCodec = result.videoCodec && result.videoCodec[2];
313
314 // parse the last field of the audio codec
315 result.audioProfile = /(^|\s|,)+mp4a.\d+\.(\d+)/i.exec(codecs);
316 result.audioProfile = result.audioProfile && result.audioProfile[2];
317
318 return result;
319 };
320
321 /**
322 * Blacklist playlists that are known to be codec or 408 * Blacklist playlists that are known to be codec or
323 * stream-incompatible with the SourceBuffer configuration. For 409 * stream-incompatible with the SourceBuffer configuration. For
324 * instance, Media Source Extensions would cause the video element to 410 * instance, Media Source Extensions would cause the video element to
...@@ -331,13 +417,12 @@ var parseCodecs = function(codecs) { ...@@ -331,13 +417,12 @@ var parseCodecs = function(codecs) {
331 * will be excluded from the default playlist selection algorithm 417 * will be excluded from the default playlist selection algorithm
332 * indefinitely. 418 * indefinitely.
333 */ 419 */
334 videojs.HlsHandler.prototype.excludeIncompatibleVariants_ = function(media) { 420 excludeIncompatibleVariants_(media) {
335 var 421 let master = this.playlists.master;
336 master = this.playlists.master, 422 let codecCount = 2;
337 codecCount = 2, 423 let videoCodec = null;
338 videoCodec = null, 424 let audioProfile = null;
339 audioProfile = null, 425 let codecs;
340 codecs;
341 426
342 if (media.attributes && media.attributes.CODECS) { 427 if (media.attributes && media.attributes.CODECS) {
343 codecs = parseCodecs(media.attributes.CODECS); 428 codecs = parseCodecs(media.attributes.CODECS);
...@@ -346,7 +431,7 @@ videojs.HlsHandler.prototype.excludeIncompatibleVariants_ = function(media) { ...@@ -346,7 +431,7 @@ videojs.HlsHandler.prototype.excludeIncompatibleVariants_ = function(media) {
346 codecCount = codecs.codecCount; 431 codecCount = codecs.codecCount;
347 } 432 }
348 master.playlists.forEach(function(variant) { 433 master.playlists.forEach(function(variant) {
349 var variantCodecs = { 434 let variantCodecs = {
350 codecCount: 2, 435 codecCount: 2,
351 videoCodec: null, 436 videoCodec: null,
352 audioProfile: null 437 audioProfile: null
...@@ -374,10 +459,11 @@ videojs.HlsHandler.prototype.excludeIncompatibleVariants_ = function(media) { ...@@ -374,10 +459,11 @@ videojs.HlsHandler.prototype.excludeIncompatibleVariants_ = function(media) {
374 variant.excludeUntil = Infinity; 459 variant.excludeUntil = Infinity;
375 } 460 }
376 }); 461 });
377 }; 462 }
378 463
379 videojs.HlsHandler.prototype.setupSourceBuffer_ = function() { 464 setupSourceBuffer_() {
380 var media = this.playlists.media(), mimeType; 465 let media = this.playlists.media();
466 let mimeType;
381 467
382 // wait until a media playlist is available and the Media Source is 468 // wait until a media playlist is available and the Media Source is
383 // attached 469 // attached
...@@ -400,16 +486,15 @@ videojs.HlsHandler.prototype.setupSourceBuffer_ = function() { ...@@ -400,16 +486,15 @@ videojs.HlsHandler.prototype.setupSourceBuffer_ = function() {
400 // transition the sourcebuffer to the ended state if we've hit the end of 486 // transition the sourcebuffer to the ended state if we've hit the end of
401 // the playlist 487 // the playlist
402 this.sourceBuffer.addEventListener('updateend', this.updateEndHandler_.bind(this)); 488 this.sourceBuffer.addEventListener('updateend', this.updateEndHandler_.bind(this));
403 }; 489 }
404 490
405 /** 491 /**
406 * Seek to the latest media position if this is a live video and the 492 * Seek to the latest media position if this is a live video and the
407 * player and video are loaded and initialized. 493 * player and video are loaded and initialized.
408 */ 494 */
409 videojs.HlsHandler.prototype.setupFirstPlay = function() { 495 setupFirstPlay() {
410 var seekable, media; 496 let seekable;
411 media = this.playlists.media(); 497 let media = this.playlists.media();
412
413 498
414 // check that everything is ready to begin buffering 499 // check that everything is ready to begin buffering
415 500
...@@ -439,12 +524,12 @@ videojs.HlsHandler.prototype.setupFirstPlay = function() { ...@@ -439,12 +524,12 @@ videojs.HlsHandler.prototype.setupFirstPlay = function() {
439 this.tech_.setCurrentTime(seekable.end(0)); 524 this.tech_.setCurrentTime(seekable.end(0));
440 } 525 }
441 } 526 }
442 }; 527 }
443 528
444 /** 529 /**
445 * Begin playing the video. 530 * Begin playing the video.
446 */ 531 */
447 videojs.HlsHandler.prototype.play = function() { 532 play() {
448 this.loadingState_ = 'segments'; 533 this.loadingState_ = 'segments';
449 534
450 if (this.tech_.ended()) { 535 if (this.tech_.ended()) {
...@@ -462,11 +547,10 @@ videojs.HlsHandler.prototype.play = function() { ...@@ -462,11 +547,10 @@ videojs.HlsHandler.prototype.play = function() {
462 this.tech_.setCurrentTime(this.seekable().start(0)); 547 this.tech_.setCurrentTime(this.seekable().start(0));
463 } 548 }
464 } 549 }
465 }; 550 }
466 551
467 videojs.HlsHandler.prototype.setCurrentTime = function(currentTime) { 552 setCurrentTime(currentTime) {
468 var 553 let buffered = this.findBufferedRange_();
469 buffered = this.findBufferedRange_();
470 554
471 if (!(this.playlists && this.playlists.media())) { 555 if (!(this.playlists && this.playlists.media())) {
472 // return immediately if the metadata is not ready yet 556 // return immediately if the metadata is not ready yet
...@@ -503,18 +587,20 @@ videojs.HlsHandler.prototype.setCurrentTime = function(currentTime) { ...@@ -503,18 +587,20 @@ videojs.HlsHandler.prototype.setCurrentTime = function(currentTime) {
503 587
504 // begin filling the buffer at the new position 588 // begin filling the buffer at the new position
505 this.fillBuffer(this.playlists.getMediaIndexForTime_(currentTime)); 589 this.fillBuffer(this.playlists.getMediaIndexForTime_(currentTime));
506 }; 590 }
591
592 duration() {
593 let playlists = this.playlists;
507 594
508 videojs.HlsHandler.prototype.duration = function() {
509 var playlists = this.playlists;
510 if (playlists) { 595 if (playlists) {
511 return videojs.Hls.Playlist.duration(playlists.media()); 596 return Hls.Playlist.duration(playlists.media());
512 } 597 }
513 return 0; 598 return 0;
514 }; 599 }
515 600
516 videojs.HlsHandler.prototype.seekable = function() { 601 seekable() {
517 var media, seekable; 602 let media;
603 let seekable;
518 604
519 if (!this.playlists) { 605 if (!this.playlists) {
520 return videojs.createTimeRanges(); 606 return videojs.createTimeRanges();
...@@ -524,7 +610,7 @@ videojs.HlsHandler.prototype.seekable = function() { ...@@ -524,7 +610,7 @@ videojs.HlsHandler.prototype.seekable = function() {
524 return videojs.createTimeRanges(); 610 return videojs.createTimeRanges();
525 } 611 }
526 612
527 seekable = videojs.Hls.Playlist.seekable(media); 613 seekable = Hls.Playlist.seekable(media);
528 if (seekable.length === 0) { 614 if (seekable.length === 0) {
529 return seekable; 615 return seekable;
530 } 616 }
...@@ -534,29 +620,27 @@ videojs.HlsHandler.prototype.seekable = function() { ...@@ -534,29 +620,27 @@ videojs.HlsHandler.prototype.seekable = function() {
534 // fall back to the playlist loader's running estimate of expired 620 // fall back to the playlist loader's running estimate of expired
535 // time 621 // time
536 if (seekable.start(0) === 0) { 622 if (seekable.start(0) === 0) {
537 return videojs.createTimeRanges([[ 623 return videojs.createTimeRanges([[this.playlists.expired_,
538 this.playlists.expired_, 624 this.playlists.expired_ + seekable.end(0)]]);
539 this.playlists.expired_ + seekable.end(0)
540 ]]);
541 } 625 }
542 626
543 // seekable has been calculated based on buffering video data so it 627 // seekable has been calculated based on buffering video data so it
544 // can be returned directly 628 // can be returned directly
545 return seekable; 629 return seekable;
546 }; 630 }
547 631
548 /** 632 /**
549 * Update the player duration 633 * Update the player duration
550 */ 634 */
551 videojs.HlsHandler.prototype.updateDuration = function(playlist) { 635 updateDuration(playlist) {
552 var oldDuration = this.mediaSource.duration, 636 let oldDuration = this.mediaSource.duration;
553 newDuration = videojs.Hls.Playlist.duration(playlist), 637 let newDuration = Hls.Playlist.duration(playlist);
554 setDuration = function() { 638 let setDuration = () => {
555 this.mediaSource.duration = newDuration; 639 this.mediaSource.duration = newDuration;
556 this.tech_.trigger('durationchange'); 640 this.tech_.trigger('durationchange');
557 641
558 this.mediaSource.removeEventListener('sourceopen', setDuration); 642 this.mediaSource.removeEventListener('sourceopen', setDuration);
559 }.bind(this); 643 };
560 644
561 // if the duration has changed, invalidate the cached value 645 // if the duration has changed, invalidate the cached value
562 if (oldDuration !== newDuration) { 646 if (oldDuration !== newDuration) {
...@@ -568,31 +652,31 @@ videojs.HlsHandler.prototype.updateDuration = function(playlist) { ...@@ -568,31 +652,31 @@ videojs.HlsHandler.prototype.updateDuration = function(playlist) {
568 this.tech_.trigger('durationchange'); 652 this.tech_.trigger('durationchange');
569 } 653 }
570 } 654 }
571 }; 655 }
572 656
573 /** 657 /**
574 * Clear all buffers and reset any state relevant to the current 658 * Clear all buffers and reset any state relevant to the current
575 * source. After this function is called, the tech should be in a 659 * source. After this function is called, the tech should be in a
576 * state suitable for switching to a different video. 660 * state suitable for switching to a different video.
577 */ 661 */
578 videojs.HlsHandler.prototype.resetSrc_ = function() { 662 resetSrc_() {
579 this.cancelSegmentXhr(); 663 this.cancelSegmentXhr();
580 this.cancelKeyXhr(); 664 this.cancelKeyXhr();
581 665
582 if (this.sourceBuffer && this.mediaSource.readyState === 'open') { 666 if (this.sourceBuffer && this.mediaSource.readyState === 'open') {
583 this.sourceBuffer.abort(); 667 this.sourceBuffer.abort();
584 } 668 }
585 }; 669 }
586 670
587 videojs.HlsHandler.prototype.cancelKeyXhr = function() { 671 cancelKeyXhr() {
588 if (this.keyXhr_) { 672 if (this.keyXhr_) {
589 this.keyXhr_.onreadystatechange = null; 673 this.keyXhr_.onreadystatechange = null;
590 this.keyXhr_.abort(); 674 this.keyXhr_.abort();
591 this.keyXhr_ = null; 675 this.keyXhr_ = null;
592 } 676 }
593 }; 677 }
594 678
595 videojs.HlsHandler.prototype.cancelSegmentXhr = function() { 679 cancelSegmentXhr() {
596 if (this.segmentXhr_) { 680 if (this.segmentXhr_) {
597 // Prevent error handler from running. 681 // Prevent error handler from running.
598 this.segmentXhr_.onreadystatechange = null; 682 this.segmentXhr_.onreadystatechange = null;
...@@ -602,12 +686,12 @@ videojs.HlsHandler.prototype.cancelSegmentXhr = function() { ...@@ -602,12 +686,12 @@ videojs.HlsHandler.prototype.cancelSegmentXhr = function() {
602 686
603 // clear out the segment being processed 687 // clear out the segment being processed
604 this.pendingSegment_ = null; 688 this.pendingSegment_ = null;
605 }; 689 }
606 690
607 /** 691 /**
608 * Abort all outstanding work and cleanup. 692 * Abort all outstanding work and cleanup.
609 */ 693 */
610 videojs.HlsHandler.prototype.dispose = function() { 694 dispose() {
611 this.stopCheckingBuffer_(); 695 this.stopCheckingBuffer_();
612 696
613 if (this.playlists) { 697 if (this.playlists) {
...@@ -615,36 +699,35 @@ videojs.HlsHandler.prototype.dispose = function() { ...@@ -615,36 +699,35 @@ videojs.HlsHandler.prototype.dispose = function() {
615 } 699 }
616 700
617 this.resetSrc_(); 701 this.resetSrc_();
618 Component.prototype.dispose.call(this); 702 super.dispose();
619 }; 703 }
620 704
621 /** 705 /**
622 * Chooses the appropriate media playlist based on the current 706 * Chooses the appropriate media playlist based on the current
623 * bandwidth estimate and the player size. 707 * bandwidth estimate and the player size.
624 * @return the highest bitrate playlist less than the currently detected 708 * @return the highest bitrate playlist less than the currently detected
625 * bandwidth, accounting for some amount of bandwidth variance 709 * bandwidth, accounting for some amount of bandwidth variance
626 */ 710 */
627 videojs.HlsHandler.prototype.selectPlaylist = function () { 711 selectPlaylist() {
628 var 712 let effectiveBitrate;
629 effectiveBitrate, 713 let sortedPlaylists = this.playlists.master.playlists.slice();
630 sortedPlaylists = this.playlists.master.playlists.slice(), 714 let bandwidthPlaylists = [];
631 bandwidthPlaylists = [], 715 let now = +new Date();
632 now = +new Date(), 716 let i;
633 i, 717 let variant;
634 variant, 718 let bandwidthBestVariant;
635 bandwidthBestVariant, 719 let resolutionPlusOne;
636 resolutionPlusOne, 720 let resolutionBestVariant;
637 resolutionBestVariant, 721 let width;
638 width, 722 let height;
639 height; 723
640 724 sortedPlaylists.sort(Hls.comparePlaylistBandwidth);
641 sortedPlaylists.sort(videojs.Hls.comparePlaylistBandwidth);
642 725
643 // filter out any playlists that have been excluded due to 726 // filter out any playlists that have been excluded due to
644 // incompatible configurations or playback errors 727 // incompatible configurations or playback errors
645 sortedPlaylists = sortedPlaylists.filter(function(variant) { 728 sortedPlaylists = sortedPlaylists.filter((localvariant) => {
646 if (variant.excludeUntil !== undefined) { 729 if (typeof localvariant.excludeUntil !== 'undefined') {
647 return now >= variant.excludeUntil; 730 return now >= localvariant.excludeUntil;
648 } 731 }
649 return true; 732 return true;
650 }); 733 });
...@@ -676,9 +759,10 @@ videojs.HlsHandler.prototype.selectPlaylist = function () { ...@@ -676,9 +759,10 @@ videojs.HlsHandler.prototype.selectPlaylist = function () {
676 i = bandwidthPlaylists.length; 759 i = bandwidthPlaylists.length;
677 760
678 // sort variants by resolution 761 // sort variants by resolution
679 bandwidthPlaylists.sort(videojs.Hls.comparePlaylistResolution); 762 bandwidthPlaylists.sort(Hls.comparePlaylistResolution);
680 763
681 // forget our old variant from above, or we might choose that in high-bandwidth scenarios 764 // forget our old variant from above,
765 // or we might choose that in high-bandwidth scenarios
682 // (this could be the lowest bitrate rendition as we go through all of them above) 766 // (this could be the lowest bitrate rendition as we go through all of them above)
683 variant = null; 767 variant = null;
684 768
...@@ -712,9 +796,10 @@ videojs.HlsHandler.prototype.selectPlaylist = function () { ...@@ -712,9 +796,10 @@ videojs.HlsHandler.prototype.selectPlaylist = function () {
712 // if both dimensions are less than the player use the 796 // if both dimensions are less than the player use the
713 // previous (next-largest) variant 797 // previous (next-largest) variant
714 break; 798 break;
715 } else if (!resolutionPlusOne || 799 } else if (!resolutionPlusOne || (variant.attributes.RESOLUTION.width <
716 (variant.attributes.RESOLUTION.width < resolutionPlusOne.attributes.RESOLUTION.width && 800 resolutionPlusOne.attributes.RESOLUTION.width &&
717 variant.attributes.RESOLUTION.height < resolutionPlusOne.attributes.RESOLUTION.height)) { 801 variant.attributes.RESOLUTION.height <
802 resolutionPlusOne.attributes.RESOLUTION.height)) {
718 // If we still haven't found a good match keep a 803 // If we still haven't found a good match keep a
719 // reference to the previous variant for the next loop 804 // reference to the previous variant for the next loop
720 // iteration 805 // iteration
...@@ -728,13 +813,16 @@ videojs.HlsHandler.prototype.selectPlaylist = function () { ...@@ -728,13 +813,16 @@ videojs.HlsHandler.prototype.selectPlaylist = function () {
728 } 813 }
729 814
730 // fallback chain of variants 815 // fallback chain of variants
731 return resolutionPlusOne || resolutionBestVariant || bandwidthBestVariant || sortedPlaylists[0]; 816 return resolutionPlusOne ||
732 }; 817 resolutionBestVariant ||
818 bandwidthBestVariant ||
819 sortedPlaylists[0];
820 }
733 821
734 /** 822 /**
735 * Periodically request new segments and append video data. 823 * Periodically request new segments and append video data.
736 */ 824 */
737 videojs.HlsHandler.prototype.checkBuffer_ = function() { 825 checkBuffer_() {
738 // calling this method directly resets any outstanding buffer checks 826 // calling this method directly resets any outstanding buffer checks
739 if (this.checkBufferTimeout_) { 827 if (this.checkBufferTimeout_) {
740 window.clearTimeout(this.checkBufferTimeout_); 828 window.clearTimeout(this.checkBufferTimeout_);
...@@ -747,101 +835,44 @@ videojs.HlsHandler.prototype.checkBuffer_ = function() { ...@@ -747,101 +835,44 @@ videojs.HlsHandler.prototype.checkBuffer_ = function() {
747 // wait awhile and try again 835 // wait awhile and try again
748 this.checkBufferTimeout_ = window.setTimeout((this.checkBuffer_).bind(this), 836 this.checkBufferTimeout_ = window.setTimeout((this.checkBuffer_).bind(this),
749 bufferCheckInterval); 837 bufferCheckInterval);
750 }; 838 }
751 839
752 /** 840 /**
753 * Setup a periodic task to request new segments if necessary and 841 * Setup a periodic task to request new segments if necessary and
754 * append bytes into the SourceBuffer. 842 * append bytes into the SourceBuffer.
755 */ 843 */
756 videojs.HlsHandler.prototype.startCheckingBuffer_ = function() { 844 startCheckingBuffer_() {
757 this.checkBuffer_(); 845 this.checkBuffer_();
758 }; 846 }
759 847
760 /** 848 /**
761 * Stop the periodic task requesting new segments and feeding the 849 * Stop the periodic task requesting new segments and feeding the
762 * SourceBuffer. 850 * SourceBuffer.
763 */ 851 */
764 videojs.HlsHandler.prototype.stopCheckingBuffer_ = function() { 852 stopCheckingBuffer_() {
765 if (this.checkBufferTimeout_) { 853 if (this.checkBufferTimeout_) {
766 window.clearTimeout(this.checkBufferTimeout_); 854 window.clearTimeout(this.checkBufferTimeout_);
767 this.checkBufferTimeout_ = null; 855 this.checkBufferTimeout_ = null;
768 } 856 }
769 };
770
771 var filterBufferedRanges = function(predicate) {
772 return function(time) {
773 var
774 i,
775 ranges = [],
776 tech = this.tech_,
777 // !!The order of the next two assignments is important!!
778 // `currentTime` must be equal-to or greater-than the start of the
779 // buffered range. Flash executes out-of-process so, every value can
780 // change behind the scenes from line-to-line. By reading `currentTime`
781 // after `buffered`, we ensure that it is always a current or later
782 // value during playback.
783 buffered = tech.buffered();
784
785
786 if (time === undefined) {
787 time = tech.currentTime();
788 }
789
790 if (buffered && buffered.length) {
791 // Search for a range containing the play-head
792 for (i = 0; i < buffered.length; i++) {
793 if (predicate(buffered.start(i), buffered.end(i), time)) {
794 ranges.push([buffered.start(i), buffered.end(i)]);
795 }
796 } 857 }
797 }
798
799 return videojs.createTimeRanges(ranges);
800 };
801 };
802
803 /**
804 * Attempts to find the buffered TimeRange that contains the specified
805 * time, or where playback is currently happening if no specific time
806 * is specified.
807 * @param time (optional) {number} the time to filter on. Defaults to
808 * currentTime.
809 * @return a new TimeRanges object.
810 */
811 videojs.HlsHandler.prototype.findBufferedRange_ = filterBufferedRanges(function(start, end, time) {
812 return start - TIME_FUDGE_FACTOR <= time &&
813 end + TIME_FUDGE_FACTOR >= time;
814 });
815
816 /**
817 * Returns the TimeRanges that begin at or later than the specified
818 * time.
819 * @param time (optional) {number} the time to filter on. Defaults to
820 * currentTime.
821 * @return a new TimeRanges object.
822 */
823 videojs.HlsHandler.prototype.findNextBufferedRange_ = filterBufferedRanges(function(start, end, time) {
824 return start - TIME_FUDGE_FACTOR >= time;
825 });
826 858
827 /** 859 /**
828 * Determines whether there is enough video data currently in the buffer 860 * Determines whether there is enough video data currently in the buffer
829 * and downloads a new segment if the buffered time is less than the goal. 861 * and downloads a new segment if the buffered time is less than the goal.
830 * @param seekToTime (optional) {number} the offset into the downloaded segment 862 * @param seekToTime (optional) {number} the offset into the downloaded segment
831 * to seek to, in seconds 863 * to seek to, in seconds
832 */ 864 */
833 videojs.HlsHandler.prototype.fillBuffer = function(mediaIndex) { 865 fillBuffer(mediaIndex) {
834 var 866 let tech = this.tech_;
835 tech = this.tech_, 867 let currentTime = tech.currentTime();
836 currentTime = tech.currentTime(), 868 let hasBufferedContent = (this.tech_.buffered().length !== 0);
837 hasBufferedContent = (this.tech_.buffered().length !== 0), 869 let currentBuffered = this.findBufferedRange_();
838 currentBuffered = this.findBufferedRange_(), 870 let outsideBufferedRanges = !(currentBuffered && currentBuffered.length);
839 outsideBufferedRanges = !(currentBuffered && currentBuffered.length), 871 let currentBufferedEnd = 0;
840 currentBufferedEnd = 0, 872 let bufferedTime = 0;
841 bufferedTime = 0, 873 let segment;
842 segment, 874 let segmentInfo;
843 segmentInfo, 875 let segmentTimestampOffset;
844 segmentTimestampOffset;
845 876
846 // if preload is set to "none", do not download segments until playback is requested 877 // if preload is set to "none", do not download segments until playback is requested
847 if (this.loadingState_ !== 'segments') { 878 if (this.loadingState_ !== 'segments') {
...@@ -864,7 +895,7 @@ videojs.HlsHandler.prototype.fillBuffer = function(mediaIndex) { ...@@ -864,7 +895,7 @@ videojs.HlsHandler.prototype.fillBuffer = function(mediaIndex) {
864 } 895 }
865 896
866 // if no segments are available, do nothing 897 // if no segments are available, do nothing
867 if (this.playlists.state === "HAVE_NOTHING" || 898 if (this.playlists.state === 'HAVE_NOTHING' ||
868 !this.playlists.media() || 899 !this.playlists.media() ||
869 !this.playlists.media().segments) { 900 !this.playlists.media().segments) {
870 return; 901 return;
...@@ -875,7 +906,7 @@ videojs.HlsHandler.prototype.fillBuffer = function(mediaIndex) { ...@@ -875,7 +906,7 @@ videojs.HlsHandler.prototype.fillBuffer = function(mediaIndex) {
875 return; 906 return;
876 } 907 }
877 908
878 if (mediaIndex === undefined) { 909 if (typeof mediaIndex === 'undefined') {
879 if (currentBuffered && currentBuffered.length) { 910 if (currentBuffered && currentBuffered.length) {
880 currentBufferedEnd = currentBuffered.end(0); 911 currentBufferedEnd = currentBuffered.end(0);
881 mediaIndex = this.playlists.getMediaIndexForTime_(currentBufferedEnd); 912 mediaIndex = this.playlists.getMediaIndexForTime_(currentBufferedEnd);
...@@ -883,7 +914,7 @@ videojs.HlsHandler.prototype.fillBuffer = function(mediaIndex) { ...@@ -883,7 +914,7 @@ videojs.HlsHandler.prototype.fillBuffer = function(mediaIndex) {
883 914
884 // if there is plenty of content in the buffer and we're not 915 // if there is plenty of content in the buffer and we're not
885 // seeking, relax for awhile 916 // seeking, relax for awhile
886 if (bufferedTime >= videojs.Hls.GOAL_BUFFER_LENGTH) { 917 if (bufferedTime >= Hls.GOAL_BUFFER_LENGTH) {
887 return; 918 return;
888 } 919 }
889 } else { 920 } else {
...@@ -900,7 +931,8 @@ videojs.HlsHandler.prototype.fillBuffer = function(mediaIndex) { ...@@ -900,7 +931,8 @@ videojs.HlsHandler.prototype.fillBuffer = function(mediaIndex) {
900 // we have entered a state where we are fetching the same segment, 931 // we have entered a state where we are fetching the same segment,
901 // try to walk forward 932 // try to walk forward
902 if (this.lastSegmentLoaded_ && 933 if (this.lastSegmentLoaded_ &&
903 this.playlistUriToUrl(this.lastSegmentLoaded_.uri) === this.playlistUriToUrl(segment.uri) && 934 this.playlistUriToUrl(this.lastSegmentLoaded_.uri) ===
935 this.playlistUriToUrl(segment.uri) &&
904 this.lastSegmentLoaded_.byterange === segment.byterange) { 936 this.lastSegmentLoaded_.byterange === segment.byterange) {
905 return this.fillBuffer(mediaIndex + 1); 937 return this.fillBuffer(mediaIndex + 1);
906 } 938 }
...@@ -910,12 +942,12 @@ videojs.HlsHandler.prototype.fillBuffer = function(mediaIndex) { ...@@ -910,12 +942,12 @@ videojs.HlsHandler.prototype.fillBuffer = function(mediaIndex) {
910 // resolve the segment URL relative to the playlist 942 // resolve the segment URL relative to the playlist
911 uri: this.playlistUriToUrl(segment.uri), 943 uri: this.playlistUriToUrl(segment.uri),
912 // the segment's mediaIndex & mediaSequence at the time it was requested 944 // the segment's mediaIndex & mediaSequence at the time it was requested
913 mediaIndex: mediaIndex, 945 mediaIndex,
914 mediaSequence: this.playlists.media().mediaSequence, 946 mediaSequence: this.playlists.media().mediaSequence,
915 // the segment's playlist 947 // the segment's playlist
916 playlist: this.playlists.media(), 948 playlist: this.playlists.media(),
917 // The state of the buffer when this segment was requested 949 // The state of the buffer when this segment was requested
918 currentBufferedEnd: currentBufferedEnd, 950 currentBufferedEnd,
919 // unencrypted bytes of the segment 951 // unencrypted bytes of the segment
920 bytes: null, 952 bytes: null,
921 // when a key is defined for this segment, the encrypted bytes 953 // when a key is defined for this segment, the encrypted bytes
...@@ -932,7 +964,7 @@ videojs.HlsHandler.prototype.fillBuffer = function(mediaIndex) { ...@@ -932,7 +964,7 @@ videojs.HlsHandler.prototype.fillBuffer = function(mediaIndex) {
932 }; 964 };
933 965
934 if (mediaIndex > 0) { 966 if (mediaIndex > 0) {
935 segmentTimestampOffset = videojs.Hls.Playlist.duration(segmentInfo.playlist, 967 segmentTimestampOffset = Hls.Playlist.duration(segmentInfo.playlist,
936 segmentInfo.playlist.mediaSequence + mediaIndex) + this.playlists.expired_; 968 segmentInfo.playlist.mediaSequence + mediaIndex) + this.playlists.expired_;
937 } 969 }
938 970
...@@ -955,43 +987,50 @@ videojs.HlsHandler.prototype.fillBuffer = function(mediaIndex) { ...@@ -955,43 +987,50 @@ videojs.HlsHandler.prototype.fillBuffer = function(mediaIndex) {
955 } 987 }
956 988
957 this.loadSegment(segmentInfo); 989 this.loadSegment(segmentInfo);
958 }; 990 }
991
992 playlistUriToUrl(segmentRelativeUrl) {
993 let playListUrl;
959 994
960 videojs.HlsHandler.prototype.playlistUriToUrl = function(segmentRelativeUrl) {
961 var playListUrl;
962 // resolve the segment URL relative to the playlist 995 // resolve the segment URL relative to the playlist
963 if (this.playlists.media().uri === this.source_.src) { 996 if (this.playlists.media().uri === this.source_.src) {
964 playListUrl = resolveUrl(this.source_.src, segmentRelativeUrl); 997 playListUrl = resolveUrl(this.source_.src, segmentRelativeUrl);
965 } else { 998 } else {
966 playListUrl = resolveUrl(resolveUrl(this.source_.src, this.playlists.media().uri || ''), segmentRelativeUrl); 999 playListUrl =
1000 resolveUrl(resolveUrl(this.source_.src, this.playlists.media().uri || ''),
1001 segmentRelativeUrl);
967 } 1002 }
968 return playListUrl; 1003 return playListUrl;
969 }; 1004 }
970 1005
971 /* Turns segment byterange into a string suitable for use in 1006 /*
1007 * Turns segment byterange into a string suitable for use in
972 * HTTP Range requests 1008 * HTTP Range requests
973 */ 1009 */
974 videojs.HlsHandler.prototype.byterangeStr_ = function(byterange) { 1010 byterangeStr_(byterange) {
975 var byterangeStart, byterangeEnd; 1011 let byterangeStart;
1012 let byterangeEnd;
976 1013
977 // `byterangeEnd` is one less than `offset + length` because the HTTP range 1014 // `byterangeEnd` is one less than `offset + length` because the HTTP range
978 // header uses inclusive ranges 1015 // header uses inclusive ranges
979 byterangeEnd = byterange.offset + byterange.length - 1; 1016 byterangeEnd = byterange.offset + byterange.length - 1;
980 byterangeStart = byterange.offset; 1017 byterangeStart = byterange.offset;
981 return "bytes=" + byterangeStart + "-" + byterangeEnd; 1018 return 'bytes=' + byterangeStart + '-' + byterangeEnd;
982 }; 1019 }
983 1020
984 /* Defines headers for use in the xhr request for a particular segment. 1021 /*
1022 * Defines headers for use in the xhr request for a particular segment.
985 */ 1023 */
986 videojs.HlsHandler.prototype.segmentXhrHeaders_ = function(segment) { 1024 segmentXhrHeaders_(segment) {
987 var headers = {}; 1025 let headers = {};
1026
988 if ('byterange' in segment) { 1027 if ('byterange' in segment) {
989 headers['Range'] = this.byterangeStr_(segment.byterange); 1028 headers.Range = this.byterangeStr_(segment.byterange);
990 } 1029 }
991 return headers; 1030 return headers;
992 }; 1031 }
993 1032
994 /* 1033 /*
995 * Sets `bandwidth`, `segmentXhrTime`, and appends to the `bytesReceived. 1034 * Sets `bandwidth`, `segmentXhrTime`, and appends to the `bytesReceived.
996 * Expects an object with: 1035 * Expects an object with:
997 * * `roundTripTime` - the round trip time for the request we're setting the time for 1036 * * `roundTripTime` - the round trip time for the request we're setting the time for
...@@ -999,22 +1038,23 @@ videojs.HlsHandler.prototype.segmentXhrHeaders_ = function(segment) { ...@@ -999,22 +1038,23 @@ videojs.HlsHandler.prototype.segmentXhrHeaders_ = function(segment) {
999 * * `bytesReceived` - amount of bytes downloaded 1038 * * `bytesReceived` - amount of bytes downloaded
1000 * `bandwidth` is the only required property. 1039 * `bandwidth` is the only required property.
1001 */ 1040 */
1002 videojs.HlsHandler.prototype.setBandwidth = function(xhr) { 1041 setBandwidth(localXhr) {
1003 // calculate the download bandwidth 1042 // calculate the download bandwidth
1004 this.segmentXhrTime = xhr.roundTripTime; 1043 this.segmentXhrTime = localXhr.roundTripTime;
1005 this.bandwidth = xhr.bandwidth; 1044 this.bandwidth = localXhr.bandwidth;
1006 this.bytesReceived += xhr.bytesReceived || 0; 1045 this.bytesReceived += localXhr.bytesReceived || 0;
1007 1046
1008 this.tech_.trigger('bandwidthupdate'); 1047 this.tech_.trigger('bandwidthupdate');
1009 }; 1048 }
1010 1049
1011 /* 1050 /*
1012 * Blacklists a playlist when an error occurs for a set amount of time 1051 * Blacklists a playlist when an error occurs for a set amount of time
1013 * making it unavailable for selection by the rendition selection algorithm 1052 * making it unavailable for selection by the rendition selection algorithm
1014 * and then forces a new playlist (rendition) selection. 1053 * and then forces a new playlist (rendition) selection.
1015 */ 1054 */
1016 videojs.HlsHandler.prototype.blacklistCurrentPlaylist_ = function(error) { 1055 blacklistCurrentPlaylist_(error) {
1017 var currentPlaylist, nextPlaylist; 1056 let currentPlaylist;
1057 let nextPlaylist;
1018 1058
1019 // If the `error` was generated by the playlist loader, it will contain 1059 // If the `error` was generated by the playlist loader, it will contain
1020 // the playlist we were trying to load (but failed) and that should be 1060 // the playlist we were trying to load (but failed) and that should be
...@@ -1036,26 +1076,27 @@ videojs.HlsHandler.prototype.blacklistCurrentPlaylist_ = function(error) { ...@@ -1036,26 +1076,27 @@ videojs.HlsHandler.prototype.blacklistCurrentPlaylist_ = function(error) {
1036 nextPlaylist = this.selectPlaylist(); 1076 nextPlaylist = this.selectPlaylist();
1037 1077
1038 if (nextPlaylist) { 1078 if (nextPlaylist) {
1039 videojs.log.warn('Problem encountered with the current HLS playlist. Switching to another playlist.'); 1079 videojs.log.warn('Problem encountered with the current ' +
1080 'HLS playlist. Switching to another playlist.');
1040 1081
1041 return this.playlists.media(nextPlaylist); 1082 return this.playlists.media(nextPlaylist);
1042 } else { 1083 }
1043 videojs.log.warn('Problem encountered with the current HLS playlist. No suitable alternatives found.'); 1084 videojs.log.warn('Problem encountered with the current ' +
1085 'HLS playlist. No suitable alternatives found.');
1044 // We have no more playlists we can select so we must fail 1086 // We have no more playlists we can select so we must fail
1045 this.error = error; 1087 this.error = error;
1046 return this.mediaSource.endOfStream('network'); 1088 return this.mediaSource.endOfStream('network');
1047 } 1089 }
1048 };
1049 1090
1050 videojs.HlsHandler.prototype.loadSegment = function(segmentInfo) { 1091 loadSegment(segmentInfo) {
1051 var 1092 let segment = segmentInfo.playlist.segments[segmentInfo.mediaIndex];
1052 self = this, 1093 let removeToTime = 0;
1053 segment = segmentInfo.playlist.segments[segmentInfo.mediaIndex], 1094 let seekable = this.seekable();
1054 removeToTime = 0,
1055 seekable = this.seekable();
1056 1095
1057 // Chrome has a hard limit of 150mb of buffer and a very conservative "garbage collector" 1096 // Chrome has a hard limit of 150mb of
1058 // We manually clear out the old buffer to ensure we don't trigger the QuotaExceeded error 1097 // buffer and a very conservative "garbage collector"
1098 // We manually clear out the old buffer to ensure
1099 // we don't trigger the QuotaExceeded error
1059 // on the source buffer during subsequent appends 1100 // on the source buffer during subsequent appends
1060 if (this.sourceBuffer && !this.sourceBuffer.updating) { 1101 if (this.sourceBuffer && !this.sourceBuffer.updating) {
1061 // If we have a seekable range use that as the limit for what can be removed safely 1102 // If we have a seekable range use that as the limit for what can be removed safely
...@@ -1077,7 +1118,7 @@ videojs.HlsHandler.prototype.loadSegment = function(segmentInfo) { ...@@ -1077,7 +1118,7 @@ videojs.HlsHandler.prototype.loadSegment = function(segmentInfo) {
1077 } 1118 }
1078 1119
1079 // request the next segment 1120 // request the next segment
1080 this.segmentXhr_ = videojs.Hls.xhr({ 1121 this.segmentXhr_ = Hls.xhr({
1081 uri: segmentInfo.uri, 1122 uri: segmentInfo.uri,
1082 responseType: 'arraybuffer', 1123 responseType: 'arraybuffer',
1083 withCredentials: this.source_.withCredentials, 1124 withCredentials: this.source_.withCredentials,
...@@ -1086,25 +1127,25 @@ videojs.HlsHandler.prototype.loadSegment = function(segmentInfo) { ...@@ -1086,25 +1127,25 @@ videojs.HlsHandler.prototype.loadSegment = function(segmentInfo) {
1086 // decrease in network performance or a server issue. 1127 // decrease in network performance or a server issue.
1087 timeout: (segment.duration * 1.5) * 1000, 1128 timeout: (segment.duration * 1.5) * 1000,
1088 headers: this.segmentXhrHeaders_(segment) 1129 headers: this.segmentXhrHeaders_(segment)
1089 }, function(error, request) { 1130 }, (error, request) => {
1090 // This is a timeout of a previously aborted segment request 1131 // This is a timeout of a previously aborted segment request
1091 // so simply ignore it 1132 // so simply ignore it
1092 if (!self.segmentXhr_ || request !== self.segmentXhr_) { 1133 if (!this.segmentXhr_ || request !== this.segmentXhr_) {
1093 return; 1134 return;
1094 } 1135 }
1095 1136
1096 // the segment request is no longer outstanding 1137 // the segment request is no longer outstanding
1097 self.segmentXhr_ = null; 1138 this.segmentXhr_ = null;
1098 1139
1099 // if a segment request times out, we may have better luck with another playlist 1140 // if a segment request times out, we may have better luck with another playlist
1100 if (request.timedout) { 1141 if (request.timedout) {
1101 self.bandwidth = 1; 1142 this.bandwidth = 1;
1102 return self.playlists.media(self.selectPlaylist()); 1143 return this.playlists.media(this.selectPlaylist());
1103 } 1144 }
1104 1145
1105 // otherwise, trigger a network error 1146 // otherwise, trigger a network error
1106 if (!request.aborted && error) { 1147 if (!request.aborted && error) {
1107 return self.blacklistCurrentPlaylist_({ 1148 return this.blacklistCurrentPlaylist_({
1108 status: request.status, 1149 status: request.status,
1109 message: 'HLS segment request error at URL: ' + segmentInfo.uri, 1150 message: 'HLS segment request error at URL: ' + segmentInfo.uri,
1110 code: (request.status >= 500) ? 4 : 2 1151 code: (request.status >= 500) ? 4 : 2
...@@ -1116,8 +1157,8 @@ videojs.HlsHandler.prototype.loadSegment = function(segmentInfo) { ...@@ -1116,8 +1157,8 @@ videojs.HlsHandler.prototype.loadSegment = function(segmentInfo) {
1116 return; 1157 return;
1117 } 1158 }
1118 1159
1119 self.lastSegmentLoaded_ = segment; 1160 this.lastSegmentLoaded_ = segment;
1120 self.setBandwidth(request); 1161 this.setBandwidth(request);
1121 1162
1122 if (segment.key) { 1163 if (segment.key) {
1123 segmentInfo.encryptedBytes = new Uint8Array(request.response); 1164 segmentInfo.encryptedBytes = new Uint8Array(request.response);
...@@ -1125,28 +1166,26 @@ videojs.HlsHandler.prototype.loadSegment = function(segmentInfo) { ...@@ -1125,28 +1166,26 @@ videojs.HlsHandler.prototype.loadSegment = function(segmentInfo) {
1125 segmentInfo.bytes = new Uint8Array(request.response); 1166 segmentInfo.bytes = new Uint8Array(request.response);
1126 } 1167 }
1127 1168
1128 self.pendingSegment_ = segmentInfo; 1169 this.pendingSegment_ = segmentInfo;
1129 1170
1130 self.tech_.trigger('progress'); 1171 this.tech_.trigger('progress');
1131 self.drainBuffer(); 1172 this.drainBuffer();
1132 1173
1133 // figure out what stream the next segment should be downloaded from 1174 // figure out what stream the next segment should be downloaded from
1134 // with the updated bandwidth information 1175 // with the updated bandwidth information
1135 self.playlists.media(self.selectPlaylist()); 1176 this.playlists.media(this.selectPlaylist());
1136 }); 1177 });
1137 1178
1138 }; 1179 }
1139 1180
1140 videojs.HlsHandler.prototype.drainBuffer = function() { 1181 drainBuffer() {
1141 var 1182 let segmentInfo;
1142 segmentInfo, 1183 let mediaIndex;
1143 mediaIndex, 1184 let playlist;
1144 playlist, 1185 let bytes;
1145 offset, 1186 let segment;
1146 bytes, 1187 let decrypter;
1147 segment, 1188 let segIv;
1148 decrypter,
1149 segIv;
1150 1189
1151 // if the buffer is empty or the source buffer hasn't been created 1190 // if the buffer is empty or the source buffer hasn't been created
1152 // yet, do nothing 1191 // yet, do nothing
...@@ -1169,7 +1208,6 @@ videojs.HlsHandler.prototype.drainBuffer = function() { ...@@ -1169,7 +1208,6 @@ videojs.HlsHandler.prototype.drainBuffer = function() {
1169 segmentInfo = this.pendingSegment_; 1208 segmentInfo = this.pendingSegment_;
1170 mediaIndex = segmentInfo.mediaIndex; 1209 mediaIndex = segmentInfo.mediaIndex;
1171 playlist = segmentInfo.playlist; 1210 playlist = segmentInfo.playlist;
1172 offset = segmentInfo.offset;
1173 bytes = segmentInfo.bytes; 1211 bytes = segmentInfo.bytes;
1174 segment = playlist.segments[mediaIndex]; 1212 segment = playlist.segments[mediaIndex];
1175 1213
...@@ -1183,30 +1221,30 @@ videojs.HlsHandler.prototype.drainBuffer = function() { ...@@ -1183,30 +1221,30 @@ videojs.HlsHandler.prototype.drainBuffer = function() {
1183 code: 4 1221 code: 4
1184 }); 1222 });
1185 } else if (!segment.key.bytes) { 1223 } else if (!segment.key.bytes) {
1186
1187 // waiting for the key bytes, try again later 1224 // waiting for the key bytes, try again later
1188 return; 1225 return;
1189 } else if (segmentInfo.decrypter) { 1226 } else if (segmentInfo.decrypter) {
1190
1191 // decryption is in progress, try again later 1227 // decryption is in progress, try again later
1192 return; 1228 return;
1193 } else { 1229 }
1194
1195 // if the media sequence is greater than 2^32, the IV will be incorrect 1230 // if the media sequence is greater than 2^32, the IV will be incorrect
1196 // assuming 10s segments, that would be about 1300 years 1231 // assuming 10s segments, that would be about 1300 years
1197 segIv = segment.key.iv || new Uint32Array([0, 0, 0, mediaIndex + playlist.mediaSequence]); 1232 segIv = segment.key.iv ||
1233 new Uint32Array([0, 0, 0, mediaIndex + playlist.mediaSequence]);
1198 1234
1199 // create a decrypter to incrementally decrypt the segment 1235 // create a decrypter to incrementally decrypt the segment
1200 decrypter = new videojs.Hls.Decrypter(segmentInfo.encryptedBytes, 1236 decrypter = new Hls.Decrypter(segmentInfo.encryptedBytes,
1201 segment.key.bytes, 1237 segment.key.bytes,
1202 segIv, 1238 segIv,
1203 function(err, bytes) { 1239 function(err, localBytes) {
1204 segmentInfo.bytes = bytes; 1240 if (err) {
1241 throw new Error(err);
1242 }
1243 segmentInfo.bytes = localBytes;
1205 }); 1244 });
1206 segmentInfo.decrypter = decrypter; 1245 segmentInfo.decrypter = decrypter;
1207 return; 1246 return;
1208 } 1247 }
1209 }
1210 1248
1211 this.pendingSegment_.buffered = this.tech_.buffered(); 1249 this.pendingSegment_.buffered = this.tech_.buffered();
1212 1250
...@@ -1216,18 +1254,17 @@ videojs.HlsHandler.prototype.drainBuffer = function() { ...@@ -1216,18 +1254,17 @@ videojs.HlsHandler.prototype.drainBuffer = function() {
1216 1254
1217 // the segment is asynchronously added to the current buffered data 1255 // the segment is asynchronously added to the current buffered data
1218 this.sourceBuffer.appendBuffer(bytes); 1256 this.sourceBuffer.appendBuffer(bytes);
1219 }; 1257 }
1220 1258
1221 videojs.HlsHandler.prototype.updateEndHandler_ = function () { 1259 updateEndHandler_() {
1222 var 1260 let segmentInfo = this.pendingSegment_;
1223 segmentInfo = this.pendingSegment_, 1261 let segment;
1224 segment, 1262 let segments;
1225 segments, 1263 let playlist;
1226 playlist, 1264 let currentMediaIndex;
1227 currentMediaIndex, 1265 let currentBuffered;
1228 currentBuffered, 1266 let seekable;
1229 seekable, 1267 let timelineUpdate;
1230 timelineUpdate;
1231 1268
1232 this.pendingSegment_ = null; 1269 this.pendingSegment_ = null;
1233 1270
...@@ -1238,7 +1275,8 @@ videojs.HlsHandler.prototype.updateEndHandler_ = function () { ...@@ -1238,7 +1275,8 @@ videojs.HlsHandler.prototype.updateEndHandler_ = function () {
1238 1275
1239 playlist = this.playlists.media(); 1276 playlist = this.playlists.media();
1240 segments = playlist.segments; 1277 segments = playlist.segments;
1241 currentMediaIndex = segmentInfo.mediaIndex + (segmentInfo.mediaSequence - playlist.mediaSequence); 1278 currentMediaIndex = segmentInfo.mediaIndex +
1279 (segmentInfo.mediaSequence - playlist.mediaSequence);
1242 currentBuffered = this.findBufferedRange_(); 1280 currentBuffered = this.findBufferedRange_();
1243 1281
1244 // if we switched renditions don't try to add segment timeline 1282 // if we switched renditions don't try to add segment timeline
...@@ -1260,16 +1298,17 @@ videojs.HlsHandler.prototype.updateEndHandler_ = function () { ...@@ -1260,16 +1298,17 @@ videojs.HlsHandler.prototype.updateEndHandler_ = function () {
1260 currentBuffered.length === 0) { 1298 currentBuffered.length === 0) {
1261 if (seekable.length && 1299 if (seekable.length &&
1262 this.tech_.currentTime() < seekable.start(0)) { 1300 this.tech_.currentTime() < seekable.start(0)) {
1263 var next = this.findNextBufferedRange_(); 1301 let next = this.findNextBufferedRange_();
1302
1264 if (next.length) { 1303 if (next.length) {
1265 videojs.log('tried seeking to', this.tech_.currentTime(), 'but that was too early, retrying at', next.start(0)); 1304 videojs.log('tried seeking to', this.tech_.currentTime(),
1305 'but that was too early, retrying at', next.start(0));
1266 this.tech_.setCurrentTime(next.start(0) + TIME_FUDGE_FACTOR); 1306 this.tech_.setCurrentTime(next.start(0) + TIME_FUDGE_FACTOR);
1267 } 1307 }
1268 } 1308 }
1269 } 1309 }
1270 1310
1271 1311 timelineUpdate = Hls.findSoleUncommonTimeRangesEnd_(segmentInfo.buffered,
1272 timelineUpdate = videojs.Hls.findSoleUncommonTimeRangesEnd_(segmentInfo.buffered,
1273 this.tech_.buffered()); 1312 this.tech_.buffered());
1274 1313
1275 if (timelineUpdate && segment) { 1314 if (timelineUpdate && segment) {
...@@ -1299,42 +1338,44 @@ videojs.HlsHandler.prototype.updateEndHandler_ = function () { ...@@ -1299,42 +1338,44 @@ videojs.HlsHandler.prototype.updateEndHandler_ = function () {
1299 // improves subsequent media index calculations. 1338 // improves subsequent media index calculations.
1300 this.fillBuffer(currentMediaIndex + 1); 1339 this.fillBuffer(currentMediaIndex + 1);
1301 return; 1340 return;
1302 }; 1341 }
1303 1342
1304 /** 1343 /**
1305 * Attempt to retrieve the key for a particular media segment. 1344 * Attempt to retrieve the key for a particular media segment.
1306 */ 1345 */
1307 videojs.HlsHandler.prototype.fetchKey_ = function(segment) { 1346 fetchKey_(segment) {
1308 var key, self, settings, receiveKey; 1347 let key;
1348 let settings;
1349 let receiveKey;
1309 1350
1310 // if there is a pending XHR or no segments, don't do anything 1351 // if there is a pending XHR or no segments, don't do anything
1311 if (this.keyXhr_) { 1352 if (this.keyXhr_) {
1312 return; 1353 return;
1313 } 1354 }
1314 1355
1315 self = this;
1316 settings = this.options_; 1356 settings = this.options_;
1317 1357
1318 /** 1358 /**
1319 * Handle a key XHR response. 1359 * Handle a key XHR response.
1320 */ 1360 */
1321 receiveKey = function(key) { 1361 receiveKey = (keyRecieved) => {
1322 return function(error, request) { 1362 return (error, request) => {
1323 var view; 1363 let view;
1324 self.keyXhr_ = null; 1364
1365 this.keyXhr_ = null;
1325 1366
1326 if (error || !request.response || request.response.byteLength !== 16) { 1367 if (error || !request.response || request.response.byteLength !== 16) {
1327 key.retries = key.retries || 0; 1368 keyRecieved.retries = keyRecieved.retries || 0;
1328 key.retries++; 1369 keyRecieved.retries++;
1329 if (!request.aborted) { 1370 if (!request.aborted) {
1330 // try fetching again 1371 // try fetching again
1331 self.fetchKey_(segment); 1372 this.fetchKey_(segment);
1332 } 1373 }
1333 return; 1374 return;
1334 } 1375 }
1335 1376
1336 view = new DataView(request.response); 1377 view = new DataView(request.response);
1337 key.bytes = new Uint32Array([ 1378 keyRecieved.bytes = new Uint32Array([
1338 view.getUint32(0), 1379 view.getUint32(0),
1339 view.getUint32(4), 1380 view.getUint32(4),
1340 view.getUint32(8), 1381 view.getUint32(8),
...@@ -1342,7 +1383,7 @@ videojs.HlsHandler.prototype.fetchKey_ = function(segment) { ...@@ -1342,7 +1383,7 @@ videojs.HlsHandler.prototype.fetchKey_ = function(segment) {
1342 ]); 1383 ]);
1343 1384
1344 // check to see if this allows us to make progress buffering now 1385 // check to see if this allows us to make progress buffering now
1345 self.checkBuffer_(); 1386 this.checkBuffer_();
1346 }; 1387 };
1347 }; 1388 };
1348 1389
...@@ -1355,135 +1396,105 @@ videojs.HlsHandler.prototype.fetchKey_ = function(segment) { ...@@ -1355,135 +1396,105 @@ videojs.HlsHandler.prototype.fetchKey_ = function(segment) {
1355 1396
1356 // request the key if the retry limit hasn't been reached 1397 // request the key if the retry limit hasn't been reached
1357 if (!key.bytes && !keyFailed(key)) { 1398 if (!key.bytes && !keyFailed(key)) {
1358 this.keyXhr_ = videojs.Hls.xhr({ 1399 this.keyXhr_ = Hls.xhr({
1359 uri: this.playlistUriToUrl(key.uri), 1400 uri: this.playlistUriToUrl(key.uri),
1360 responseType: 'arraybuffer', 1401 responseType: 'arraybuffer',
1361 withCredentials: settings.withCredentials 1402 withCredentials: settings.withCredentials
1362 }, receiveKey(key)); 1403 }, receiveKey(key));
1363 return; 1404 return;
1364 } 1405 }
1365 }; 1406 }
1407 }
1366 1408
1367 /** 1409 /**
1368 * Whether the browser has built-in HLS support. 1410 * Attempts to find the buffered TimeRange that contains the specified
1411 * time, or where playback is currently happening if no specific time
1412 * is specified.
1413 * @param time (optional) {number} the time to filter on. Defaults to
1414 * currentTime.
1415 * @return a new TimeRanges object.
1369 */ 1416 */
1370 videojs.Hls.supportsNativeHls = (function() { 1417 HlsHandler.prototype.findBufferedRange_ =
1371 var 1418 filterBufferedRanges(function(start, end, time) {
1372 video = document.createElement('video'), 1419 return start - TIME_FUDGE_FACTOR <= time &&
1373 xMpegUrl, 1420 end + TIME_FUDGE_FACTOR >= time;
1374 vndMpeg; 1421 });
1375
1376 // native HLS is definitely not supported if HTML5 video isn't
1377 if (!videojs.getComponent('Html5').isSupported()) {
1378 return false;
1379 }
1380
1381 xMpegUrl = video.canPlayType('application/x-mpegURL');
1382 vndMpeg = video.canPlayType('application/vnd.apple.mpegURL');
1383 return (/probably|maybe/).test(xMpegUrl) ||
1384 (/probably|maybe/).test(vndMpeg);
1385 })();
1386
1387 // HLS is a source handler, not a tech. Make sure attempts to use it
1388 // as one do not cause exceptions.
1389 videojs.Hls.isSupported = function() {
1390 return videojs.log.warn('HLS is no longer a tech. Please remove it from ' +
1391 'your player\'s techOrder.');
1392 };
1393
1394 /** 1422 /**
1395 * A comparator function to sort two playlist object by bandwidth. 1423 * Returns the TimeRanges that begin at or later than the specified
1396 * @param left {object} a media playlist object 1424 * time.
1397 * @param right {object} a media playlist object 1425 * @param time (optional) {number} the time to filter on. Defaults to
1398 * @return {number} Greater than zero if the bandwidth attribute of 1426 * currentTime.
1399 * left is greater than the corresponding attribute of right. Less 1427 * @return a new TimeRanges object.
1400 * than zero if the bandwidth of right is greater than left and
1401 * exactly zero if the two are equal.
1402 */ 1428 */
1403 videojs.Hls.comparePlaylistBandwidth = function(left, right) { 1429 HlsHandler.prototype.findNextBufferedRange_ =
1404 var leftBandwidth, rightBandwidth; 1430 filterBufferedRanges(function(start, end, time) {
1405 if (left.attributes && left.attributes.BANDWIDTH) { 1431 return start - TIME_FUDGE_FACTOR >= time;
1406 leftBandwidth = left.attributes.BANDWIDTH; 1432 });
1407 }
1408 leftBandwidth = leftBandwidth || window.Number.MAX_VALUE;
1409 if (right.attributes && right.attributes.BANDWIDTH) {
1410 rightBandwidth = right.attributes.BANDWIDTH;
1411 }
1412 rightBandwidth = rightBandwidth || window.Number.MAX_VALUE;
1413
1414 return leftBandwidth - rightBandwidth;
1415 };
1416 1433
1417 /** 1434 /**
1418 * A comparator function to sort two playlist object by resolution (width). 1435 * The Source Handler object, which informs video.js what additional
1419 * @param left {object} a media playlist object 1436 * MIME types are supported and sets up playback. It is registered
1420 * @param right {object} a media playlist object 1437 * automatically to the appropriate tech based on the capabilities of
1421 * @return {number} Greater than zero if the resolution.width attribute of 1438 * the browser it is running in. It is not necessary to use or modify
1422 * left is greater than the corresponding attribute of right. Less 1439 * this object in normal usage.
1423 * than zero if the resolution.width of right is greater than left and
1424 * exactly zero if the two are equal.
1425 */ 1440 */
1426 videojs.Hls.comparePlaylistResolution = function(left, right) { 1441 const HlsSourceHandler = function(mode) {
1427 var leftWidth, rightWidth; 1442 return {
1428 1443 canHandleSource(srcObj) {
1429 if (left.attributes && left.attributes.RESOLUTION && left.attributes.RESOLUTION.width) { 1444 return HlsSourceHandler.canPlayType(srcObj.type);
1430 leftWidth = left.attributes.RESOLUTION.width; 1445 },
1446 handleSource(source, tech) {
1447 if (mode === 'flash') {
1448 // We need to trigger this asynchronously to give others the chance
1449 // to bind to the event when a source is set at player creation
1450 tech.setTimeout(function() {
1451 tech.trigger('loadstart');
1452 }, 1);
1431 } 1453 }
1432 1454 tech.hls = new HlsHandler(tech, {
1433 leftWidth = leftWidth || window.Number.MAX_VALUE; 1455 source,
1434 1456 mode
1435 if (right.attributes && right.attributes.RESOLUTION && right.attributes.RESOLUTION.width) { 1457 });
1436 rightWidth = right.attributes.RESOLUTION.width; 1458 tech.hls.src(source.src);
1459 return tech.hls;
1460 },
1461 canPlayType(type) {
1462 return HlsSourceHandler.canPlayType(type);
1437 } 1463 }
1464 };
1465 };
1438 1466
1439 rightWidth = rightWidth || window.Number.MAX_VALUE; 1467 HlsSourceHandler.canPlayType = function(type) {
1468 let mpegurlRE = /^application\/(?:x-|vnd\.apple\.)mpegurl/i;
1440 1469
1441 // NOTE - Fallback to bandwidth sort as appropriate in cases where multiple renditions 1470 // favor native HLS support if it's available
1442 // have the same media dimensions/ resolution 1471 if (Hls.supportsNativeHls()) {
1443 if (leftWidth === rightWidth && left.attributes.BANDWIDTH && right.attributes.BANDWIDTH) { 1472 return false;
1444 return left.attributes.BANDWIDTH - right.attributes.BANDWIDTH;
1445 } else {
1446 return leftWidth - rightWidth;
1447 } 1473 }
1474 return mpegurlRE.test(type);
1448 }; 1475 };
1449 1476
1450 /** 1477 if (typeof videojs.MediaSource === 'undefined' ||
1451 * Constructs a new URI by interpreting a path relative to another 1478 typeof videojs.URL === 'undefined') {
1452 * URI. 1479 videojs.MediaSource = MediaSource;
1453 * @param basePath {string} a relative or absolute URI 1480 videojs.URL = URL;
1454 * @param path {string} a path part to combine with the base 1481 }
1455 * @return {string} a URI that is equivalent to composing `base` 1482
1456 * with `path` 1483 // register source handlers with the appropriate techs
1457 * @see http://stackoverflow.com/questions/470832/getting-an-absolute-url-from-a-relative-one-ie6-issue 1484 if (MediaSource.supportsNativeMediaSources()) {
1458 */ 1485 videojs.getComponent('Html5').registerSourceHandler(HlsSourceHandler('html5'));
1459 resolveUrl = videojs.Hls.resolveUrl = function(basePath, path) { 1486 }
1460 // use the base element to get the browser to handle URI resolution 1487 if (window.Uint8Array) {
1461 var 1488 videojs.getComponent('Flash').registerSourceHandler(HlsSourceHandler('flash'));
1462 oldBase = document.querySelector('base'), 1489 }
1463 docHead = document.querySelector('head'),
1464 a = document.createElement('a'),
1465 base = oldBase,
1466 oldHref,
1467 result;
1468
1469 // prep the document
1470 if (oldBase) {
1471 oldHref = oldBase.href;
1472 } else {
1473 base = docHead.appendChild(document.createElement('base'));
1474 }
1475 1490
1476 base.href = basePath; 1491 videojs.HlsHandler = HlsHandler;
1477 a.href = path; 1492 videojs.HlsSourceHandler = HlsSourceHandler;
1478 result = a.href; 1493 videojs.Hls = Hls;
1494 videojs.m3u8 = m3u8;
1479 1495
1480 // clean up 1496 export default {
1481 if (oldBase) { 1497 Hls,
1482 oldBase.href = oldHref; 1498 HlsHandler,
1483 } else { 1499 HlsSourceHandler
1484 docHead.removeChild(base);
1485 }
1486 return result;
1487 }; 1500 };
1488
1489 })(window, window.videojs, document);
......
...@@ -90,7 +90,6 @@ function() { ...@@ -90,7 +90,6 @@ function() {
90 90
91 }); 91 });
92 92
93
94 QUnit.module('Incremental Processing', { 93 QUnit.module('Incremental Processing', {
95 beforeEach() { 94 beforeEach() {
96 this.clock = sinon.useFakeTimers(); 95 this.clock = sinon.useFakeTimers();
......
...@@ -13,13 +13,6 @@ ...@@ -13,13 +13,6 @@
13 <script src="/node_modules/sinon/pkg/sinon.js"></script> 13 <script src="/node_modules/sinon/pkg/sinon.js"></script>
14 <script src="/node_modules/qunitjs/qunit/qunit.js"></script> 14 <script src="/node_modules/qunitjs/qunit/qunit.js"></script>
15 <script src="/node_modules/video.js/dist/video.js"></script> 15 <script src="/node_modules/video.js/dist/video.js"></script>
16 <script src="/node_modules/videojs-contrib-media-sources/src/videojs-media-sources.js"></script>
17
18 <script src="/src/videojs-contrib-hls.js"></script>
19 <script src="/dist/videojs-contrib-hls.js"></script>
20 <script src="/src/bin-utils.js"></script>
21
22 <script src="/test/videojs-contrib-hls.test.js"></script>
23 <script src="/dist-test/videojs-contrib-hls.js"></script> 16 <script src="/dist-test/videojs-contrib-hls.js"></script>
24 17
25 </body> 18 </body>
......
...@@ -11,29 +11,11 @@ var DEFAULTS = { ...@@ -11,29 +11,11 @@ var DEFAULTS = {
11 'node_modules/video.js/dist/video.js', 11 'node_modules/video.js/dist/video.js',
12 'node_modules/video.js/dist/video-js.css', 12 'node_modules/video.js/dist/video-js.css',
13 13
14 // REMOVE ME WHEN BROWSERIFIED 14 'test/**/*.test.js'
15 'node_modules/videojs-contrib-media-sources/src/videojs-media-sources.js',
16
17 // these two stub old functionality
18 'src/videojs-contrib-hls.js',
19 'dist/videojs-contrib-hls.js',
20
21 'src/bin-utils.js',
22
23 'test/stub.test.js',
24
25 'test/videojs-contrib-hls.test.js',
26 'test/m3u8.test.js',
27 'test/playlist.test.js',
28 'test/playlist-loader.test.js',
29 'test/decrypter.test.js',
30 // END REMOVE ME
31 // 'test/**/*.js'
32 ], 15 ],
33 16
34 exclude: [ 17 exclude: [
35 'test/bundle.js', 18 'test/data/**'
36 // 'test/data/**'
37 ], 19 ],
38 20
39 plugins: [ 21 plugins: [
...@@ -42,7 +24,7 @@ var DEFAULTS = { ...@@ -42,7 +24,7 @@ var DEFAULTS = {
42 ], 24 ],
43 25
44 preprocessors: { 26 preprocessors: {
45 'test/{playlist*,decrypter,stub,m3u8}.test.js': ['browserify'] 27 'test/**/*.test.js': ['browserify']
46 }, 28 },
47 29
48 reporters: ['dots'], 30 reporters: ['dots'],
......
...@@ -29,9 +29,11 @@ QUnit.test('the environment is sane', function(assert) { ...@@ -29,9 +29,11 @@ QUnit.test('the environment is sane', function(assert) {
29 assert.strictEqual(typeof Array.isArray, 'function', 'es5 exists'); 29 assert.strictEqual(typeof Array.isArray, 'function', 'es5 exists');
30 assert.strictEqual(typeof sinon, 'object', 'sinon exists'); 30 assert.strictEqual(typeof sinon, 'object', 'sinon exists');
31 assert.strictEqual(typeof videojs, 'function', 'videojs exists'); 31 assert.strictEqual(typeof videojs, 'function', 'videojs exists');
32 assert.strictEqual(typeof videojs.MediaSource, 'object', 'MediaSource is an object'); 32 assert.strictEqual(typeof videojs.MediaSource, 'function', 'MediaSource is an object');
33 assert.strictEqual(typeof videojs.URL, 'object', 'URL is an object'); 33 assert.strictEqual(typeof videojs.URL, 'object', 'URL is an object');
34 assert.strictEqual(typeof videojs.Hls, 'object', 'Hls is an object'); 34 assert.strictEqual(typeof videojs.Hls, 'object', 'Hls is an object');
35 assert.strictEqual(typeof videojs.HlsSourceHandler,'function', 'HlsSourceHandler is a function'); 35 assert.strictEqual(typeof videojs.HlsSourceHandler,
36 'function',
37 'HlsSourceHandler is a function');
36 assert.strictEqual(typeof videojs.HlsHandler, 'function', 'HlsHandler is a function'); 38 assert.strictEqual(typeof videojs.HlsHandler, 'function', 'HlsHandler is a function');
37 }); 39 });
......
1 import manifests from './test-manifests';
2 import expected from './test-expected';
3 window.manifests = manifests;
4 window.expected = expected;
5
This diff could not be displayed because it is too large.