forked from MirrorHub/synapse
409 lines
14 KiB
JavaScript
409 lines
14 KiB
JavaScript
|
/**
|
||
|
* Contains utility classes used in SDP class.
|
||
|
*
|
||
|
*/
|
||
|
|
||
|
/**
|
||
|
* Class holds a=ssrc lines and media type a=mid
|
||
|
* @param ssrc synchronization source identifier number(a=ssrc lines from SDP)
|
||
|
* @param type media type eg. "audio" or "video"(a=mid frm SDP)
|
||
|
* @constructor
|
||
|
*/
|
||
|
function ChannelSsrc(ssrc, type) {
|
||
|
this.ssrc = ssrc;
|
||
|
this.type = type;
|
||
|
this.lines = [];
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Class holds a=ssrc-group: lines
|
||
|
* @param semantics
|
||
|
* @param ssrcs
|
||
|
* @constructor
|
||
|
*/
|
||
|
function ChannelSsrcGroup(semantics, ssrcs, line) {
|
||
|
this.semantics = semantics;
|
||
|
this.ssrcs = ssrcs;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Helper class represents media channel. Is a container for ChannelSsrc, holds channel idx and media type.
|
||
|
* @param channelNumber channel idx in SDP media array.
|
||
|
* @param mediaType media type(a=mid)
|
||
|
* @constructor
|
||
|
*/
|
||
|
function MediaChannel(channelNumber, mediaType) {
|
||
|
/**
|
||
|
* SDP channel number
|
||
|
* @type {*}
|
||
|
*/
|
||
|
this.chNumber = channelNumber;
|
||
|
/**
|
||
|
* Channel media type(a=mid)
|
||
|
* @type {*}
|
||
|
*/
|
||
|
this.mediaType = mediaType;
|
||
|
/**
|
||
|
* The maps of ssrc numbers to ChannelSsrc objects.
|
||
|
*/
|
||
|
this.ssrcs = {};
|
||
|
|
||
|
/**
|
||
|
* The array of ChannelSsrcGroup objects.
|
||
|
* @type {Array}
|
||
|
*/
|
||
|
this.ssrcGroups = [];
|
||
|
}
|
||
|
|
||
|
SDPUtil = {
|
||
|
iceparams: function (mediadesc, sessiondesc) {
|
||
|
var data = null;
|
||
|
if (SDPUtil.find_line(mediadesc, 'a=ice-ufrag:', sessiondesc) &&
|
||
|
SDPUtil.find_line(mediadesc, 'a=ice-pwd:', sessiondesc)) {
|
||
|
data = {
|
||
|
ufrag: SDPUtil.parse_iceufrag(SDPUtil.find_line(mediadesc, 'a=ice-ufrag:', sessiondesc)),
|
||
|
pwd: SDPUtil.parse_icepwd(SDPUtil.find_line(mediadesc, 'a=ice-pwd:', sessiondesc))
|
||
|
};
|
||
|
}
|
||
|
return data;
|
||
|
},
|
||
|
parse_iceufrag: function (line) {
|
||
|
return line.substring(12);
|
||
|
},
|
||
|
build_iceufrag: function (frag) {
|
||
|
return 'a=ice-ufrag:' + frag;
|
||
|
},
|
||
|
parse_icepwd: function (line) {
|
||
|
return line.substring(10);
|
||
|
},
|
||
|
build_icepwd: function (pwd) {
|
||
|
return 'a=ice-pwd:' + pwd;
|
||
|
},
|
||
|
parse_mid: function (line) {
|
||
|
return line.substring(6);
|
||
|
},
|
||
|
parse_mline: function (line) {
|
||
|
var parts = line.substring(2).split(' '),
|
||
|
data = {};
|
||
|
data.media = parts.shift();
|
||
|
data.port = parts.shift();
|
||
|
data.proto = parts.shift();
|
||
|
if (parts[parts.length - 1] === '') { // trailing whitespace
|
||
|
parts.pop();
|
||
|
}
|
||
|
data.fmt = parts;
|
||
|
return data;
|
||
|
},
|
||
|
build_mline: function (mline) {
|
||
|
return 'm=' + mline.media + ' ' + mline.port + ' ' + mline.proto + ' ' + mline.fmt.join(' ');
|
||
|
},
|
||
|
parse_rtpmap: function (line) {
|
||
|
var parts = line.substring(9).split(' '),
|
||
|
data = {};
|
||
|
data.id = parts.shift();
|
||
|
parts = parts[0].split('/');
|
||
|
data.name = parts.shift();
|
||
|
data.clockrate = parts.shift();
|
||
|
data.channels = parts.length ? parts.shift() : '1';
|
||
|
return data;
|
||
|
},
|
||
|
/**
|
||
|
* Parses SDP line "a=sctpmap:..." and extracts SCTP port from it.
|
||
|
* @param line eg. "a=sctpmap:5000 webrtc-datachannel"
|
||
|
* @returns [SCTP port number, protocol, streams]
|
||
|
*/
|
||
|
parse_sctpmap: function (line)
|
||
|
{
|
||
|
var parts = line.substring(10).split(' ');
|
||
|
var sctpPort = parts[0];
|
||
|
var protocol = parts[1];
|
||
|
// Stream count is optional
|
||
|
var streamCount = parts.length > 2 ? parts[2] : null;
|
||
|
return [sctpPort, protocol, streamCount];// SCTP port
|
||
|
},
|
||
|
build_rtpmap: function (el) {
|
||
|
var line = 'a=rtpmap:' + el.getAttribute('id') + ' ' + el.getAttribute('name') + '/' + el.getAttribute('clockrate');
|
||
|
if (el.getAttribute('channels') && el.getAttribute('channels') != '1') {
|
||
|
line += '/' + el.getAttribute('channels');
|
||
|
}
|
||
|
return line;
|
||
|
},
|
||
|
parse_crypto: function (line) {
|
||
|
var parts = line.substring(9).split(' '),
|
||
|
data = {};
|
||
|
data.tag = parts.shift();
|
||
|
data['crypto-suite'] = parts.shift();
|
||
|
data['key-params'] = parts.shift();
|
||
|
if (parts.length) {
|
||
|
data['session-params'] = parts.join(' ');
|
||
|
}
|
||
|
return data;
|
||
|
},
|
||
|
parse_fingerprint: function (line) { // RFC 4572
|
||
|
var parts = line.substring(14).split(' '),
|
||
|
data = {};
|
||
|
data.hash = parts.shift();
|
||
|
data.fingerprint = parts.shift();
|
||
|
// TODO assert that fingerprint satisfies 2UHEX *(":" 2UHEX) ?
|
||
|
return data;
|
||
|
},
|
||
|
parse_fmtp: function (line) {
|
||
|
var parts = line.split(' '),
|
||
|
i, key, value,
|
||
|
data = [];
|
||
|
parts.shift();
|
||
|
parts = parts.join(' ').split(';');
|
||
|
for (i = 0; i < parts.length; i++) {
|
||
|
key = parts[i].split('=')[0];
|
||
|
while (key.length && key[0] == ' ') {
|
||
|
key = key.substring(1);
|
||
|
}
|
||
|
value = parts[i].split('=')[1];
|
||
|
if (key && value) {
|
||
|
data.push({name: key, value: value});
|
||
|
} else if (key) {
|
||
|
// rfc 4733 (DTMF) style stuff
|
||
|
data.push({name: '', value: key});
|
||
|
}
|
||
|
}
|
||
|
return data;
|
||
|
},
|
||
|
parse_icecandidate: function (line) {
|
||
|
var candidate = {},
|
||
|
elems = line.split(' ');
|
||
|
candidate.foundation = elems[0].substring(12);
|
||
|
candidate.component = elems[1];
|
||
|
candidate.protocol = elems[2].toLowerCase();
|
||
|
candidate.priority = elems[3];
|
||
|
candidate.ip = elems[4];
|
||
|
candidate.port = elems[5];
|
||
|
// elems[6] => "typ"
|
||
|
candidate.type = elems[7];
|
||
|
candidate.generation = 0; // default value, may be overwritten below
|
||
|
for (var i = 8; i < elems.length; i += 2) {
|
||
|
switch (elems[i]) {
|
||
|
case 'raddr':
|
||
|
candidate['rel-addr'] = elems[i + 1];
|
||
|
break;
|
||
|
case 'rport':
|
||
|
candidate['rel-port'] = elems[i + 1];
|
||
|
break;
|
||
|
case 'generation':
|
||
|
candidate.generation = elems[i + 1];
|
||
|
break;
|
||
|
case 'tcptype':
|
||
|
candidate.tcptype = elems[i + 1];
|
||
|
break;
|
||
|
default: // TODO
|
||
|
console.log('parse_icecandidate not translating "' + elems[i] + '" = "' + elems[i + 1] + '"');
|
||
|
}
|
||
|
}
|
||
|
candidate.network = '1';
|
||
|
candidate.id = Math.random().toString(36).substr(2, 10); // not applicable to SDP -- FIXME: should be unique, not just random
|
||
|
return candidate;
|
||
|
},
|
||
|
build_icecandidate: function (cand) {
|
||
|
var line = ['a=candidate:' + cand.foundation, cand.component, cand.protocol, cand.priority, cand.ip, cand.port, 'typ', cand.type].join(' ');
|
||
|
line += ' ';
|
||
|
switch (cand.type) {
|
||
|
case 'srflx':
|
||
|
case 'prflx':
|
||
|
case 'relay':
|
||
|
if (cand.hasOwnAttribute('rel-addr') && cand.hasOwnAttribute('rel-port')) {
|
||
|
line += 'raddr';
|
||
|
line += ' ';
|
||
|
line += cand['rel-addr'];
|
||
|
line += ' ';
|
||
|
line += 'rport';
|
||
|
line += ' ';
|
||
|
line += cand['rel-port'];
|
||
|
line += ' ';
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
if (cand.hasOwnAttribute('tcptype')) {
|
||
|
line += 'tcptype';
|
||
|
line += ' ';
|
||
|
line += cand.tcptype;
|
||
|
line += ' ';
|
||
|
}
|
||
|
line += 'generation';
|
||
|
line += ' ';
|
||
|
line += cand.hasOwnAttribute('generation') ? cand.generation : '0';
|
||
|
return line;
|
||
|
},
|
||
|
parse_ssrc: function (desc) {
|
||
|
// proprietary mapping of a=ssrc lines
|
||
|
// TODO: see "Jingle RTP Source Description" by Juberti and P. Thatcher on google docs
|
||
|
// and parse according to that
|
||
|
var lines = desc.split('\r\n'),
|
||
|
data = {};
|
||
|
for (var i = 0; i < lines.length; i++) {
|
||
|
if (lines[i].substring(0, 7) == 'a=ssrc:') {
|
||
|
var idx = lines[i].indexOf(' ');
|
||
|
data[lines[i].substr(idx + 1).split(':', 2)[0]] = lines[i].substr(idx + 1).split(':', 2)[1];
|
||
|
}
|
||
|
}
|
||
|
return data;
|
||
|
},
|
||
|
parse_rtcpfb: function (line) {
|
||
|
var parts = line.substr(10).split(' ');
|
||
|
var data = {};
|
||
|
data.pt = parts.shift();
|
||
|
data.type = parts.shift();
|
||
|
data.params = parts;
|
||
|
return data;
|
||
|
},
|
||
|
parse_extmap: function (line) {
|
||
|
var parts = line.substr(9).split(' ');
|
||
|
var data = {};
|
||
|
data.value = parts.shift();
|
||
|
if (data.value.indexOf('/') != -1) {
|
||
|
data.direction = data.value.substr(data.value.indexOf('/') + 1);
|
||
|
data.value = data.value.substr(0, data.value.indexOf('/'));
|
||
|
} else {
|
||
|
data.direction = 'both';
|
||
|
}
|
||
|
data.uri = parts.shift();
|
||
|
data.params = parts;
|
||
|
return data;
|
||
|
},
|
||
|
find_line: function (haystack, needle, sessionpart) {
|
||
|
var lines = haystack.split('\r\n');
|
||
|
for (var i = 0; i < lines.length; i++) {
|
||
|
if (lines[i].substring(0, needle.length) == needle) {
|
||
|
return lines[i];
|
||
|
}
|
||
|
}
|
||
|
if (!sessionpart) {
|
||
|
return false;
|
||
|
}
|
||
|
// search session part
|
||
|
lines = sessionpart.split('\r\n');
|
||
|
for (var j = 0; j < lines.length; j++) {
|
||
|
if (lines[j].substring(0, needle.length) == needle) {
|
||
|
return lines[j];
|
||
|
}
|
||
|
}
|
||
|
return false;
|
||
|
},
|
||
|
find_lines: function (haystack, needle, sessionpart) {
|
||
|
var lines = haystack.split('\r\n'),
|
||
|
needles = [];
|
||
|
for (var i = 0; i < lines.length; i++) {
|
||
|
if (lines[i].substring(0, needle.length) == needle)
|
||
|
needles.push(lines[i]);
|
||
|
}
|
||
|
if (needles.length || !sessionpart) {
|
||
|
return needles;
|
||
|
}
|
||
|
// search session part
|
||
|
lines = sessionpart.split('\r\n');
|
||
|
for (var j = 0; j < lines.length; j++) {
|
||
|
if (lines[j].substring(0, needle.length) == needle) {
|
||
|
needles.push(lines[j]);
|
||
|
}
|
||
|
}
|
||
|
return needles;
|
||
|
},
|
||
|
candidateToJingle: function (line) {
|
||
|
// a=candidate:2979166662 1 udp 2113937151 192.168.2.100 57698 typ host generation 0
|
||
|
// <candidate component=... foundation=... generation=... id=... ip=... network=... port=... priority=... protocol=... type=.../>
|
||
|
if (line.indexOf('candidate:') === 0) {
|
||
|
line = 'a=' + line;
|
||
|
} else if (line.substring(0, 12) != 'a=candidate:') {
|
||
|
console.log('parseCandidate called with a line that is not a candidate line');
|
||
|
console.log(line);
|
||
|
return null;
|
||
|
}
|
||
|
if (line.substring(line.length - 2) == '\r\n') // chomp it
|
||
|
line = line.substring(0, line.length - 2);
|
||
|
var candidate = {},
|
||
|
elems = line.split(' '),
|
||
|
i;
|
||
|
if (elems[6] != 'typ') {
|
||
|
console.log('did not find typ in the right place');
|
||
|
console.log(line);
|
||
|
return null;
|
||
|
}
|
||
|
candidate.foundation = elems[0].substring(12);
|
||
|
candidate.component = elems[1];
|
||
|
candidate.protocol = elems[2].toLowerCase();
|
||
|
candidate.priority = elems[3];
|
||
|
candidate.ip = elems[4];
|
||
|
candidate.port = elems[5];
|
||
|
// elems[6] => "typ"
|
||
|
candidate.type = elems[7];
|
||
|
|
||
|
candidate.generation = '0'; // default, may be overwritten below
|
||
|
for (i = 8; i < elems.length; i += 2) {
|
||
|
switch (elems[i]) {
|
||
|
case 'raddr':
|
||
|
candidate['rel-addr'] = elems[i + 1];
|
||
|
break;
|
||
|
case 'rport':
|
||
|
candidate['rel-port'] = elems[i + 1];
|
||
|
break;
|
||
|
case 'generation':
|
||
|
candidate.generation = elems[i + 1];
|
||
|
break;
|
||
|
case 'tcptype':
|
||
|
candidate.tcptype = elems[i + 1];
|
||
|
break;
|
||
|
default: // TODO
|
||
|
console.log('not translating "' + elems[i] + '" = "' + elems[i + 1] + '"');
|
||
|
}
|
||
|
}
|
||
|
candidate.network = '1';
|
||
|
candidate.id = Math.random().toString(36).substr(2, 10); // not applicable to SDP -- FIXME: should be unique, not just random
|
||
|
return candidate;
|
||
|
},
|
||
|
candidateFromJingle: function (cand) {
|
||
|
var line = 'a=candidate:';
|
||
|
line += cand.getAttribute('foundation');
|
||
|
line += ' ';
|
||
|
line += cand.getAttribute('component');
|
||
|
line += ' ';
|
||
|
line += cand.getAttribute('protocol'); //.toUpperCase(); // chrome M23 doesn't like this
|
||
|
line += ' ';
|
||
|
line += cand.getAttribute('priority');
|
||
|
line += ' ';
|
||
|
line += cand.getAttribute('ip');
|
||
|
line += ' ';
|
||
|
line += cand.getAttribute('port');
|
||
|
line += ' ';
|
||
|
line += 'typ';
|
||
|
line += ' ' + cand.getAttribute('type');
|
||
|
line += ' ';
|
||
|
switch (cand.getAttribute('type')) {
|
||
|
case 'srflx':
|
||
|
case 'prflx':
|
||
|
case 'relay':
|
||
|
if (cand.getAttribute('rel-addr') && cand.getAttribute('rel-port')) {
|
||
|
line += 'raddr';
|
||
|
line += ' ';
|
||
|
line += cand.getAttribute('rel-addr');
|
||
|
line += ' ';
|
||
|
line += 'rport';
|
||
|
line += ' ';
|
||
|
line += cand.getAttribute('rel-port');
|
||
|
line += ' ';
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
if (cand.getAttribute('protocol').toLowerCase() == 'tcp') {
|
||
|
line += 'tcptype';
|
||
|
line += ' ';
|
||
|
line += cand.getAttribute('tcptype');
|
||
|
line += ' ';
|
||
|
}
|
||
|
line += 'generation';
|
||
|
line += ' ';
|
||
|
line += cand.getAttribute('generation') || '0';
|
||
|
return line + '\r\n';
|
||
|
}
|
||
|
};
|
||
|
|
||
|
exports.SDPUtil = SDPUtil;
|
||
|
|