Vulture/VApp/node_modules/m3u8-parser/test/parser.test.js

1484 lines
45 KiB
JavaScript
Raw Permalink Normal View History

import QUnit from 'qunit';
import testDataExpected from 'data-files!expecteds';
import testDataManifests from 'data-files!manifests';
import {Parser} from '../src';
QUnit.module('m3u8s', function(hooks) {
hooks.beforeEach(function() {
this.parser = new Parser();
QUnit.dump.maxDepth = 8;
});
QUnit.module('general');
QUnit.test('can be constructed', function(assert) {
assert.notStrictEqual(this.parser, 'undefined', 'parser is defined');
});
QUnit.test('can set custom parsers', function(assert) {
const manifest = [
'#EXTM3U',
'#EXT-X-VERSION:3',
'#EXT-X-TARGETDURATION:10',
'#EXT-X-MEDIA-SEQUENCE:0',
'#EXT-X-PROGRAM-DATE-TIME:2017-07-31T20:35:35.053+00:00',
'#VOD-STARTTIMESTAMP:1501533337573',
'#VOD-TOTALDELETEDDURATION:0.0',
'#VOD-FRAMERATE:29.97',
''
].join('\n');
this.parser.addParser({
expression: /^#VOD-STARTTIMESTAMP/,
customType: 'startTimestamp'
});
this.parser.addParser({
expression: /^#VOD-TOTALDELETEDDURATION/,
customType: 'totalDeleteDuration'
});
this.parser.addParser({
expression: /^#VOD-FRAMERATE/,
customType: 'framerate',
dataParser: (line) => (line.split(':')[1])
});
this.parser.push(manifest);
this.parser.end();
assert.strictEqual(
this.parser.manifest.custom.startTimestamp,
'#VOD-STARTTIMESTAMP:1501533337573',
'sets custom timestamp line'
);
assert.strictEqual(
this.parser.manifest.custom.totalDeleteDuration,
'#VOD-TOTALDELETEDDURATION:0.0',
'sets custom delete duration'
);
assert.strictEqual(this.parser.manifest.custom.framerate, '29.97', 'sets framerate');
});
QUnit.test('segment level custom data', function(assert) {
const manifest = [
'#EXTM3U',
'#VOD-TIMING:1511816599485',
'#COMMENT',
'#EXTINF:8.0,',
'ex1.ts',
'#VOD-TIMING',
'#EXTINF:8.0,',
'ex2.ts',
'#VOD-TIMING:1511816615485',
'#EXT-UNKNOWN',
'#EXTINF:8.0,',
'ex3.ts',
'#VOD-TIMING:1511816623485',
'#EXTINF:8.0,',
'ex3.ts',
'#EXT-X-ENDLIST'
].join('\n');
this.parser.addParser({
expression: /^#VOD-TIMING/,
customType: 'vodTiming',
segment: true
});
this.parser.push(manifest);
this.parser.end();
assert.equal(
this.parser.manifest.segments[0].custom.vodTiming,
'#VOD-TIMING:1511816599485',
'parser attached segment level custom data'
);
assert.equal(
this.parser.manifest.segments[1].custom.vodTiming,
'#VOD-TIMING',
'parser got segment level custom data without :'
);
});
QUnit.test('attaches cue-out data to segment', function(assert) {
const manifest = [
'#EXTM3U',
'#EXTINF:5,',
'#COMMENT',
'ex1.ts',
'#EXT-X-CUE-OUT:10',
'#EXTINF:5,',
'ex2.ts',
'#EXT-UKNOWN-TAG',
'#EXTINF:5,',
'ex3.ts',
'#EXT-X-CUE-OUT:',
'#EXTINF:5,',
'ex3.ts',
'#EXT-X-ENDLIST'
].join('\n');
this.parser.push(manifest);
this.parser.end();
assert.equal(this.parser.manifest.segments[1].cueOut, '10', 'parser attached cue out tag');
assert.equal(this.parser.manifest.segments[3].cueOut, '', 'cue out without data');
});
QUnit.test('attaches cue-out-cont data to segment', function(assert) {
const manifest = [
'#EXTM3U',
'#EXTINF:5,',
'#COMMENT',
'ex1.ts',
'#EXT-X-CUE-OUT-CONT:10/60',
'#EXTINF:5,',
'ex2.ts',
'#EXT-UKNOWN-TAG',
'#EXTINF:5,',
'ex3.ts',
'#EXT-X-CUE-OUT-CONT:',
'#EXTINF:5,',
'ex3.ts',
'#EXT-X-ENDLIST'
].join('\n');
this.parser.push(manifest);
this.parser.end();
assert.equal(
this.parser.manifest.segments[1].cueOutCont, '10/60',
'parser attached cue out cont tag'
);
assert.equal(this.parser.manifest.segments[3].cueOutCont, '', 'cue out cont without data');
});
QUnit.test('attaches cue-in data to segment', function(assert) {
const manifest = [
'#EXTM3U',
'#EXTINF:5,',
'#COMMENT',
'ex1.ts',
'#EXT-X-CUE-IN:',
'#EXTINF:5,',
'ex2.ts',
'#EXT-X-CUE-IN:15',
'#EXT-UKNOWN-TAG',
'#EXTINF:5,',
'ex3.ts',
'#EXTINF:5,',
'ex3.ts',
'#EXT-X-ENDLIST'
].join('\n');
this.parser.push(manifest);
this.parser.end();
assert.equal(this.parser.manifest.segments[1].cueIn, '', 'parser attached cue in tag');
assert.equal(this.parser.manifest.segments[2].cueIn, '15', 'cue in with data');
});
QUnit.test('parses characteristics attribute', function(assert) {
const manifest = [
'#EXTM3U',
'#EXT-X-MEDIA:TYPE=SUBTITLES,GROUP-ID="subs",CHARACTERISTICS="char",NAME="test"',
'#EXT-X-STREAM-INF:BANDWIDTH=1,CODECS="mp4a.40.2, avc1.4d400d",SUBTITLES="subs"',
'index.m3u8'
].join('\n');
this.parser.push(manifest);
this.parser.end();
assert.equal(
this.parser.manifest.mediaGroups.SUBTITLES.subs.test.characteristics,
'char',
'parsed CHARACTERISTICS attribute'
);
});
QUnit.test('parses FORCED attribute', function(assert) {
const manifest = [
'#EXTM3U',
'#EXT-X-MEDIA:TYPE=SUBTITLES,GROUP-ID="subs",CHARACTERISTICS="char",NAME="test",FORCED=YES',
'#EXT-X-STREAM-INF:BANDWIDTH=1,CODECS="mp4a.40.2, avc1.4d400d",SUBTITLES="subs"',
'index.m3u8'
].join('\n');
this.parser.push(manifest);
this.parser.end();
assert.ok(
this.parser.manifest.mediaGroups.SUBTITLES.subs.test.forced,
'parsed FORCED attribute'
);
});
QUnit.test('parses Widevine #EXT-X-KEY attributes and attaches to manifest', function(assert) {
const manifest = [
'#EXTM3U',
'#EXT-X-KEY:METHOD=SAMPLE-AES-CTR,' +
'URI="data:text/plain;base64,AAAAPnBzc2gAAAAA7e+LqXnWSs6jyCfc1R0h7QAAAB4iFnN' +
'oYWthX2NlYzJmNjRhYTc4OTBhMTFI49yVmwY=",KEYID=0x800AACAA522958AE888062B5695DB6BF,' +
'KEYFORMATVERSIONS="1",KEYFORMAT="urn:uuid:edef8ba9-79d6-4ace-a3c8-27dcd51d21ed"',
'#EXTINF:5,',
'ex1.ts',
'#EXT-X-ENDLIST'
].join('\n');
this.parser.push(manifest);
this.parser.end();
assert.ok(this.parser.manifest.contentProtection, 'contentProtection property added');
assert.equal(
this.parser.manifest.contentProtection['com.widevine.alpha'].attributes.schemeIdUri,
'urn:uuid:edef8ba9-79d6-4ace-a3c8-27dcd51d21ed',
'schemeIdUri set correctly'
);
assert.equal(
this.parser.manifest.contentProtection['com.widevine.alpha'].attributes.keyId,
'800AACAA522958AE888062B5695DB6BF',
'keyId set correctly'
);
assert.equal(
this.parser.manifest.contentProtection['com.widevine.alpha'].pssh.byteLength,
62,
'base64 URI decoded to TypedArray'
);
});
QUnit.test('Widevine #EXT-X-KEY attributes not attached to manifest if METHOD is invalid', function(assert) {
const manifest = [
'#EXTM3U',
'#EXT-X-KEY:METHOD=NONE,' +
'URI="data:text/plain;base64,AAAAPnBzc2gAAAAA7e+LqXnWSs6jyCfc1R0h7QAAAB4iFnN' +
'oYWthX2NlYzJmNjRhYTc4OTBhMTFI49yVmwY=",KEYID=0x800AACAA522958AE888062B5695DB6BF,' +
'KEYFORMATVERSIONS="1",KEYFORMAT="urn:uuid:edef8ba9-79d6-4ace-a3c8-27dcd51d21ed"',
'#EXTINF:5,',
'ex1.ts',
'#EXT-X-ENDLIST'
].join('\n');
this.parser.push(manifest);
this.parser.end();
assert.notOk(this.parser.manifest.contentProtection, 'contentProtection not added');
});
QUnit.test('Widevine #EXT-X-KEY attributes not attached to manifest if URI is invalid', function(assert) {
const manifest = [
'#EXTM3U',
'#EXT-X-KEY:METHOD=SAMPLE-AES-CTR,' +
'URI="AAAAPnBzc2gAAAAA7e+LqXnWSs6jyCfc1R0h7QAAAB4iFnN' +
'oYWthX2NlYzJmNjRhYTc4OTBhMTFI49yVmwY=",KEYID=0x800AACAA522958AE888062B5695DB6BF,' +
'KEYFORMATVERSIONS="1",KEYFORMAT="urn:uuid:edef8ba9-79d6-4ace-a3c8-27dcd51d21ed"',
'#EXTINF:5,',
'ex1.ts',
'#EXT-X-ENDLIST'
].join('\n');
this.parser.push(manifest);
this.parser.end();
assert.notOk(this.parser.manifest.contentProtection, 'contentProtection not added');
});
QUnit.test('Widevine #EXT-X-KEY attributes not attached to manifest if KEYID is invalid', function(assert) {
const manifest = [
'#EXTM3U',
'#EXT-X-KEY:METHOD=SAMPLE-AES-CTR,' +
'URI="data:text/plain;base64,AAAAPnBzc2gAAAAA7e+LqXnWSs6jyCfc1R0h7QAAAB4iFnN' +
'oYWthX2NlYzJmNjRhYTc4OTBhMTFI49yVmwY=",KEYID=800AACAA522958AE888062B5695DB6BF,' +
'KEYFORMATVERSIONS="1",KEYFORMAT="urn:uuid:edef8ba9-79d6-4ace-a3c8-27dcd51d21ed"',
'#EXTINF:5,',
'ex1.ts',
'#EXT-X-ENDLIST'
].join('\n');
this.parser.push(manifest);
this.parser.end();
assert.notOk(this.parser.manifest.contentProtection, 'contentProtection not added');
});
QUnit.test('Widevine #EXT-X-KEY attributes not attached to manifest if KEYFORMAT is not Widevine UUID', function(assert) {
const manifest = [
'#EXTM3U',
'#EXT-X-KEY:METHOD=SAMPLE-AES-CTR,' +
'URI="data:text/plain;base64,AAAAPnBzc2gAAAAA7e+LqXnWSs6jyCfc1R0h7QAAAB4iFnN' +
'oYWthX2NlYzJmNjRhYTc4OTBhMTFI49yVmwY=",KEYID=0x800AACAA522958AE888062B5695DB6BF,' +
'KEYFORMATVERSIONS="1",KEYFORMAT="invalid-keyformat"',
'#EXTINF:5,',
'ex1.ts',
'#EXT-X-ENDLIST'
].join('\n');
this.parser.push(manifest);
this.parser.end();
assert.notOk(this.parser.manifest.contentProtection, 'contentProtection not added');
});
QUnit.test('byterange offset defaults to next byte', function(assert) {
const manifest = [
'#EXTM3U',
'#EXTINF:5,',
'#EXT-X-BYTERANGE:10@5',
'segment.ts',
'#EXTINF:5,',
'#EXT-X-BYTERANGE:20',
'segment.ts',
'#EXTINF:5,',
'#EXT-X-BYTERANGE:30',
'segment.ts',
'#EXTINF:5,',
'segment2.ts',
'#EXT-X-BYTERANGE:15@100',
'segment.ts',
'#EXT-X-BYTERANGE:17',
'segment.ts',
'#EXT-X-ENDLIST'
].join('\n');
this.parser.push(manifest);
this.parser.end();
assert.deepEqual(
this.parser.manifest.segments[0].byterange,
{ length: 10, offset: 5 },
'first segment has correct byterange'
);
assert.deepEqual(
this.parser.manifest.segments[1].byterange,
{ length: 20, offset: 15 },
'second segment has correct byterange'
);
assert.deepEqual(
this.parser.manifest.segments[2].byterange,
{ length: 30, offset: 35 },
'third segment has correct byterange'
);
assert.notOk(this.parser.manifest.segments[3].byterange, 'fourth segment has no byterange');
assert.deepEqual(
this.parser.manifest.segments[4].byterange,
{ length: 15, offset: 100 },
'fifth segment has correct byterange'
);
// not tested is a segment with no offset coming after a segment that isn't a sub range,
// as the spec requires that a byterange without an offset must follow a segment that
// is a sub range of the same media resource
assert.deepEqual(
this.parser.manifest.segments[5].byterange,
{ length: 17, offset: 115 },
'sixth segment has correct byterange'
);
});
QUnit.module('warn/info', {
beforeEach() {
this.warnings = [];
this.infos = [];
this.parser.on('warn', (warn) => this.warnings.push(warn.message));
this.parser.on('info', (info) => this.infos.push(info.message));
}
});
QUnit.test('warn when #EXT-X-TARGETDURATION is invalid', function(assert) {
this.parser.push([
'#EXT-X-VERSION:3',
'#EXT-X-MEDIA-SEQUENCE:0',
'#EXT-X-DISCONTINUITY-SEQUENCE:0',
'#EXT-X-TARGETDURATION:foo',
'#EXTINF:10,',
'media-00001.ts',
'#EXT-X-ENDLIST'
].join('\n'));
this.parser.end();
const warnings = [
'ignoring invalid target duration: undefined'
];
assert.deepEqual(
this.warnings,
warnings,
'warnings as expected'
);
assert.deepEqual(
this.infos,
[],
'info as expected'
);
});
QUnit.test('warns when #EXT-X-START missing TIME-OFFSET attribute', function(assert) {
this.parser.push([
'#EXT-X-VERSION:3',
'#EXT-X-MEDIA-SEQUENCE:0',
'#EXT-X-DISCONTINUITY-SEQUENCE:0',
'#EXT-X-TARGETDURATION:10',
'#EXT-X-START:PRECISE=YES',
'#EXTINF:10,',
'media-00001.ts',
'#EXT-X-ENDLIST'
].join('\n'));
this.parser.end();
assert.deepEqual(
this.warnings,
['ignoring start declaration without appropriate attribute list'],
'warnings as expected'
);
assert.deepEqual(
this.infos,
[],
'info as expected'
);
assert.strictEqual(typeof this.parser.manifest.start, 'undefined', 'does not parse start');
});
QUnit.test('warning when #EXT-X-SKIP missing SKIPPED-SEGMENTS attribute', function(assert) {
this.parser.push([
'#EXT-X-VERSION:3',
'#EXT-X-MEDIA-SEQUENCE:0',
'#EXT-X-DISCONTINUITY-SEQUENCE:0',
'#EXT-X-TARGETDURATION:10',
'#EXT-X-SKIP:foo=bar',
'#EXTINF:10,',
'media-00001.ts',
'#EXT-X-ENDLIST'
].join('\n'));
this.parser.end();
assert.deepEqual(
this.warnings,
['#EXT-X-SKIP lacks required attribute(s): SKIPPED-SEGMENTS'],
'warnings as expected'
);
assert.deepEqual(
this.infos,
[],
'info as expected'
);
});
QUnit.test('warns when #EXT-X-PART missing URI/DURATION attributes', function(assert) {
this.parser.push([
'#EXT-X-VERSION:3',
'#EXT-X-MEDIA-SEQUENCE:0',
'#EXT-X-DISCONTINUITY-SEQUENCE:0',
'#EXT-X-TARGETDURATION:10',
'#EXT-X-PART:DURATION=1',
'#EXT-X-PART:URI=2',
'#EXT-X-PART:foo=bar',
'#EXTINF:10,',
'media-00001.ts',
'#EXT-X-ENDLIST'
].join('\n'));
this.parser.end();
const warnings = [
'#EXT-X-PART #0 for segment #0 lacks required attribute(s): URI',
'#EXT-X-PART #1 for segment #0 lacks required attribute(s): DURATION',
'#EXT-X-PART #2 for segment #0 lacks required attribute(s): URI, DURATION'
];
assert.deepEqual(
this.warnings,
warnings,
'warnings as expected'
);
assert.deepEqual(
this.infos,
[],
'info as expected'
);
});
QUnit.test('warns when #EXT-X-PRELOAD-HINT missing TYPE/URI attribute', function(assert) {
this.parser.push([
'#EXT-X-VERSION:3',
'#EXT-X-MEDIA-SEQUENCE:0',
'#EXT-X-DISCONTINUITY-SEQUENCE:0',
'#EXT-X-TARGETDURATION:10',
'#EXT-X-PRELOAD-HINT:TYPE=foo',
'#EXT-X-PRELOAD-HINT:URI=foo',
'#EXT-X-PRELOAD-HINT:foo=bar',
'#EXTINF:10,',
'media-00001.ts',
'#EXT-X-ENDLIST'
].join('\n'));
this.parser.end();
const warnings = [
'#EXT-X-PRELOAD-HINT #0 for segment #0 lacks required attribute(s): URI',
'#EXT-X-PRELOAD-HINT #1 for segment #0 lacks required attribute(s): TYPE',
'#EXT-X-PRELOAD-HINT #2 for segment #0 lacks required attribute(s): TYPE, URI'
];
assert.deepEqual(
this.warnings,
warnings,
'warnings as expected'
);
assert.deepEqual(
this.infos,
[],
'info as expected'
);
});
QUnit.test('warns when we get #EXT-X-PRELOAD-HINT with the same TYPE', function(assert) {
this.parser.push([
'#EXT-X-VERSION:3',
'#EXT-X-MEDIA-SEQUENCE:0',
'#EXT-X-DISCONTINUITY-SEQUENCE:0',
'#EXT-X-TARGETDURATION:10',
'#EXT-X-PRELOAD-HINT:TYPE=foo,URI=foo1',
'#EXT-X-PRELOAD-HINT:TYPE=foo,URI=foo2',
'#EXTINF:10,',
'media-00001.ts',
'#EXT-X-ENDLIST'
].join('\n'));
this.parser.end();
const warnings = [
'#EXT-X-PRELOAD-HINT #1 for segment #0 has the same TYPE foo as preload hint #0'
];
assert.deepEqual(
this.warnings,
warnings,
'warnings as expected'
);
assert.deepEqual(
this.infos,
[],
'info as expected'
);
});
QUnit.test('warn when #EXT-X-RENDITION-REPORT missing LAST-MSN/URI attribute', function(assert) {
this.parser.push([
'#EXT-X-VERSION:3',
'#EXT-X-MEDIA-SEQUENCE:0',
'#EXT-X-DISCONTINUITY-SEQUENCE:0',
'#EXT-X-TARGETDURATION:10',
'#EXT-X-RENDITION-REPORT:URI=foo',
'#EXT-X-RENDITION-REPORT:LAST-MSN=2',
'#EXT-X-RENDITION-REPORT:foo=bar',
'#EXTINF:10,',
'media-00001.ts',
'#EXT-X-ENDLIST'
].join('\n'));
this.parser.end();
const warnings = [
'#EXT-X-RENDITION-REPORT #0 lacks required attribute(s): LAST-MSN',
'#EXT-X-RENDITION-REPORT #1 lacks required attribute(s): URI',
'#EXT-X-RENDITION-REPORT #2 lacks required attribute(s): LAST-MSN, URI'
];
assert.deepEqual(
this.warnings,
warnings,
'warnings as expected'
);
assert.deepEqual(
this.infos,
[],
'info as expected'
);
});
QUnit.test('warns when #EXT-X-RENDITION-REPORT missing LAST-PART attribute with parts', function(assert) {
this.parser.push([
'#EXT-X-VERSION:3',
'#EXT-X-MEDIA-SEQUENCE:0',
'#EXT-X-DISCONTINUITY-SEQUENCE:0',
'#EXT-X-TARGETDURATION:10',
'#EXT-X-RENDITION-REPORT:URI=foo,LAST-MSN=4',
'#EXT-X-PART:URI=foo,DURATION=10',
'#EXT-X-RENDITION-REPORT:URI=foo,LAST-MSN=4',
'#EXTINF:10,',
'media-00001.ts',
'#EXT-X-ENDLIST'
].join('\n'));
this.parser.end();
const warnings = [
'#EXT-X-RENDITION-REPORT #0 lacks required attribute(s): LAST-PART',
'#EXT-X-RENDITION-REPORT #1 lacks required attribute(s): LAST-PART'
];
assert.deepEqual(
this.warnings,
warnings,
'warnings as expected'
);
assert.deepEqual(
this.infos,
[],
'info as expected'
);
});
QUnit.test('warns when #EXT-X-PART-INF missing PART-TARGET attribute', function(assert) {
this.parser.push([
'#EXT-X-VERSION:3',
'#EXT-X-MEDIA-SEQUENCE:0',
'#EXT-X-DISCONTINUITY-SEQUENCE:0',
'#EXT-X-TARGETDURATION:10',
'#EXT-X-PART-INF:URI=foo',
'#EXTINF:10,',
'media-00001.ts',
'#EXT-X-ENDLIST'
].join('\n'));
this.parser.end();
const warnings = [
'#EXT-X-PART-INF lacks required attribute(s): PART-TARGET'
];
assert.deepEqual(
this.warnings,
warnings,
'warnings as expected'
);
assert.deepEqual(
this.infos,
[],
'info as expected'
);
});
QUnit.test('warns when #EXT-X-SERVER-CONTROL missing CAN-SKIP-UNTIL with CAN-SKIP-DATERANGES attribute', function(assert) {
this.parser.push([
'#EXT-X-VERSION:3',
'#EXT-X-MEDIA-SEQUENCE:0',
'#EXT-X-DISCONTINUITY-SEQUENCE:0',
'#EXT-X-TARGETDURATION:10',
'#EXT-X-SERVER-CONTROL:CAN-BLOCK-RELOAD=NO,HOLD-BACK=30,CAN-SKIP-DATERANGES=YES',
'#EXTINF:10,',
'media-00001.ts',
'#EXT-X-ENDLIST'
].join('\n'));
this.parser.end();
const warnings = [
'#EXT-X-SERVER-CONTROL lacks required attribute CAN-SKIP-UNTIL which is required when CAN-SKIP-DATERANGES is set'
];
assert.deepEqual(
this.warnings,
warnings,
'warnings as expected'
);
assert.deepEqual(
this.infos,
[],
'info as expected'
);
});
QUnit.test('warn when #EXT-X-SERVER-CONTROL HOLD-BACK and PART-HOLD-BACK too low', function(assert) {
this.parser.push([
'#EXT-X-VERSION:3',
'#EXT-X-MEDIA-SEQUENCE:0',
'#EXT-X-DISCONTINUITY-SEQUENCE:0',
'#EXT-X-TARGETDURATION:10',
'#EXT-X-PART-INF:PART-TARGET=1',
'#EXT-X-SERVER-CONTROL:CAN-BLOCK-RELOAD=YES,HOLD-BACK=1,PART-HOLD-BACK=0.5',
'#EXTINF:10,',
'media-00001.ts',
'#EXT-X-ENDLIST'
].join('\n'));
this.parser.end();
const warnings = [
'#EXT-X-SERVER-CONTROL clamping HOLD-BACK (1) to targetDuration * 3 (30)',
'#EXT-X-SERVER-CONTROL clamping PART-HOLD-BACK (0.5) to partTargetDuration * 2 (2).'
];
assert.deepEqual(
this.warnings,
warnings,
'warnings as expected'
);
assert.deepEqual(
this.infos,
[],
'info as expected'
);
});
QUnit.test('warn when #EXT-X-SERVER-CONTROL before target durations HOLD-BACK/PART-HOLD-BACK too low', function(assert) {
this.parser.push([
'#EXT-X-VERSION:3',
'#EXT-X-MEDIA-SEQUENCE:0',
'#EXT-X-SERVER-CONTROL:CAN-BLOCK-RELOAD=YES,HOLD-BACK=1,PART-HOLD-BACK=0.5',
'#EXT-X-DISCONTINUITY-SEQUENCE:0',
'#EXT-X-TARGETDURATION:10',
'#EXT-X-PART-INF:PART-TARGET=1',
'#EXTINF:10,',
'media-00001.ts',
'#EXT-X-ENDLIST'
].join('\n'));
this.parser.end();
const warnings = [
'#EXT-X-SERVER-CONTROL clamping HOLD-BACK (1) to targetDuration * 3 (30)',
'#EXT-X-SERVER-CONTROL clamping PART-HOLD-BACK (0.5) to partTargetDuration * 2 (2).'
];
assert.deepEqual(
this.warnings,
warnings,
'warnings as expected'
);
assert.deepEqual(
this.infos,
[],
'info as expected'
);
});
QUnit.test('info when #EXT-X-SERVER-CONTROL sets defaults', function(assert) {
this.parser.push([
'#EXT-X-VERSION:3',
'#EXT-X-MEDIA-SEQUENCE:0',
'#EXT-X-DISCONTINUITY-SEQUENCE:0',
'#EXT-X-TARGETDURATION:10',
'#EXT-X-PART-INF:PART-TARGET=1',
'#EXT-X-SERVER-CONTROL:foo=bar',
'#EXTINF:10,',
'media-00001.ts',
'#EXT-X-ENDLIST'
].join('\n'));
this.parser.end();
const infos = [
'#EXT-X-SERVER-CONTROL defaulting CAN-BLOCK-RELOAD to false',
'#EXT-X-SERVER-CONTROL defaulting HOLD-BACK to targetDuration * 3 (30).',
'#EXT-X-SERVER-CONTROL defaulting PART-HOLD-BACK to partTargetDuration * 3 (3).'
];
assert.deepEqual(
this.warnings,
[],
'warnings as expected'
);
assert.deepEqual(
this.infos,
infos,
'info as expected'
);
});
QUnit.test('info when #EXT-X-SERVER-CONTROL before target durations and sets defaults', function(assert) {
this.parser.push([
'#EXT-X-VERSION:3',
'#EXT-X-MEDIA-SEQUENCE:0',
'#EXT-X-DISCONTINUITY-SEQUENCE:0',
'#EXT-X-SERVER-CONTROL:foo=bar',
'#EXT-X-TARGETDURATION:10',
'#EXT-X-PART-INF:PART-TARGET=1',
'#EXTINF:10,',
'media-00001.ts',
'#EXT-X-ENDLIST'
].join('\n'));
this.parser.end();
const infos = [
'#EXT-X-SERVER-CONTROL defaulting CAN-BLOCK-RELOAD to false',
'#EXT-X-SERVER-CONTROL defaulting HOLD-BACK to targetDuration * 3 (30).',
'#EXT-X-SERVER-CONTROL defaulting PART-HOLD-BACK to partTargetDuration * 3 (3).'
];
assert.deepEqual(
this.warnings,
[],
'warnings as expected'
);
assert.deepEqual(
this.infos,
infos,
'info as expected'
);
});
QUnit.test('Can understand widevine/fairplay/playready drm ext-x-key', function(assert) {
this.parser.push([
'#EXT-X-VERSION:3',
'#EXT-X-MEDIA-SEQUENCE:0',
'#EXT-X-DISCONTINUITY-SEQUENCE:0',
'#EXT-X-TARGETDURATION:10',
'#EXT-X-PART-INF:PART-TARGET=1',
'#EXT-X-SERVER-CONTROL:foo=bar',
'#EXT-X-KEY:METHOD=SAMPLE-AES,URI="data:text/plain;base64,foo",KEYID=0x555777,IV=1234567890abcdef1234567890abcdef,KEYFORMATVERSIONS="1",KEYFORMAT="urn:uuid:edef8ba9-79d6-4ace-a3c8-27dcd51d21ed"',
'#EXT-X-KEY:METHOD=SAMPLE-AES,URI="skd://foo",KEYFORMATVERSIONS="1",KEYFORMAT="com.apple.streamingkeydelivery"',
'#EXT-X-KEY:METHOD=SAMPLE-AES,URI="http://example.com",KEYFORMATVERSIONS="1",KEYFORMAT="com.microsoft.playready"',
'#EXTINF:10,',
'media-00001.ts',
'#EXT-X-ENDLIST'
].join('\n'));
this.parser.end();
assert.deepEqual(
Object.keys(this.parser.manifest.contentProtection),
['com.widevine.alpha', 'com.apple.fps.1_0', 'com.microsoft.playready'],
'info as expected'
);
});
QUnit.test('PDT value is assigned to segments with explicit #EXT-X-PROGRAM-DATE-TIME tags', function(assert) {
this.parser.push([
'#EXTM3U',
'#EXT-X-VERSION:6',
'#EXT-X-TARGETDURATION:8',
'#EXT-X-MEDIA-SEQUENCE:0',
'#EXTINF:8.0',
'#EXT-X-PROGRAM-DATE-TIME:2017-07-31T20:35:35.053+00:00',
'https://example.com/playlist1.m3u8',
'#EXTINF:8.0,',
'#EXT-X-PROGRAM-DATE-TIME:2017-07-31T22:14:10.053+00:00',
'https://example.com/playlist2.m3u8',
'#EXT-X-ENDLIST'
].join('\n'));
this.parser.end();
assert.equal(this.parser.manifest.segments[0].programDateTime, new Date('2017-07-31T20:35:35.053+00:00').getTime());
assert.equal(this.parser.manifest.segments[1].programDateTime, new Date('2017-07-31T22:14:10.053+00:00').getTime());
});
QUnit.test('backfill PDT values when the first EXT-X-PROGRAM-DATE-TIME tag appears after one or more Media Segment URIs', function(assert) {
this.parser.push([
'#EXTM3U',
'#EXT-X-VERSION:6',
'#EXT-X-TARGETDURATION:8',
'#EXT-X-MEDIA-SEQUENCE:0',
'#EXTINF:8.0',
'https://example.com/playlist1.m3u8',
'#EXTINF:8.0,',
'https://example.com/playlist2.m3u8',
'#EXTINF:8.0',
'#EXT-X-PROGRAM-DATE-TIME:2017-07-31T20:35:35.053+00:00',
'https://example.com/playlist3.m3u8',
'#EXT-X-ENDLIST'
].join('\n'));
this.parser.end();
const segments = this.parser.manifest.segments;
assert.equal(segments[2].programDateTime, new Date('2017-07-31T20:35:35.053+00:00').getTime());
assert.equal(segments[1].programDateTime, segments[2].programDateTime - (segments[1].duration * 1000));
assert.equal(segments[0].programDateTime, segments[1].programDateTime - (segments[0].duration * 1000));
});
QUnit.test('extrapolates forward when subsequent fragments do not have explicit PDT tags', function(assert) {
this.parser.push([
'#EXTM3U',
'#EXT-X-VERSION:6',
'#EXT-X-TARGETDURATION:8',
'#EXT-X-MEDIA-SEQUENCE:0',
'#EXTINF:8.0',
'#EXT-X-PROGRAM-DATE-TIME:2017-07-31T20:35:35.053+00:00',
'https://example.com/playlist1.m3u8',
'#EXTINF:8.0,',
'https://example.com/playlist2.m3u8',
'#EXTINF:8.0',
'https://example.com/playlist3.m3u8',
'#EXT-X-ENDLIST'
].join('\n'));
this.parser.end();
const segments = this.parser.manifest.segments;
assert.equal(segments[0].programDateTime, new Date('2017-07-31T20:35:35.053+00:00').getTime());
assert.equal(segments[1].programDateTime, segments[0].programDateTime + segments[1].duration * 1000);
assert.equal(segments[2].programDateTime, segments[1].programDateTime + segments[2].duration * 1000);
});
QUnit.test('warns when #EXT-X-DATERANGE missing attribute', function(assert) {
this.parser.push([
'#EXT-X-VERSION:3',
'#EXT-X-MEDIA-SEQUENCE:0',
'#EXT-X-DISCONTINUITY-SEQUENCE:0',
'#EXTINF:10,',
'media-00001.ts',
'#EXT-X-ENDLIST',
'#EXT-X-PROGRAM-DATE-TIME:2017-07-31T20:35:35.053+00:00',
'#EXT-X-DATERANGE:ID="12345"'
].join('\n'));
this.parser.end();
const warnings = [
'#EXT-X-DATERANGE #0 lacks required attribute(s): START-DATE'
];
assert.deepEqual(
this.warnings,
warnings,
'warnings as expected'
);
});
QUnit.test('warns when #EXT-X-DATERANGE end date attribute is less than start date', function(assert) {
this.parser.push([
'#EXT-X-VERSION:3',
'#EXT-X-MEDIA-SEQUENCE:0',
'#EXT-X-DISCONTINUITY-SEQUENCE:0',
'#EXTINF:10,',
'media-00001.ts',
'#EXT-X-ENDLIST',
'#EXT-X-PROGRAM-DATE-TIME:2017-07-31T20:35:35.053+00:00',
'#EXT-X-DATERANGE:ID="12345",START-DATE="2023-04-13T18:16:15.840000Z",END-DATE="2023-04-13T15:15:15.840000Z"'
].join('\n'));
this.parser.end();
const warnings = [
'EXT-X-DATERANGE END-DATE must be equal to or later than the value of the START-DATE'
];
assert.deepEqual(
this.warnings,
warnings,
'warnings as expected'
);
});
QUnit.test('warns when #EXT-X-DATERANGE duration or planned duration attribute is negative', function(assert) {
this.parser.push([
'#EXT-X-VERSION:3',
'#EXT-X-MEDIA-SEQUENCE:0',
'#EXT-X-DISCONTINUITY-SEQUENCE:0',
'#EXTINF:10,',
'media-00001.ts',
'#EXT-X-ENDLIST',
'#EXT-X-PROGRAM-DATE-TIME:2017-07-31T20:35:35.053+00:00',
'#EXT-X-DATERANGE:ID="12345",START-DATE="2023-04-13T18:16:15.840000Z",PLANNED-DURATION=-38.4,DURATION=-15.5'
].join('\n'));
this.parser.end();
const warnings = [
'EXT-X-DATERANGE DURATION must not be negative',
'EXT-X-DATERANGE PLANNED-DURATION must not be negative'
];
assert.deepEqual(
this.warnings,
warnings,
'warnings as expected'
);
});
QUnit.test('warns when #EXT-X-DATERANGE has a END-ON-NEXT=YES attribute and a DURATION or END-DATE attribute', function(assert) {
this.parser.push([
'#EXT-X-VERSION:3',
'#EXT-X-MEDIA-SEQUENCE:0',
'#EXT-X-DISCONTINUITY-SEQUENCE:0',
'#EXTINF:10,',
'media-00001.ts',
'#EXT-X-ENDLIST',
'#EXT-X-PROGRAM-DATE-TIME:2017-07-31T20:35:35.053+00:00',
'#EXT-X-DATERANGE:ID="12345",START-DATE="2023-04-13T15:15:15.840000Z",END-ON-NEXT=YES, END-DATE="2023-04-13T18:16:15.840000Z",CLASS="CLASSATTRIBUTE"'
].join('\n'));
this.parser.end();
const warnings = [
'EXT-X-DATERANGE with an END-ON-NEXT=YES attribute must not contain DURATION or END-DATE attributes'
];
assert.deepEqual(
this.warnings,
warnings,
'warnings as expected'
);
});
QUnit.test('warns when #EXT-X-DATERANGE has a END-ON-NEXT=YES attribute but not a CLASS attribute', function(assert) {
this.parser.push([
'#EXT-X-VERSION:3',
'#EXT-X-MEDIA-SEQUENCE:0',
'#EXT-X-DISCONTINUITY-SEQUENCE:0',
'#EXTINF:10,',
'media-00001.ts',
'#EXT-X-ENDLIST',
'#EXT-X-PROGRAM-DATE-TIME:2017-07-31T20:35:35.053+00:00',
'#EXT-X-DATERANGE:ID="12345",START-DATE="2023-04-13T18:16:15.840000Z",END-ON-NEXT=YES'
].join('\n'));
this.parser.end();
const warnings = [
'EXT-X-DATERANGE with an END-ON-NEXT=YES attribute must have a CLASS attribute'
];
assert.deepEqual(
this.warnings,
warnings,
'warnings as expected'
);
});
QUnit.test('warns when playlist has multiple #EXT-X-DATERANGE tag same ID but different attribute values', function(assert) {
this.parser.push([
'#EXT-X-VERSION:3',
'#EXT-X-MEDIA-SEQUENCE:0',
'#EXT-X-DISCONTINUITY-SEQUENCE:0',
'#EXTINF:10,',
'media-00001.ts',
'#EXT-X-ENDLIST',
'#EXT-X-PROGRAM-DATE-TIME:2017-07-31T20:35:35.053+00:00',
'#EXT-X-DATERANGE:ID="12345",START-DATE="2023-04-13T18:16:15.840000Z",END-ON-NEXT=YES,CLASS="CLASSATTRIBUTE"',
'#EXT-X-DATERANGE:ID="12345",START-DATE="2023-04-13T18:16:15.840000Z",CLASS="CLASSATTRIBUTE1"'
].join('\n'));
this.parser.end();
const warnings = [
'EXT-X-DATERANGE tags with the same ID in a playlist must have the same attributes values'
];
assert.deepEqual(
this.warnings,
warnings,
'warnings as expected'
);
});
QUnit.test('when #EXT-X-DATERANGE has both DURATION and END-DATE attributes, value of the END-DATE attribute must be START-DATE + DURATION', function(assert) {
this.parser.push([
'#EXT-X-VERSION:3',
'#EXT-X-MEDIA-SEQUENCE:0',
'#EXT-X-DISCONTINUITY-SEQUENCE:0',
'#EXTINF:10,',
'media-00001.ts',
'#EXT-X-ENDLIST',
'#EXT-X-PROGRAM-DATE-TIME:2017-07-31T20:35:35.053+00:00',
'#EXT-X-DATERANGE:ID="12345",START-DATE="2023-04-13T15:16:15.840000Z",DURATION=14.0,END-DATE="2023-04-13T18:15:15.840000Z"'
].join('\n'));
this.parser.end();
assert.deepEqual(this.parser.manifest.dateRanges[0].endDate, new Date('2023-04-13T15:16:29.840000Z'));
});
QUnit.test('warns when playlist contains #EXT-X-DATERANGE tag but no #EXT-X-PROGRAM-DATE-TIME', function(assert) {
this.parser.push([
'#EXT-X-VERSION:3',
'#EXT-X-MEDIA-SEQUENCE:0',
'#EXT-X-DISCONTINUITY-SEQUENCE:0',
'#EXTINF:10,',
'media-00001.ts',
'#EXT-X-ENDLIST',
'#EXT-X-DATERANGE:ID="12345",START-DATE="2023-04-13T18:16:15.840000Z",END-ON-NEXT=YES,CLASS="sampleClassAttrib"'
].join('\n'));
this.parser.end();
const warnings = [
'A playlist with EXT-X-DATERANGE tag must contain atleast one EXT-X-PROGRAM-DATE-TIME tag'
];
assert.deepEqual(
this.warnings,
warnings,
'warnings as expected'
);
});
QUnit.test('playlist with multiple ext-x-daterange with same ID but no conflicting attributes', function(assert) {
const expectedDateRange = {
id: '12345',
scte35In: '0xFC30200FFF2',
scte35Out: '0xFC30200FFF2',
startDate: new Date('2023-04-13T18:16:15.840000Z'),
class: 'CLASSATTRIBUTE'
};
this.parser.push([
'#EXT-X-VERSION:3',
'#EXT-X-MEDIA-SEQUENCE:0',
'#EXT-X-DISCONTINUITY-SEQUENCE:0',
'#EXTINF:10,',
'media-00001.ts',
'#EXT-X-ENDLIST',
'#EXT-X-PROGRAM-DATE-TIME:2017-07-31T20:35:35.053+00:00',
'#EXT-X-DATERANGE:ID="12345",SCTE35-IN=0xFC30200FFF2,START-DATE="2023-04-13T18:16:15.840000Z",CLASS="CLASSATTRIBUTE"',
'#EXT-X-DATERANGE:ID="12345",SCTE35-OUT=0xFC30200FFF2,START-DATE="2023-04-13T18:16:15.840000Z"'
].join('\n'));
this.parser.end();
assert.equal(this.parser.manifest.dateRanges.length, 1, 'two dateranges with same ID are merged');
assert.deepEqual(this.parser.manifest.dateRanges[0], expectedDateRange);
});
QUnit.test('playlist with multiple ext-x-daterange ', function(assert) {
this.parser.push([
' #EXTM3U',
'#EXT-X-VERSION:6',
'#EXT-X-TARGETDURATION:8',
'#EXT-X-MEDIA-SEQUENCE:0',
'#EXT-X-PROGRAM-DATE-TIME:2017-07-31T20:35:35.053+00:00',
'#EXT-X-DATERANGE:ID="event1",START-DATE="2023-04-20T10:00:00Z",DURATION=30.0,END-DATE="2023-04-20T10:00:30Z",X-CUSTOM-KEY="value"',
'#EXTINF:8.0',
'https://example.com/playlist1.m3u8',
'#EXT-SCTE35-IN:0xFC002F0000000000FF000014056FFFFFFF065870697070657220506F6F7200',
'#EXT-X-DATERANGE:ID="event2",START-DATE="2023-04-20T11:00:00Z",DURATION=60.0,END-DATE="2023-04-20T11:01:00Z",X-CUSTOM-KEY="value"',
'#EXTINF:8.0,',
'https://example.com/playlist2.m3u8',
'#EXT-SCTE35-OUT:0xFC002F0000000000FF000014056FFFFFFF065870697070657220506F6F7200',
'#EXT-X-DATERANGE:ID="event3",START-DATE="2023-04-20T12:00:00Z",DURATION=120.0,END-DATE="2023-04-20T12:02:00Z",X-CUSTOM-KEY="value"',
'#EXTINF:8.0',
'https://example.com/playlist3.m3u8',
'#EXT-SCTE35-IN:0xFC002F0000000000FF000014056FFFFFFF065870697070657220506F6F7200',
'#EXT-SCTE35-OUT:0xFC002F0000000000FF000014056FFFFFFF065870697070657220506F6F7200',
'#EXT-X-ENDLIST'
].join('\n'));
this.parser.end();
assert.equal(this.parser.manifest.dateRanges.length, 3);
});
QUnit.test('parses #EXT-X-INDEPENDENT-SEGMENTS', function(assert) {
this.parser.push([
'#EXTM3U',
'#EXT-X-VERSION:6',
'#EXT-X-SERVER-CONTROL:CAN-BLOCK-RELOAD=YES,PART-HOLD-BACK=3.252,CAN-SKIP-UNTIL=42.0',
'#EXT-X-INDEPENDENT-SEGMENTS'
].join('\n'));
this.parser.end();
assert.equal(this.parser.manifest.independentSegments, true);
});
QUnit.test('parses #EXT-X-I-FRAME-STREAM-INF', function(assert) {
this.parser.push([
'#EXTM3U',
'#EXT-X-I-FRAME-STREAM-INF:BANDWIDTH=86000,URI="low/iframe.m3u8"',
'#EXT-X-I-FRAME-STREAM-INF:BANDWIDTH=150000,URI="mid/iframe.m3u8"',
'#EXT-X-I-FRAME-STREAM-INF:BANDWIDTH=550000,URI="hi/iframe.m3u8"',
'#EXT-X-STREAM-INF:BANDWIDTH=1280000',
'low/audio-video.m3u8',
'#EXT-X-STREAM-INF:BANDWIDTH=2560000',
'mid/audio-video.m3u8',
'#EXT-X-STREAM-INF:BANDWIDTH=7680000',
'hi/audio-video.m3u8',
'#EXT-X-STREAM-INF:BANDWIDTH=65000,CODECS="mp4a.40.5"',
'audio-only.m3u8'
].join('\n'));
this.parser.end();
assert.equal(this.parser.manifest.iFramePlaylists.length, 3);
assert.equal(this.parser.manifest.iFramePlaylists[0].uri, 'low/iframe.m3u8');
assert.strictEqual(this.parser.manifest.iFramePlaylists[0].attributes.BANDWIDTH, 86000);
});
QUnit.test('warns when #EXT-X-I-FRAME-STREAM-INF missing BANDWIDTH/URI attributes', function(assert) {
this.parser.push([
'#EXTM3U',
'#EXT-X-I-FRAME-STREAM-INF:URI="low/iframe.m3u8"',
'#EXT-X-I-FRAME-STREAM-INF:BANDWIDTH=150000,URI="mid/iframe.m3u8"',
'#EXT-X-I-FRAME-STREAM-INF:',
'#EXT-X-STREAM-INF:BANDWIDTH=1280000',
'low/audio-video.m3u8',
'#EXT-X-STREAM-INF:BANDWIDTH=2560000',
'mid/audio-video.m3u8',
'#EXT-X-STREAM-INF:BANDWIDTH=7680000',
'hi/audio-video.m3u8',
'#EXT-X-STREAM-INF:BANDWIDTH=65000,CODECS="mp4a.40.5"',
'audio-only.m3u8'
].join('\n'));
this.parser.end();
const warnings = [
'#EXT-X-I-FRAME-STREAM-INF lacks required attribute(s): BANDWIDTH',
'#EXT-X-I-FRAME-STREAM-INF lacks required attribute(s): BANDWIDTH, URI'
];
assert.deepEqual(
this.warnings,
warnings,
'warnings as expected'
);
});
QUnit.test('warns when #EXT-X-I-FRAMES-ONLY the minimum version required is not supported', function(assert) {
this.parser.push([
'#EXTM3U',
'#EXT-X-VERSION:3',
'#EXT-X-PLAYLIST-TYPE:VOD',
'#EXT-X-MEDIA-SEQUENCE:0',
'#EXT-X-TARGETDURATION:3',
'#EXT-X-I-FRAMES-ONLY',
'#EXTINF:2.002,',
'001.ts',
'#EXTINF:2.002,',
'002.ts',
'#EXTINF:2.002,',
'003.ts',
'#EXTINF:2.002,',
'004.ts',
'#EXTINF:2.002,',
'005.ts',
'#EXTINF:2.002,',
'006.ts',
'#EXT-X-ENDLIST'
].join('\n'));
this.parser.end();
const warnings = [
'manifest must be at least version 4'
];
assert.deepEqual(
this.warnings,
warnings,
'warnings as expected'
);
});
QUnit.test('warns when #EXT-X-I-FRAMES-ONLY does not contain a version number', function(assert) {
this.parser.push([
'#EXTM3U',
'#EXT-X-PLAYLIST-TYPE:VOD',
'#EXT-X-MEDIA-SEQUENCE:0',
'#EXT-X-TARGETDURATION:3',
'#EXT-X-I-FRAMES-ONLY',
'#EXTINF:2.002,',
'001.ts',
'#EXTINF:2.002,',
'002.ts',
'#EXTINF:2.002,',
'003.ts',
'#EXTINF:2.002,',
'004.ts',
'#EXTINF:2.002,',
'005.ts',
'#EXTINF:2.002,',
'006.ts',
'#EXT-X-ENDLIST'
].join('\n'));
this.parser.end();
const warnings = [
'manifest must be at least version 4'
];
assert.deepEqual(
this.warnings,
warnings,
'warnings as expected'
);
});
QUnit.test('parses #EXT-X-CONTENT-STEERING', function(assert) {
const expectedContentSteeringObject = {
serverUri: '/foo?bar=00012',
pathwayId: 'CDN-A'
};
this.parser.push('#EXT-X-CONTENT-STEERING:SERVER-URI="/foo?bar=00012",PATHWAY-ID="CDN-A"');
this.parser.end();
assert.deepEqual(this.parser.manifest.contentSteering, expectedContentSteeringObject);
});
QUnit.test('parses #EXT-X-CONTENT-STEERING without PATHWAY-ID', function(assert) {
const expectedContentSteeringObject = {
serverUri: '/bar?foo=00012'
};
this.parser.push('#EXT-X-CONTENT-STEERING:SERVER-URI="/bar?foo=00012"');
this.parser.end();
assert.deepEqual(this.parser.manifest.contentSteering, expectedContentSteeringObject);
});
QUnit.test('warns on #EXT-X-CONTENT-STEERING missing SERVER-URI', function(assert) {
const warning = ['#EXT-X-CONTENT-STEERING lacks required attribute(s): SERVER-URI'];
this.parser.push('#EXT-X-CONTENT-STEERING:PATHWAY-ID="CDN-A"');
this.parser.end();
assert.deepEqual(this.warnings, warning, 'warnings as expected');
});
QUnit.module('define', {
// https://datatracker.ietf.org/doc/html/draft-pantos-hls-rfc8216bis#section-4.4.2.3
beforeEach() {
this.errors = [];
this.parser.on('error', (err) => this.errors.push(err.message));
}
});
QUnit.test('fails on missing attributes', function(assert) {
const err = ['EXT-X-DEFINE: No attribute'];
this.parser.push('#EXT-X-DEFINE:');
this.parser.end();
assert.deepEqual(this.errors, err, 'errors as expected');
});
QUnit.test('fails on disallowed combinatons', function(assert) {
const permutations = [
'#EXT-X-DEFINE:NAME="a",QUERYPARAM="b"',
'#EXT-X-DEFINE:NAME="a",IMPORT="b"',
'#EXT-X-DEFINE:QUERYPARAM="a",IMPORT="b"',
'#EXT-X-DEFINE:NAME="a",QUERYPARAM="b",IMPORT="c"'
];
assert.expect(permutations.length);
permutations.forEach((p) => {
this.parser = new Parser();
this.parser.on('error', (e) => {
assert.equal(e.message, 'EXT-X-DEFINE: Invalid attributes', `${p} errors as expected`);
});
this.parser.push(p);
this.parser.end();
});
});
QUnit.test('query params substituted', function(assert) {
this.parser = new Parser({
uri: 'https://example.com?aParam=aValue'
});
this.parser.push([
'#EXTM3U',
'#EXT-X-DEFINE:QUERYPARAM="aParam"',
'#EXTINF:10',
'segment.ts?replaced_param={$aParam}'
].join('\n'));
this.parser.end();
assert.equal('aValue', this.parser.manifest.definitions.aParam, 'value of param stored');
assert.equal('segment.ts?replaced_param=aValue', this.parser.manifest.segments[0].uri, 'substituted in url');
});
QUnit.test('query params substituted with relative URL', function(assert) {
this.parser = new Parser({
uri: 'playlist.m3u8?aParam=aValue'
});
this.parser.push([
'#EXTM3U',
'#EXT-X-DEFINE:QUERYPARAM="aParam"',
'#EXTINF:10',
'segment.ts?replaced_param={$aParam}'
].join('\n'));
this.parser.end();
assert.equal('aValue', this.parser.manifest.definitions.aParam, 'value of param stored');
assert.equal('segment.ts?replaced_param=aValue', this.parser.manifest.segments[0].uri, 'substituted in url');
});
QUnit.test('fails with missing query params', function(assert) {
assert.expect(1);
this.parser = new Parser({
uri: 'https://example.com?bParam=bValue'
});
this.parser.on('error', (e) => {
assert.equal(e.message, 'EXT-X-DEFINE: No query param aParam');
});
this.parser.push([
'#EXTM3U',
'#EXT-X-DEFINE:QUERYPARAM="aParam"',
'#EXTINF:10',
'segment.ts?replacedparam={$aParam}'
].join('\n'));
this.parser.end();
});
QUnit.test('fails on redefinition', function(assert) {
const permutations = [
['#EXT-X-DEFINE:NAME="a",VALUE="b"', '#EXT-X-DEFINE:NAME="a",VALUE="c"'],
['#EXT-X-DEFINE:NAME="a",VALUE="b"', '#EXT-X-DEFINE:IMPORT="a"'],
['#EXT-X-DEFINE:NAME="a",VALUE="b"', '#EXT-X-DEFINE:QUERYPARAM="a"'],
['#EXT-X-DEFINE:IMPORT="a"', '#EXT-X-DEFINE:IMPORT="a"'],
['#EXT-X-DEFINE:IMPORT="a"', '#EXT-X-DEFINE:QUERYPARAM="a"'],
['#EXT-X-DEFINE:IMPORT="a"', '#EXT-X-DEFINE:NAME="a",VALUE="c"'],
['#EXT-X-DEFINE:QUERYPARAM="a"', '#EXT-X-DEFINE:IMPORT="a"'],
['#EXT-X-DEFINE:QUERYPARAM="a"', '#EXT-X-DEFINE:QUERYPARAM="a"'],
['#EXT-X-DEFINE:QUERYPARAM="a"', '#EXT-X-DEFINE:NAME="a",VALUE="c"']
];
assert.expect(permutations.length);
permutations.forEach((p) => {
this.parser = new Parser({
uri: 'https:example.com?a=1',
mainDefinitions: {
a: 2
}
});
this.parser.on('error', (e) => {
assert.equal(e.message, 'EXT-X-DEFINE: Duplicate name a', 'errosr on combination');
});
this.parser.push(p.join('\n'));
this.parser.end();
});
});
QUnit.test('fails with IMPORT on main playlist', function(assert) {
this.parser.on('error', function(e) {
assert.equal(e.message, 'EXT-X-DEFINE: No value imported_param to import, or IMPORT used on main playlist', 'fails when missing');
});
this.parser.push('#EXT-X-DEFINE:IMPORT="imported_param"');
this.parser.end();
});
QUnit.test('named and imported substiutions work', function(assert) {
this.parser = new Parser({
mainDefinitions: {
aParam: 'aValue',
engLabel: 'Anglais'
}
});
this.parser.push([
'#EXTM3U',
'#EXT-X-DEFINE:IMPORT="aParam"',
'#EXT-X-DEFINE:NAME="bParam",VALUE="bValue"',
'#EXT-X-DEFINE:IMPORT="engLabel"',
'#EXT-X-MEDIA:TYPE=AUDIO,GROUP-ID="aac",LANGUAGE="eng",NAME="{$engLabel}",AUTOSELECT=YES,DEFAULT=YES,URI="eng/prog_index.m3u8?bParam={$bParam}"',
'#EXTINF:10',
'segment.ts?aParam={$aParam}&bParam={$bParam}'
].join('\n'));
this.parser.end();
assert.equal('aValue', this.parser.manifest.definitions.aParam, 'value of param from import stored');
assert.equal('bValue', this.parser.manifest.definitions.bParam, 'value of param from name stored');
assert.equal('segment.ts?aParam=aValue&bParam=bValue', this.parser.manifest.segments[0].uri, 'substituted in uri');
assert.ok(this.parser.manifest.mediaGroups.AUDIO.aac.hasOwnProperty('Anglais'), 'replacement in attribute');
assert.equal('eng/prog_index.m3u8?bParam=bValue', this.parser.manifest.mediaGroups.AUDIO.aac.Anglais.uri, 'replacement in uri in attribute');
});
QUnit.module('integration');
for (const key in testDataExpected) {
if (!testDataManifests[key]) {
throw new Error(`${key}.js does not have an equivelent m3u8 manifest to test against`);
}
}
for (const key in testDataManifests) {
if (!testDataExpected[key]) {
throw new Error(`${key}.m3u8 does not have an equivelent expected js file to test against`);
}
QUnit.test(`parses ${key}.m3u8 as expected in ${key}.js`, function(assert) {
this.parser.push(testDataManifests[key]());
this.parser.end();
assert.deepEqual(
this.parser.manifest,
testDataExpected[key](),
key + '.m3u8 was parsed correctly'
);
});
}
});