2014-06-01 18:43:18 +00:00
|
|
|
var Logger = require("./logger");
|
|
|
|
var Config = require("./config");
|
|
|
|
var Metadata;
|
|
|
|
var enabled = false;
|
|
|
|
|
|
|
|
function init() {
|
|
|
|
if (Config.get("ffmpeg.enabled")) {
|
|
|
|
try {
|
|
|
|
Metadata = require("fluent-ffmpeg").Metadata;
|
|
|
|
Logger.syslog.log("Enabling raw file support with fluent-ffmpeg");
|
|
|
|
enabled = true;
|
2014-06-06 04:58:26 +00:00
|
|
|
} catch (e) {
|
2014-06-01 18:43:18 +00:00
|
|
|
Logger.errlog.log("Failed to load fluent-ffmpeg. Did you remember to " +
|
|
|
|
"execute `npm install fluent-ffmpeg` ?");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
var acceptedCodecs = {
|
|
|
|
"mov/h264": true,
|
2014-06-04 04:21:00 +00:00
|
|
|
"flv/h264": true,
|
|
|
|
"matroska/vp8": true,
|
|
|
|
"matroska/vp9": true,
|
2014-06-07 17:45:52 +00:00
|
|
|
"ogg/theora": true
|
2014-06-01 18:43:18 +00:00
|
|
|
};
|
|
|
|
|
2014-06-06 04:58:26 +00:00
|
|
|
var acceptedAudioCodecs = {
|
|
|
|
"mp3": true,
|
|
|
|
"vorbis": true
|
|
|
|
};
|
|
|
|
|
2014-06-07 17:45:52 +00:00
|
|
|
var audioOnlyContainers = {
|
|
|
|
"mp3": true
|
|
|
|
};
|
|
|
|
|
2014-06-01 18:43:18 +00:00
|
|
|
exports.query = function (filename, cb) {
|
|
|
|
if (!Metadata) {
|
|
|
|
init();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!enabled) {
|
|
|
|
return cb("Raw file playback is not enabled on this server");
|
|
|
|
}
|
|
|
|
|
2014-06-07 17:45:52 +00:00
|
|
|
if (!filename.match(/^https?:\/\//)) {
|
|
|
|
return cb("Raw file playback is only supported for links accessible via HTTP " +
|
|
|
|
"or HTTPS");
|
|
|
|
}
|
|
|
|
|
2014-06-01 18:43:18 +00:00
|
|
|
new Metadata(filename, function (meta, err) {
|
|
|
|
if (err) {
|
|
|
|
return cb(err);
|
|
|
|
}
|
|
|
|
|
2014-06-06 04:58:26 +00:00
|
|
|
if (isVideo(meta)) {
|
|
|
|
var video = meta.video;
|
|
|
|
var codec = video.container + "/" + video.codec;
|
2014-06-01 18:43:18 +00:00
|
|
|
|
2014-06-06 04:58:26 +00:00
|
|
|
if (!(codec in acceptedCodecs)) {
|
|
|
|
return cb("Unsupported video codec " + codec);
|
|
|
|
}
|
2014-06-01 18:43:18 +00:00
|
|
|
|
2014-06-06 04:58:26 +00:00
|
|
|
var data = {
|
|
|
|
title: meta.title || "Raw Video",
|
2014-06-07 17:45:52 +00:00
|
|
|
duration: Math.ceil(meta.durationsec),
|
2014-06-06 04:58:26 +00:00
|
|
|
bitrate: video.bitrate,
|
|
|
|
codec: codec
|
|
|
|
};
|
|
|
|
|
|
|
|
cb(null, data);
|
|
|
|
} else if (isAudio(meta)) {
|
|
|
|
var audio = meta.audio;
|
|
|
|
var codec = audio.codec;
|
|
|
|
|
|
|
|
if (!(codec in acceptedAudioCodecs)) {
|
|
|
|
return cb("Unsupported audio codec " + codec);
|
|
|
|
}
|
2014-06-01 18:43:18 +00:00
|
|
|
|
2014-06-06 04:58:26 +00:00
|
|
|
var data = {
|
|
|
|
title: meta.title || "Raw Audio",
|
2014-06-07 17:45:52 +00:00
|
|
|
duration: Math.ceil(meta.durationsec),
|
2014-06-06 04:58:26 +00:00
|
|
|
bitrate: audio.bitrate,
|
|
|
|
codec: codec
|
|
|
|
};
|
2014-06-01 18:43:18 +00:00
|
|
|
|
2014-06-06 04:58:26 +00:00
|
|
|
cb(null, data);
|
2014-06-07 17:45:52 +00:00
|
|
|
} else if (data.ffmpegErr.match(/Protocol not found/)) {
|
|
|
|
return cb("This server is unable to load videos over the " +
|
|
|
|
filename.split(":")[0] + " protocol.");
|
2014-06-06 04:58:26 +00:00
|
|
|
} else {
|
|
|
|
return cb("Parsed metadata did not contain a valid video or audio stream. " +
|
|
|
|
"Either the file is invalid or it has a format unsupported by " +
|
|
|
|
"this server's version of ffmpeg.");
|
|
|
|
}
|
2014-06-01 18:43:18 +00:00
|
|
|
});
|
|
|
|
};
|
2014-06-06 04:58:26 +00:00
|
|
|
|
|
|
|
function isVideo(meta) {
|
2014-06-07 17:45:52 +00:00
|
|
|
return meta.video &&
|
|
|
|
meta.video.bitrate > 0 &&
|
|
|
|
meta.video.container &&
|
|
|
|
meta.video.codec &&
|
|
|
|
!(meta.video.container in audioOnlyContainers);
|
2014-06-06 04:58:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
function isAudio(meta) {
|
|
|
|
return meta.audio && meta.audio.bitrate > 0 && meta.audio.codec;
|
|
|
|
}
|