2015-07-10 09:57:20 +00:00
|
|
|
var SDPUtil = require("./SDPUtil");
|
|
|
|
|
2015-09-09 19:09:58 +00:00
|
|
|
function SDPDiffer(mySDP, otherSDP)
|
|
|
|
{
|
2015-01-19 09:20:00 +00:00
|
|
|
this.mySDP = mySDP;
|
|
|
|
this.otherSDP = otherSDP;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2015-09-09 19:09:58 +00:00
|
|
|
* Returns map of MediaChannel that contains media contained in
|
|
|
|
* 'mySDP', but not contained in 'otherSdp'. Mapped by channel idx.
|
2015-01-19 09:20:00 +00:00
|
|
|
*/
|
|
|
|
SDPDiffer.prototype.getNewMedia = function() {
|
|
|
|
|
|
|
|
// this could be useful in Array.prototype.
|
|
|
|
function arrayEquals(array) {
|
|
|
|
// if the other array is a falsy value, return
|
|
|
|
if (!array)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// compare lengths - can save a lot of time
|
|
|
|
if (this.length != array.length)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
for (var i = 0, l=this.length; i < l; i++) {
|
|
|
|
// Check if we have nested arrays
|
|
|
|
if (this[i] instanceof Array && array[i] instanceof Array) {
|
|
|
|
// recurse into the nested arrays
|
|
|
|
if (!this[i].equals(array[i]))
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
else if (this[i] != array[i]) {
|
2015-09-09 19:09:58 +00:00
|
|
|
// Warning - two different object instances will never be
|
|
|
|
// equal: {x:20} != {x:20}
|
2015-01-19 09:20:00 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
var myMedias = this.mySDP.getMediaSsrcMap();
|
|
|
|
var othersMedias = this.otherSDP.getMediaSsrcMap();
|
|
|
|
var newMedia = {};
|
|
|
|
Object.keys(othersMedias).forEach(function(othersMediaIdx) {
|
|
|
|
var myMedia = myMedias[othersMediaIdx];
|
|
|
|
var othersMedia = othersMedias[othersMediaIdx];
|
|
|
|
if(!myMedia && othersMedia) {
|
|
|
|
// Add whole channel
|
|
|
|
newMedia[othersMediaIdx] = othersMedia;
|
|
|
|
return;
|
|
|
|
}
|
2015-09-09 19:09:58 +00:00
|
|
|
// Look for new ssrcs across the channel
|
2015-01-19 09:20:00 +00:00
|
|
|
Object.keys(othersMedia.ssrcs).forEach(function(ssrc) {
|
|
|
|
if(Object.keys(myMedia.ssrcs).indexOf(ssrc) === -1) {
|
2015-09-09 19:09:58 +00:00
|
|
|
// Allocate channel if we've found ssrc that doesn't exist in
|
|
|
|
// our channel
|
2015-01-19 09:20:00 +00:00
|
|
|
if(!newMedia[othersMediaIdx]){
|
|
|
|
newMedia[othersMediaIdx] = {
|
|
|
|
mediaindex: othersMedia.mediaindex,
|
|
|
|
mid: othersMedia.mid,
|
|
|
|
ssrcs: {},
|
|
|
|
ssrcGroups: []
|
|
|
|
};
|
|
|
|
}
|
|
|
|
newMedia[othersMediaIdx].ssrcs[ssrc] = othersMedia.ssrcs[ssrc];
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
// Look for new ssrc groups across the channels
|
|
|
|
othersMedia.ssrcGroups.forEach(function(otherSsrcGroup){
|
|
|
|
|
|
|
|
// try to match the other ssrc-group with an ssrc-group of ours
|
|
|
|
var matched = false;
|
|
|
|
for (var i = 0; i < myMedia.ssrcGroups.length; i++) {
|
|
|
|
var mySsrcGroup = myMedia.ssrcGroups[i];
|
2015-09-09 19:09:58 +00:00
|
|
|
if (otherSsrcGroup.semantics == mySsrcGroup.semantics &&
|
|
|
|
arrayEquals.apply(otherSsrcGroup.ssrcs,
|
|
|
|
[mySsrcGroup.ssrcs])) {
|
2015-01-19 09:20:00 +00:00
|
|
|
|
|
|
|
matched = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!matched) {
|
|
|
|
// Allocate channel if we've found an ssrc-group that doesn't
|
|
|
|
// exist in our channel
|
|
|
|
|
|
|
|
if(!newMedia[othersMediaIdx]){
|
|
|
|
newMedia[othersMediaIdx] = {
|
|
|
|
mediaindex: othersMedia.mediaindex,
|
|
|
|
mid: othersMedia.mid,
|
|
|
|
ssrcs: {},
|
|
|
|
ssrcGroups: []
|
|
|
|
};
|
|
|
|
}
|
|
|
|
newMedia[othersMediaIdx].ssrcGroups.push(otherSsrcGroup);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
return newMedia;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
2015-09-09 19:09:58 +00:00
|
|
|
* TODO: document!
|
2015-01-19 09:20:00 +00:00
|
|
|
*/
|
2015-08-05 12:09:12 +00:00
|
|
|
SDPDiffer.prototype.toJingle = function(modify) {
|
2015-01-19 09:20:00 +00:00
|
|
|
var sdpMediaSsrcs = this.getNewMedia();
|
|
|
|
|
|
|
|
var modified = false;
|
|
|
|
Object.keys(sdpMediaSsrcs).forEach(function(mediaindex){
|
|
|
|
modified = true;
|
|
|
|
var media = sdpMediaSsrcs[mediaindex];
|
|
|
|
modify.c('content', {name: media.mid});
|
|
|
|
|
2015-09-09 19:09:58 +00:00
|
|
|
modify.c('description',
|
|
|
|
{xmlns:'urn:xmpp:jingle:apps:rtp:1', media: media.mid});
|
|
|
|
// FIXME: not completely sure this operates on blocks and / or handles
|
|
|
|
// different ssrcs correctly
|
2015-01-19 09:20:00 +00:00
|
|
|
// generate sources from lines
|
|
|
|
Object.keys(media.ssrcs).forEach(function(ssrcNum) {
|
|
|
|
var mediaSsrc = media.ssrcs[ssrcNum];
|
|
|
|
modify.c('source', { xmlns: 'urn:xmpp:jingle:apps:rtp:ssma:0' });
|
|
|
|
modify.attrs({ssrc: mediaSsrc.ssrc});
|
|
|
|
// iterate over ssrc lines
|
|
|
|
mediaSsrc.lines.forEach(function (line) {
|
|
|
|
var idx = line.indexOf(' ');
|
|
|
|
var kv = line.substr(idx + 1);
|
|
|
|
modify.c('parameter');
|
|
|
|
if (kv.indexOf(':') == -1) {
|
|
|
|
modify.attrs({ name: kv });
|
|
|
|
} else {
|
2015-07-10 09:57:20 +00:00
|
|
|
var nv = kv.split(':', 2);
|
|
|
|
var name = nv[0];
|
|
|
|
var value = SDPUtil.filter_special_chars(nv[1]);
|
|
|
|
modify.attrs({ name: name });
|
|
|
|
modify.attrs({ value: value });
|
2015-01-19 09:20:00 +00:00
|
|
|
}
|
|
|
|
modify.up(); // end of parameter
|
|
|
|
});
|
|
|
|
modify.up(); // end of source
|
|
|
|
});
|
|
|
|
|
|
|
|
// generate source groups from lines
|
|
|
|
media.ssrcGroups.forEach(function(ssrcGroup) {
|
2015-09-11 03:26:29 +00:00
|
|
|
if (ssrcGroup.ssrcs.length) {
|
2015-01-19 09:20:00 +00:00
|
|
|
|
|
|
|
modify.c('ssrc-group', {
|
|
|
|
semantics: ssrcGroup.semantics,
|
|
|
|
xmlns: 'urn:xmpp:jingle:apps:rtp:ssma:0'
|
|
|
|
});
|
|
|
|
|
|
|
|
ssrcGroup.ssrcs.forEach(function (ssrc) {
|
|
|
|
modify.c('source', { ssrc: ssrc })
|
|
|
|
.up(); // end of source
|
|
|
|
});
|
|
|
|
modify.up(); // end of ssrc-group
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
modify.up(); // end of description
|
|
|
|
modify.up(); // end of content
|
|
|
|
});
|
|
|
|
|
|
|
|
return modified;
|
|
|
|
};
|
|
|
|
|
|
|
|
module.exports = SDPDiffer;
|