mirror of https://github.com/sipwise/sems.git
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
247 lines
7.9 KiB
247 lines
7.9 KiB
/*
|
|
* Copyright (C) 2010 Stefan Sayer
|
|
*
|
|
* This file is part of SEMS, a free SIP media server.
|
|
*
|
|
* SEMS is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* For a license to use the SEMS software under conditions
|
|
* other than those described here, or to purchase support for this
|
|
* software, please contact iptel.org by e-mail at the following addresses:
|
|
* info@iptel.org
|
|
*
|
|
* SEMS is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
*/
|
|
|
|
#include "SDPFilter.h"
|
|
#include <algorithm>
|
|
#include "log.h"
|
|
#include "AmUtils.h"
|
|
#include "RTPParameters.h"
|
|
|
|
int filterSDP(AmSdp& sdp, const vector<FilterEntry>& filter_list) {
|
|
|
|
for (vector<FilterEntry>::const_iterator it=
|
|
filter_list.begin(); it!=filter_list.end(); it++){
|
|
|
|
const FilterType& sdpfilter = it->filter_type;
|
|
const std::set<string>& sdpfilter_list = it->filter_list;
|
|
|
|
bool media_line_filtered_out = false;
|
|
bool media_line_left = false;
|
|
|
|
if (!isActiveFilter(sdpfilter))
|
|
continue;
|
|
|
|
for (std::vector<SdpMedia>::iterator m_it =
|
|
sdp.media.begin(); m_it != sdp.media.end(); m_it++) {
|
|
SdpMedia& media = *m_it;
|
|
|
|
std::vector<SdpPayload> new_pl;
|
|
for (std::vector<SdpPayload>::iterator p_it =
|
|
media.payloads.begin(); p_it != media.payloads.end(); p_it++) {
|
|
|
|
string c = p_it->encoding_name;
|
|
std::transform(c.begin(), c.end(), c.begin(), ::tolower);
|
|
|
|
bool is_filtered = (sdpfilter == Whitelist) ^
|
|
(sdpfilter_list.find(c) != sdpfilter_list.end());
|
|
|
|
// DBG("%s (%s) is_filtered: %s\n", p_it->encoding_name.c_str(), c.c_str(),
|
|
// is_filtered?"true":"false");
|
|
|
|
if (!is_filtered)
|
|
new_pl.push_back(*p_it);
|
|
}
|
|
// if no payload supported any more: leave at least one, and set port to 0
|
|
// RFC 3264 section 6.1
|
|
if(!new_pl.size() && media.payloads.size()) {
|
|
std::vector<SdpPayload>::iterator p_it = media.payloads.begin();
|
|
new_pl.push_back(*p_it);
|
|
media.port = 0;
|
|
media_line_filtered_out = true;
|
|
}
|
|
else media_line_left = true;
|
|
media.payloads = new_pl;
|
|
}
|
|
if ((!media_line_left) && media_line_filtered_out) {
|
|
// no filter adds new payloads, we can safely return error
|
|
DBG("all streams were marked as inactive\n");
|
|
return -488;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int filterMedia(AmSdp& sdp, const vector<FilterEntry>& filter_list)
|
|
{
|
|
unsigned filtered_out = 0;
|
|
|
|
DBG("filtering media types\n");
|
|
for (vector<FilterEntry>::const_iterator i = filter_list.begin(); i !=filter_list.end(); ++i) {
|
|
|
|
const FilterType& filter = i->filter_type;
|
|
const std::set<string>& media_list = i->filter_list;
|
|
|
|
if (!isActiveFilter(filter)) continue;
|
|
|
|
for (std::vector<SdpMedia>::iterator m = sdp.media.begin(); m != sdp.media.end(); ++m) {
|
|
if (m->port == 0) continue; // already inactive
|
|
|
|
string type(SdpMedia::type2str(m->type));
|
|
DBG("checking whether to filter out '%s'\n", type.c_str());
|
|
|
|
bool is_filtered = (filter == Whitelist) ^ (media_list.find(type) != media_list.end());
|
|
if (is_filtered) {
|
|
m->port = 0;
|
|
filtered_out++;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (filtered_out > 0 && filtered_out == sdp.media.size()) {
|
|
// we filtered out all media lines (if there was an inactive stream before
|
|
// it was probably intendend)
|
|
DBG("all streams were marked as inactive\n");
|
|
return -488;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
void fix_missing_encodings(SdpMedia& m) {
|
|
for (std::vector<SdpPayload>::iterator p_it=
|
|
m.payloads.begin(); p_it!=m.payloads.end(); p_it++) {
|
|
SdpPayload& p = *p_it;
|
|
if (!p.encoding_name.empty())
|
|
continue;
|
|
if (p.payload_type > (IANA_RTP_PAYLOADS_SIZE-1) || p.payload_type < 0)
|
|
continue; // todo: throw out this payload
|
|
if (IANA_RTP_PAYLOADS[p.payload_type].payload_name[0]=='\0')
|
|
continue; // todo: throw out this payload
|
|
|
|
p.encoding_name = IANA_RTP_PAYLOADS[p.payload_type].payload_name;
|
|
p.clock_rate = IANA_RTP_PAYLOADS[p.payload_type].clock_rate;
|
|
if (IANA_RTP_PAYLOADS[p.payload_type].channels > 1)
|
|
p.encoding_param = IANA_RTP_PAYLOADS[p.payload_type].channels;
|
|
|
|
DBG("named SDP payload type %d with %s/%d%s\n",
|
|
p.payload_type, IANA_RTP_PAYLOADS[p.payload_type].payload_name,
|
|
IANA_RTP_PAYLOADS[p.payload_type].clock_rate,
|
|
IANA_RTP_PAYLOADS[p.payload_type].channels > 1 ?
|
|
("/"+int2str(IANA_RTP_PAYLOADS[p.payload_type].channels)).c_str() : "");
|
|
}
|
|
}
|
|
|
|
void fix_incomplete_silencesupp(SdpMedia& m) {
|
|
for (std::vector<SdpAttribute>::iterator a_it =
|
|
m.attributes.begin(); a_it != m.attributes.end(); a_it++) {
|
|
if (a_it->attribute == "silenceSupp") {
|
|
vector<string> parts = explode(a_it->value, " ");
|
|
if (parts.size() < 5) {
|
|
string val_before = a_it->value;
|
|
for (int i=parts.size();i<5;i++)
|
|
a_it->value += " -";
|
|
DBG("fixed SDP attribute silenceSupp:'%s' -> '%s'\n",
|
|
val_before.c_str(), a_it->value.c_str());
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
std::vector<SdpAttribute> filterSDPAttributes(std::vector<SdpAttribute> attributes,
|
|
FilterType sdpalinesfilter, const std::set<string>& sdpalinesfilter_list) {
|
|
|
|
std::vector<SdpAttribute> res;
|
|
for (std::vector<SdpAttribute>::iterator a_it =
|
|
attributes.begin(); a_it != attributes.end(); a_it++) {
|
|
|
|
// Case insensitive search:
|
|
string c = a_it->attribute;
|
|
std::transform(c.begin(), c.end(), c.begin(), ::tolower);
|
|
|
|
// Check, if this should be filtered:
|
|
bool is_filtered = (sdpalinesfilter == Whitelist) ^
|
|
(sdpalinesfilter_list.find(c) != sdpalinesfilter_list.end());
|
|
|
|
DBG("%s (%s) is_filtered: %s\n", a_it->attribute.c_str(), c.c_str(),
|
|
is_filtered?"true":"false");
|
|
|
|
// If it is not filtered, just add it to the list:
|
|
if (!is_filtered)
|
|
res.push_back(*a_it);
|
|
}
|
|
return res;
|
|
}
|
|
|
|
int filterSDPalines(AmSdp& sdp, const vector<FilterEntry>& filter_list) {
|
|
for (vector<FilterEntry>::const_iterator it=
|
|
filter_list.begin(); it!=filter_list.end(); it++){
|
|
|
|
const FilterType& sdpalinesfilter = it->filter_type;
|
|
const std::set<string>& sdpalinesfilter_list = it->filter_list;
|
|
|
|
// If not Black- or Whitelist, simply return
|
|
if (!isActiveFilter(sdpalinesfilter))
|
|
continue;
|
|
|
|
// We start with per Session-alines
|
|
sdp.attributes =
|
|
filterSDPAttributes(sdp.attributes, sdpalinesfilter, sdpalinesfilter_list);
|
|
|
|
for (std::vector<SdpMedia>::iterator m_it =
|
|
sdp.media.begin(); m_it != sdp.media.end(); m_it++) {
|
|
SdpMedia& media = *m_it;
|
|
// todo: what if no payload supported any more?
|
|
media.attributes =
|
|
filterSDPAttributes(media.attributes, sdpalinesfilter, sdpalinesfilter_list);
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int normalizeSDP(AmSdp& sdp, bool anonymize_sdp, const string &advertised_ip) {
|
|
for (std::vector<SdpMedia>::iterator m_it=
|
|
sdp.media.begin(); m_it != sdp.media.end(); m_it++) {
|
|
if (m_it->type != MT_AUDIO && m_it->type != MT_VIDEO)
|
|
continue;
|
|
|
|
// fill missing encoding names (a= lines)
|
|
fix_missing_encodings(*m_it);
|
|
|
|
// fix incomplete silenceSupp attributes (see RFC3108)
|
|
// (only media level - RFC3108 4.)
|
|
fix_incomplete_silencesupp(*m_it);
|
|
}
|
|
|
|
if (anonymize_sdp) {
|
|
// Clear s-Line in SDP:
|
|
sdp.sessionName = "-";
|
|
// Clear u-Line in SDP:
|
|
sdp.uri.clear();
|
|
// Clear origin user
|
|
sdp.origin.user = "-";
|
|
if (!advertised_ip.empty()) {
|
|
sdp.origin.conn.address = advertised_ip;
|
|
// SdpConnection::ipv4/ipv6 seems not to be used so we won't replace it there
|
|
// TODO: replace in attributes
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|