Source code

Revision control

Copy as Markdown

Other Tools

/* -*- Mode: JavaScript; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
/* vim: set ts=8 sts=2 et sw=2 tw=80: */
/* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
/* dash_detect_stream_switch.sjs
*
* Parses requests for DASH manifests and ensures stream switching takes place
* by verifying the subsegments downloaded and the streams they belong to.
* If unexpected subsegments (byte ranges) are requested, the script will
* will respond with a 404.
*/
var DEBUG = false;
function parseQuery(request, key) {
var params = request.queryString.split("&");
if (DEBUG) {
dump('DASH-SJS: request params = "' + params + '"\n');
}
for (var j = 0; j < params.length; ++j) {
var p = params[j];
if (p == key) {
return true;
}
if (p.indexOf(key + "=") === 0) {
return p.substring(key.length + 1);
}
if (!p.includes("=") && key === "") {
return p;
}
}
return false;
}
function handleRequest(request, response) {
try {
var name = parseQuery(request, "name");
var range = request.hasHeader("Range")
? request.getHeader("Range")
: undefined;
// Should not get request for 1st subsegment from 2nd stream, nor 2nd
// subsegment from 1st stream.
if (
(name == "dash-webm-video-320x180.webm" &&
range == "bytes=25514-32767") ||
(name == "dash-webm-video-428x240.webm" && range == "bytes=228-35852")
) {
throw new Error(
"Should not request " + name + " with byte-range " + range
);
} else {
var rangeSplit = range.split("=");
if (rangeSplit.length != 2) {
throw new Error(
"DASH-SJS: ERROR: invalid number of tokens (" +
rangeSplit.length +
") delimited by '=' in 'Range' header."
);
}
var offsets = rangeSplit[1].split("-");
if (offsets.length != 2) {
throw new Error(
"DASH-SJS: ERROR: invalid number of tokens (" +
offsets.length +
") delimited by '-' in 'Range' header."
);
}
var startOffset = parseInt(offsets[0]);
var endOffset = parseInt(offsets[1]);
var file = Services.dirsvc.get("CurWorkD", Ci.nsIFile);
var fis = Cc["@mozilla.org/network/file-input-stream;1"].createInstance(
Ci.nsIFileInputStream
);
var bis = Cc["@mozilla.org/binaryinputstream;1"].createInstance(
Ci.nsIBinaryInputStream
);
var paths = "tests/dom/media/test/" + name;
var split = paths.split("/");
for (var i = 0; i < split.length; ++i) {
file.append(split[i]);
}
fis.init(file, -1, -1, false);
// Exception: start offset should be within file bounds.
if (startOffset > file.fileSize) {
throw new Error(
"Starting offset [" +
startOffset +
"] is after end of file [" +
file.fileSize +
"]."
);
}
// End offset may be too large in the MPD. Real world HTTP servers just
// return what data they can; do the same here - reduce the end offset.
if (endOffset >= file.fileSize) {
if (DEBUG) {
dump(
"DASH-SJS: reducing endOffset [" +
endOffset +
"] to fileSize [" +
(file.fileSize - 1) +
"]\n"
);
}
endOffset = file.fileSize - 1;
}
fis.seek(Ci.nsISeekableStream.NS_SEEK_SET, startOffset);
bis.setInputStream(fis);
var byteLengthToRead = endOffset + 1 - startOffset;
var totalBytesExpected = byteLengthToRead + startOffset;
if (DEBUG) {
dump(
"DASH-SJS: byteLengthToRead = " +
byteLengthToRead +
" byteLengthToRead+startOffset = " +
totalBytesExpected +
" fileSize = " +
file.fileSize +
"\n"
);
}
var bytes = bis.readBytes(byteLengthToRead);
response.setStatusLine(request.httpVersion, 206, "Partial Content");
response.setHeader("Content-Length", "" + bytes.length, false);
response.setHeader("Content-Type", "application/dash+xml", false);
var contentRange =
"bytes " + startOffset + "-" + endOffset + "/" + file.fileSize;
response.setHeader("Content-Range", contentRange, false);
response.write(bytes, bytes.length);
bis.close();
}
} catch (e) {
dump("DASH-SJS-ERROR: " + e + "\n");
response.setStatusLine(request.httpVersion, 404, "Not found");
}
}