#include "SipCtrlInterface.h" #include "AmUtils.h" #include "AmSipMsg.h" #include "trans_layer.h" #include "sip_parser.h" #include "parse_header.h" #include "parse_from_to.h" #include "parse_cseq.h" #include "hash_table.h" #include "sip_trans.h" #include "wheeltimer.h" #include "udp_trsp.h" #include "log.h" #include #include using std::stack; #ifndef _STANDALONE #include "AmApi.h" #include "AmConfigReader.h" #include "AmSipDispatcher.h" #ifndef MOD_NAME #define MOD_NAME "sipctrl" #endif EXPORT_CONTROL_INTERFACE_FACTORY(SipCtrlInterfaceFactory,MOD_NAME); AmCtrlInterface* SipCtrlInterfaceFactory::instance() { SipCtrlInterface* ctrl = new SipCtrlInterface(bind_addr,bind_port); trans_layer::instance()->register_ua(ctrl); return ctrl; } int SipCtrlInterfaceFactory::onLoad() { bind_addr = AmConfig::LocalSIPIP; bind_port = AmConfig::LocalSIPPort; INFO("SIP bind_addr: `%s'.\n", bind_addr.c_str()); INFO("SIP bind_port: `%i'.\n", bind_port); return 0; } #endif // #ifndef _STANDALONE SipCtrlInterface::SipCtrlInterface(const string& bind_addr, unsigned short bind_port) : bind_addr(bind_addr), bind_port(bind_port) { tl = trans_layer::instance(); } #ifndef _STANDALONE string SipCtrlInterface::getContact(const string &displayName, const string &userName, const string &hostName, const string &uriParams, const string &hdrParams) { string localUri; if (displayName.length()) { // quoting is safer (the check for quote need doesn't really pay off) if (displayName.c_str()[0] == '"') { assert(displayName.c_str()[displayName.length() - 1] == '"'); localUri += displayName; } else { localUri += '"'; localUri += displayName; localUri += '"'; } localUri += " "; } // angular brackets not always needed (unless contact) localUri += "<"; localUri += "sip:"; //TODO: sips|tel|tels if (userName.length()) { localUri += userName; localUri += "@"; } if (hostName.length()) localUri += hostName; else { localUri += AmConfig::LocalSIPIP; // Ser will replace that... localUri += ":" + AmConfig::LocalSIPPort; } if (uriParams.length()) { if (uriParams.c_str()[0] != ';') localUri += ';'; localUri += uriParams; } localUri += ">"; if (hdrParams.length()) { if (hdrParams.c_str()[0] != ';') localUri += ';'; localUri += hdrParams; } return localUri; } #endif int SipCtrlInterface::send(const AmSipRequest &req, string &serKey) { sip_msg* msg = new sip_msg(); msg->type = SIP_REQUEST; msg->u.request = new sip_request(); msg->u.request->method_str = stl2cstr(req.method); msg->u.request->ruri_str = stl2cstr(req.r_uri); // To // From // Call-ID // CSeq // Contact // Max-Forwards char* c = (char*)req.from.c_str(); int err = parse_headers(msg,&c); c = (char*)req.to.c_str(); err = err || parse_headers(msg,&c); if(err){ ERROR("Malformed To or From header\n"); delete msg; return -1; } string cseq = int2str(req.cseq) + " " + req.method; msg->cseq = new sip_header(0,"CSeq",stl2cstr(cseq)); msg->hdrs.push_back(msg->cseq); msg->callid = new sip_header(0,"Call-ID",stl2cstr(req.callid)); msg->hdrs.push_back(msg->callid); if(!req.contact.empty()){ c = (char*)req.contact.c_str(); err = parse_headers(msg,&c); if(err){ ERROR("Malformed Contact header\n"); delete msg; return -1; } } msg->hdrs.push_back(new sip_header(0,"Max-Forwards","10")); // FIXME if(!req.route.empty()){ char *c = (char*)req.route.c_str(); err = parse_headers(msg,&c); if(err){ ERROR("Route headers parsing failed\n"); ERROR("Faulty headers were: <%s>\n",req.route.c_str()); delete msg; return -1; } // // parse_headers() appends our route headers // to msg->hdrs and msg->route. But we want // only msg->route(), so we just remove the // route headers at the end of msg->hdrs. // for(sip_header* h_rr = msg->hdrs.back(); !msg->hdrs.empty(); h_rr = msg->hdrs.back()) { if(h_rr->type != sip_header::H_ROUTE){ break; } msg->hdrs.pop_back(); } } if(!req.hdrs.empty()) { char *c = (char*)req.hdrs.c_str(); err = parse_headers(msg,&c); if(err){ ERROR("Additional headers parsing failed\n"); ERROR("Faulty headers were: <%s>\n",req.hdrs.c_str()); delete msg; return -1; } } if(!req.body.empty()){ if(req.content_type.empty()){ ERROR("Body in request without content type\n"); } else { msg->content_type = new sip_header(0,"Content-Type",stl2cstr(req.content_type)); msg->hdrs.push_back(msg->content_type); msg->body = stl2cstr(req.body); } } tl->send_request(msg); delete msg; return 0; } void SipCtrlInterface::run() { INFO("Starting SIP control interface\n"); udp_trsp* udp_server = new udp_trsp(tl); assert(tl); tl->register_transport(udp_server); udp_server->bind(bind_addr,bind_port); wheeltimer::instance()->start(); udp_server->start(); udp_server->join(); } int SipCtrlInterface::send(const AmSipReply &rep) { unsigned int h=0; unsigned long t=0; if((sscanf(rep.serKey.c_str(),"%x:%lx",&h,&t) != 2) || (h >= H_TABLE_ENTRIES)){ ERROR("Invalid transaction key: invalid bucket ID\n"); return -1; } string hdrs = rep.hdrs + rep.contact; if(!rep.body.empty()) { if(rep.content_type.empty()){ ERROR("Reply does not contain a Content-Type whereby body is not empty\n"); return -1; } hdrs += "Content-Type: " + rep.content_type + "\r\n"; } return tl->send_reply(get_trans_bucket(h),(sip_trans*)t, rep.code,stl2cstr(rep.reason), stl2cstr(rep.local_tag), stl2cstr(hdrs), stl2cstr(rep.body)); } #define DBG_PARAM(p)\ DBG("%s = <%s>\n",#p,p.c_str()); void SipCtrlInterface::handleSipMsg(AmSipRequest &req) { DBG("Received new request:\n"); // DBG_PARAM(req.cmd); DBG_PARAM(req.method); // DBG_PARAM(req.user); // DBG_PARAM(req.domain); // DBG_PARAM(req.dstip); // DBG_PARAM(req.port); DBG_PARAM(req.r_uri); DBG_PARAM(req.from_uri); DBG_PARAM(req.from); DBG_PARAM(req.to); DBG_PARAM(req.callid); DBG_PARAM(req.from_tag); DBG_PARAM(req.to_tag); DBG("cseq = <%i>\n",req.cseq); DBG_PARAM(req.serKey); DBG_PARAM(req.route); DBG_PARAM(req.next_hop); DBG("hdrs = <%s>\n",req.hdrs.c_str()); DBG("body = <%s>\n",req.body.c_str()); if(req.method == "ACK") return; #ifdef _STANDALONE // Debug code - begin AmSipReply reply; reply.method = req.method; reply.code = 200; reply.reason = "OK"; reply.serKey = req.serKey; reply.local_tag = "12345"; reply.contact = "sip:" + req.dstip + ":" + req.port; int err = send(reply); if(err < 0){ DBG("send failed with err code %i\n",err); } // Debug code - end #else AmSipDispatcher::instance()->handleSipMsg(req); #endif } void SipCtrlInterface::handleSipMsg(AmSipReply &rep) { DBG("Received reply: %i %s\n",rep.code,rep.reason.c_str()); DBG_PARAM(rep.callid); DBG_PARAM(rep.local_tag); DBG_PARAM(rep.remote_tag); DBG("cseq = <%i>\n",rep.cseq); #ifndef _STANDALONE AmSipDispatcher::instance()->handleSipMsg(rep); #endif } void SipCtrlInterface::handle_sip_request(const char* tid, sip_msg* msg) { assert(msg->from && msg->from->p); assert(msg->to && msg->to->p); AmSipRequest req; req.cmd = "sems"; req.method = c2stlstr(msg->u.request->method_str); req.user = c2stlstr(msg->u.request->ruri.user); req.domain = c2stlstr(msg->u.request->ruri.host); req.dstip = get_addr_str(((sockaddr_in*)(&msg->local_ip))->sin_addr); //FIXME: IPv6 req.port = int2str(ntohs(((sockaddr_in*)(&msg->local_ip))->sin_port)); req.r_uri = c2stlstr(msg->u.request->ruri_str); if(msg->contact){ sip_nameaddr na; char* c = msg->contact->value.s; if(parse_nameaddr(&na,&c,msg->contact->value.len) < 0){ DBG("Contact parsing failed\n"); } else { req.from_uri = c2stlstr(na.addr); } } if(req.from_uri.empty()) { req.from_uri = c2stlstr(get_from(msg)->nameaddr.addr); } if(get_from(msg)->nameaddr.name.len){ req.from += c2stlstr(get_from(msg)->nameaddr.name) + ' '; } req.from += '<' + c2stlstr(get_from(msg)->nameaddr.addr) + '>'; req.to = c2stlstr(msg->to->value); req.callid = c2stlstr(msg->callid->value); req.from_tag = c2stlstr(((sip_from_to*)msg->from->p)->tag); req.to_tag = c2stlstr(((sip_from_to*)msg->to->p)->tag); req.cseq = get_cseq(msg)->num; req.body = c2stlstr(msg->body); req.serKey = tid; prepare_routes(msg->record_route, req.route); for (list::iterator it = msg->hdrs.begin(); it != msg->hdrs.end(); ++it) { if((*it)->type == sip_header::H_OTHER){ req.hdrs += c2stlstr((*it)->name) + ": " + c2stlstr((*it)->value) + "\r\n"; } } handleSipMsg(req); } void SipCtrlInterface::handle_sip_reply(sip_msg* msg) { assert(msg->from && msg->from->p); assert(msg->to && msg->to->p); AmSipReply reply; reply.content_type = msg->content_type ? c2stlstr(msg->content_type->value): ""; reply.body = msg->body.len ? c2stlstr(msg->body) : ""; reply.cseq = get_cseq(msg)->num; reply.code = msg->u.reply->code; reply.reason = c2stlstr(msg->u.reply->reason); if(msg->contact){ char* c = msg->contact->value.s; sip_nameaddr na; int err = parse_nameaddr(&na,&c,msg->contact->value.len); if(err < 0) { ERROR("Contact nameaddr parsing failed\n"); return; } // 'Contact' header? reply.next_request_uri = c2stlstr(na.addr); } reply.callid = c2stlstr(msg->callid->value); reply.remote_tag = c2stlstr(((sip_from_to*)msg->to->p)->tag); reply.local_tag = c2stlstr(((sip_from_to*)msg->from->p)->tag); // Should i fill this one with anything // i do not understand??? -> H_OTHER ? // // reply.hdrs; if( (msg->u.reply->code >= 200 ) && (msg->u.reply->code < 300 )){ tl->send_200_ack(msg); } // // Will be computed in send_request() // // reply.next_hop; prepare_routes(msg->record_route, reply.route); for (list::iterator it = msg->hdrs.begin(); it != msg->hdrs.end(); ++it) { if((*it)->type == sip_header::H_OTHER){ reply.hdrs += c2stlstr((*it)->name) + ": " + c2stlstr((*it)->value) + "\r\n"; } } handleSipMsg(reply); } void SipCtrlInterface::prepare_routes(const list& routes, string& route_field) { if(!routes.empty()){ list::const_iterator it = routes.begin(); route_field = c2stlstr((*it)->value); ++it; for(; it != routes.end(); ++it) { route_field += ", " + c2stlstr((*it)->value); } } }