/* * 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 "HeaderFilter.h" #include "sip/parse_common.h" #include "log.h" #include "AmUtils.h" #include const char* FilterType2String(FilterType ft) { switch(ft) { case Transparent: return "transparent"; case Whitelist: return "whitelist"; case Blacklist: return "blacklist"; default: return "unknown"; }; } FilterType String2FilterType(const char* ft) { if (!ft) return Undefined; if (!strcasecmp(ft,"transparent")) return Transparent; if (!strcasecmp(ft,"whitelist")) return Whitelist; if (!strcasecmp(ft,"blacklist")) return Blacklist; return Undefined; } bool isActiveFilter(FilterType ft) { return (ft != Undefined) && (ft != Transparent); } /** @return whether successful */ bool readFilter(AmConfigReader& cfg, const char* cfg_key_filter, const char* cfg_key_list, vector& filter_list, bool keep_transparent_entry) { string filter = cfg.getParameter(cfg_key_filter); if (filter.empty()) return true; FilterEntry hf; hf.filter_type = String2FilterType(filter.c_str()); if (Undefined == hf.filter_type) { ERROR("invalid %s mode '%s'\n", cfg_key_filter, filter.c_str()); return false; } // no transparent filter if (!keep_transparent_entry && hf.filter_type==Transparent) return true; vector elems = explode(cfg.getParameter(cfg_key_list), ","); for (vector::iterator it=elems.begin(); it != elems.end(); it++) { string c = *it; std::transform(c.begin(), c.end(), c.begin(), ::tolower); hf.filter_list.insert(c); } filter_list.push_back(hf); return true; } int skip_header(const std::string& hdr, size_t start_pos, size_t& name_end, size_t& val_begin, size_t& val_end, size_t& hdr_end) { // adapted from sip/parse_header.cpp name_end = val_begin = val_end = start_pos; hdr_end = hdr.length(); // // Header states // enum { H_NAME=0, H_HCOLON, H_VALUE_SWS, H_VALUE, }; int st = H_NAME; int saved_st = 0; size_t p = start_pos; for(;p& filter_list) { if (!hdrs.length() || ! filter_list.size()) return 0; DBG("applying %zd header filters\n", filter_list.size()); for (vector::const_iterator fe = filter_list.begin(); fe != filter_list.end(); fe++) { const set& headerfilter_list = fe->filter_list; const FilterType& f_type = fe->filter_type; if (!isActiveFilter(f_type)) continue; // todo: multi-line header support size_t start_pos = 0; while (start_pos