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.
432 lines
12 KiB
432 lines
12 KiB
/*
|
|
* $Id$
|
|
*
|
|
* Copyright (C) 2002-2003 Fhg Fokus
|
|
*
|
|
* 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 ser 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 <sys/types.h>
|
|
#include <sys/socket.h>
|
|
#include <netinet/in.h>
|
|
#include <arpa/inet.h>
|
|
#include <netdb.h>
|
|
#include <stdio.h>
|
|
#include "AmConfig.h"
|
|
#include "sems.h"
|
|
#include "log.h"
|
|
#include "AmConfigReader.h"
|
|
#include "AmUtils.h"
|
|
|
|
#include <fstream>
|
|
|
|
string AmConfig::ConfigurationFile = CONFIG_FILE;
|
|
string AmConfig::ModConfigPath = MOD_CFG_PATH;
|
|
string AmConfig::PlugInPath = PLUG_IN_PATH;
|
|
string AmConfig::LoadPlugins = "";
|
|
string AmConfig::ExcludePlugins = "";
|
|
string AmConfig::ExcludePayloads = "";
|
|
int AmConfig::DaemonMode = DEFAULT_DAEMON_MODE;
|
|
string AmConfig::LocalIP = "";
|
|
string AmConfig::PublicIP = "";
|
|
string AmConfig::PrefixSep = PREFIX_SEPARATOR;
|
|
int AmConfig::RtpLowPort = RTP_LOWPORT;
|
|
int AmConfig::RtpHighPort = RTP_HIGHPORT;
|
|
int AmConfig::SessionProcessorThreads = NUM_SESSION_PROCESSORS;
|
|
int AmConfig::MediaProcessorThreads = NUM_MEDIA_PROCESSORS;
|
|
int AmConfig::SIPServerThreads = NUM_SIP_SERVERS;
|
|
int AmConfig::LocalSIPPort = 5060;
|
|
string AmConfig::LocalSIPIP = "";
|
|
string AmConfig::OutboundProxy = "";
|
|
string AmConfig::Signature = "";
|
|
bool AmConfig::SingleCodecInOK = false;
|
|
unsigned int AmConfig::DeadRtpTime = DEAD_RTP_TIME;
|
|
bool AmConfig::IgnoreRTPXHdrs = false;
|
|
string AmConfig::Application = "";
|
|
AmConfig::ApplicationSelector AmConfig::AppSelect = AmConfig::App_SPECIFIED;
|
|
AmConfig::AppMappingVector AmConfig::AppMapping;
|
|
bool AmConfig::LogSessions = false;
|
|
bool AmConfig::LogEvents = false;
|
|
int AmConfig::UnhandledReplyLoglevel = 0;
|
|
|
|
unsigned int AmConfig::SessionLimit = 0;
|
|
unsigned int AmConfig::SessionLimitErrCode = 503;
|
|
string AmConfig::SessionLimitErrReason = "Server overload";
|
|
|
|
unsigned int AmConfig::OptionsSessionLimit = 0;
|
|
unsigned int AmConfig::OptionsSessionLimitErrCode = 503;
|
|
string AmConfig::OptionsSessionLimitErrReason = "Server overload";
|
|
|
|
vector <string> AmConfig::CodecOrder;
|
|
|
|
Dtmf::InbandDetectorType
|
|
AmConfig::DefaultDTMFDetector = Dtmf::SEMSInternal;
|
|
bool AmConfig::IgnoreSIGCHLD = true;
|
|
|
|
int AmConfig::setSIPPort(const string& port)
|
|
{
|
|
if(sscanf(port.c_str(),"%u",&AmConfig::LocalSIPPort) != 1) {
|
|
return 0;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
int AmConfig::setRtpLowPort(const string& port)
|
|
{
|
|
if(sscanf(port.c_str(),"%i",&AmConfig::RtpLowPort) != 1) {
|
|
return 0;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
int AmConfig::setRtpHighPort(const string& port)
|
|
{
|
|
if(sscanf(port.c_str(),"%i",&AmConfig::RtpHighPort) != 1) {
|
|
return 0;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
int AmConfig::setLoglevel(const string& ll) {
|
|
|
|
if(sscanf(ll.c_str(),"%u",&log_level) != 1) {
|
|
return 0;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
int AmConfig::setFork(const string& fork) {
|
|
if ( strcasecmp(fork.c_str(), "yes") == 0 ) {
|
|
DaemonMode = 1;
|
|
} else if ( strcasecmp(fork.c_str(), "no") == 0 ) {
|
|
DaemonMode = 0;
|
|
} else {
|
|
return 0;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
int AmConfig::setStderr(const string& s) {
|
|
if ( strcasecmp(s.c_str(), "yes") == 0 ) {
|
|
log_stderr = 1;
|
|
AmConfig::DaemonMode = 0;
|
|
} else if ( strcasecmp(s.c_str(), "no") == 0 ) {
|
|
log_stderr = 0;
|
|
} else {
|
|
return 0;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
int AmConfig::setSessionProcessorThreads(const string& th) {
|
|
if(sscanf(th.c_str(),"%u",&SessionProcessorThreads) != 1) {
|
|
return 0;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
int AmConfig::setMediaProcessorThreads(const string& th) {
|
|
if(sscanf(th.c_str(),"%u",&MediaProcessorThreads) != 1) {
|
|
return 0;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
int AmConfig::setSIPServerThreads(const string& th){
|
|
if(sscanf(th.c_str(),"%u",&SIPServerThreads) != 1) {
|
|
return 0;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
|
|
int AmConfig::setDeadRtpTime(const string& drt)
|
|
{
|
|
if(sscanf(drt.c_str(),"%u",&AmConfig::DeadRtpTime) != 1) {
|
|
return 0;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
int AmConfig::readConfiguration()
|
|
{
|
|
DBG("Reading configuration...");
|
|
|
|
AmConfigReader cfg;
|
|
|
|
if(cfg.loadFile(ConfigurationFile.c_str())){
|
|
ERROR("while loading main configuration file\n");
|
|
return -1;
|
|
}
|
|
|
|
// take values from global configuration file
|
|
// they will be overwritten by command line args
|
|
|
|
|
|
if (cfg.hasParameter("syslog_facility")) {
|
|
set_log_facility(cfg.getParameter("syslog_facility").c_str());
|
|
}
|
|
|
|
// plugin_config_path
|
|
ModConfigPath = cfg.getParameter("plugin_config_path",ModConfigPath);
|
|
|
|
if(!ModConfigPath.empty() && (ModConfigPath[ModConfigPath.length()-1] != '/'))
|
|
ModConfigPath += '/';
|
|
|
|
// listen, sip_ip, sip_port, and media_ip
|
|
if(cfg.hasParameter("sip_ip")) {
|
|
LocalSIPIP = cfg.getParameter("sip_ip");
|
|
}
|
|
if(cfg.hasParameter("sip_port")){
|
|
if(!setSIPPort(cfg.getParameter("sip_port").c_str())){
|
|
ERROR("invalid sip port specified\n");
|
|
return -1;
|
|
}
|
|
}
|
|
if(cfg.hasParameter("media_ip")) {
|
|
LocalIP = cfg.getParameter("media_ip");
|
|
}
|
|
|
|
// public_ip
|
|
if(cfg.hasParameter("public_ip")){
|
|
string p_ip = cfg.getParameter("public_ip");
|
|
DBG("Setting public_ip parameter to %s.\n", p_ip.c_str());
|
|
PublicIP = p_ip;
|
|
}
|
|
else {
|
|
DBG("Config file has no public_ip parameter.");
|
|
}
|
|
|
|
// outbound_proxy
|
|
OutboundProxy = cfg.getParameter("outbound_proxy");
|
|
|
|
// plugin_path
|
|
PlugInPath = cfg.getParameter("plugin_path");
|
|
|
|
// load_plugins
|
|
LoadPlugins = cfg.getParameter("load_plugins");
|
|
|
|
// exclude_plugins
|
|
ExcludePlugins = cfg.getParameter("exclude_plugins");
|
|
|
|
// exclude_plugins
|
|
ExcludePayloads = cfg.getParameter("exclude_payloads");
|
|
|
|
// user_agent
|
|
if (cfg.getParameter("use_default_signature")=="yes")
|
|
Signature = DEFAULT_SIGNATURE;
|
|
else
|
|
Signature = cfg.getParameter("signature");
|
|
|
|
// log_level
|
|
if(cfg.hasParameter("loglevel")){
|
|
if(!setLoglevel(cfg.getParameter("loglevel"))){
|
|
ERROR("invalid log level specified\n");
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
LogSessions = cfg.getParameter("log_sessions")=="yes";
|
|
LogEvents = cfg.getParameter("log_events")=="yes";
|
|
|
|
if (cfg.hasParameter("unhandled_reply_loglevel")) {
|
|
string msglog = cfg.getParameter("unhandled_reply_loglevel");
|
|
if (msglog == "no") UnhandledReplyLoglevel = -1;
|
|
else if (msglog == "error") UnhandledReplyLoglevel = 0;
|
|
else if (msglog == "warn") UnhandledReplyLoglevel = 1;
|
|
else if (msglog == "info") UnhandledReplyLoglevel = 2;
|
|
else if (msglog == "debug") UnhandledReplyLoglevel = 3;
|
|
else ERROR("Could not interpret unhandled_reply_loglevel \"%s\"\n",
|
|
msglog.c_str());
|
|
}
|
|
|
|
Application = cfg.getParameter("application");
|
|
|
|
if (Application == "$(ruri.user)") {
|
|
AppSelect = App_RURIUSER;
|
|
} else if (Application == "$(ruri.param)") {
|
|
AppSelect = App_RURIPARAM;
|
|
} else if (Application == "$(apphdr)") {
|
|
AppSelect = App_APPHDR;
|
|
} else if (Application == "$(mapping)") {
|
|
AppSelect = App_MAPPING;
|
|
string appcfg_fname = ModConfigPath + "app_mapping.conf";
|
|
DBG("Loading application mapping...\n");
|
|
std::ifstream appcfg(appcfg_fname.c_str());
|
|
if (!appcfg.good()) {
|
|
ERROR("could not load application mapping file at '%s'\n",
|
|
appcfg_fname.c_str());
|
|
return -1;
|
|
}
|
|
|
|
while (!appcfg.eof()) {
|
|
string entry;
|
|
getline (appcfg,entry);
|
|
if (!entry.length() || entry[0] == '#')
|
|
continue;
|
|
vector<string> re_v = explode(entry, "=>");
|
|
if (re_v.size() != 2) {
|
|
ERROR("Incorrect line '%s' in %s: expected format 'regexp=>app_name'\n",
|
|
entry.c_str(), appcfg_fname.c_str());
|
|
return -1;
|
|
}
|
|
regex_t app_re;
|
|
if (regcomp(&app_re, re_v[0].c_str(), REG_NOSUB)) {
|
|
ERROR("compiling regex '%s' in %s.\n",
|
|
re_v[0].c_str(), appcfg_fname.c_str());
|
|
return -1;
|
|
}
|
|
DBG("adding application mapping '%s' => '%s'\n",
|
|
re_v[0].c_str(),re_v[1].c_str());
|
|
AppMapping.push_back(make_pair(app_re, re_v[1]));
|
|
}
|
|
} else {
|
|
AppSelect = App_SPECIFIED;
|
|
}
|
|
|
|
// fork
|
|
if(cfg.hasParameter("fork")){
|
|
if(!setFork(cfg.getParameter("fork"))){
|
|
ERROR("invalid fork value specified,"
|
|
" valid are only yes or no\n");
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
// stderr
|
|
if(cfg.hasParameter("stderr")){
|
|
if(!setStderr(cfg.getParameter("stderr"))){
|
|
ERROR("invalid stderr value specified,"
|
|
" valid are only yes or no\n");
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
// user_prefix_separator
|
|
PrefixSep = cfg.getParameter("user_prefix_separator",PrefixSep);
|
|
|
|
// rtp_low_port
|
|
if(cfg.hasParameter("rtp_low_port")){
|
|
if(!setRtpLowPort(cfg.getParameter("rtp_low_port"))){
|
|
ERROR("invalid rtp low port specified\n");
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
// rtp_high_port
|
|
if(cfg.hasParameter("rtp_high_port")){
|
|
if(!setRtpHighPort(cfg.getParameter("rtp_high_port"))){
|
|
ERROR("invalid rtp high port specified\n");
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
if(cfg.hasParameter("session_processor_threads")){
|
|
#ifdef SESSION_THREADPOOL
|
|
if(!setSessionProcessorThreads(cfg.getParameter("session_processor_threads"))){
|
|
ERROR("invalid session_processor_threads value specified\n");
|
|
return -1;
|
|
}
|
|
if (SessionProcessorThreads<1) {
|
|
ERROR("invalid session_processor_threads value specified."
|
|
" need at least one thread\n");
|
|
return -1;
|
|
}
|
|
#else
|
|
WARN("session_processor_threads specified in sems.conf,\n");
|
|
WARN("but SEMS is compiled without SESSION_THREADPOOL support.\n");
|
|
WARN("set USE_THREADPOOL in Makefile.defs to enable session thread pool.\n");
|
|
WARN("SEMS will start now, but every call will have its own thread.\n");
|
|
#endif
|
|
}
|
|
|
|
if(cfg.hasParameter("media_processor_threads")){
|
|
if(!setMediaProcessorThreads(cfg.getParameter("media_processor_threads"))){
|
|
ERROR("invalid media_processor_threads value specified");
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
|
|
// single codec in 200 OK
|
|
if(cfg.hasParameter("single_codec_in_ok")){
|
|
SingleCodecInOK = (cfg.getParameter("single_codec_in_ok") == "yes");
|
|
}
|
|
|
|
// single codec in 200 OK
|
|
if(cfg.hasParameter("ignore_rtpxheaders")){
|
|
IgnoreRTPXHdrs = (cfg.getParameter("ignore_rtpxheaders") == "yes");
|
|
}
|
|
|
|
// codec_order
|
|
CodecOrder = explode(cfg.getParameter("codec_order"), ",");
|
|
|
|
// dead_rtp_time
|
|
if(cfg.hasParameter("dead_rtp_time")){
|
|
if(!setDeadRtpTime(cfg.getParameter("dead_rtp_time"))){
|
|
ERROR("invalid dead_rtp_time value specified");
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
if(cfg.hasParameter("dtmf_detector")){
|
|
if (cfg.getParameter("dtmf_detector") == "spandsp") {
|
|
#ifndef USE_SPANDSP
|
|
WARN("spandsp support not compiled in.\n");
|
|
#endif
|
|
DefaultDTMFDetector = Dtmf::SpanDSP;
|
|
}
|
|
}
|
|
|
|
if(cfg.hasParameter("session_limit")){
|
|
vector<string> limit = explode(cfg.getParameter("session_limit"), ";");
|
|
if (limit.size() != 3) {
|
|
ERROR("invalid session_limit specified.\n");
|
|
} else {
|
|
if (str2i(limit[0], SessionLimit) || str2i(limit[1], SessionLimitErrCode)) {
|
|
ERROR("invalid session_limit specified.\n");
|
|
}
|
|
SessionLimitErrReason = limit[2];
|
|
}
|
|
}
|
|
|
|
if(cfg.hasParameter("options_session_limit")){
|
|
vector<string> limit = explode(cfg.getParameter("options_session_limit"), ";");
|
|
if (limit.size() != 3) {
|
|
ERROR("invalid options_session_limit specified.\n");
|
|
} else {
|
|
if (str2i(limit[0], OptionsSessionLimit) || str2i(limit[1], OptionsSessionLimitErrCode)) {
|
|
ERROR("invalid options_session_limit specified.\n");
|
|
}
|
|
OptionsSessionLimitErrReason = limit[2];
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int AmConfig::init()
|
|
{
|
|
return 0;
|
|
}
|
|
|