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.
369 lines
9.8 KiB
369 lines
9.8 KiB
/*
|
|
* Copyright (C) 2006 iptego GmbH
|
|
* Copyright (C) 2011 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. This program is released under
|
|
* the GPL with the additional exemption that compiling, linking,
|
|
* and/or using OpenSSL is allowed.
|
|
*
|
|
* 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 "AmSipRegistration.h"
|
|
#include "AmSession.h"
|
|
#include "AmSessionContainer.h"
|
|
AmSIPRegistration::AmSIPRegistration(const string& handle,
|
|
const SIPRegistrationInfo& info,
|
|
const string& sess_link)
|
|
: info(info),
|
|
dlg(this),
|
|
cred(info.domain, info.auth_user, info.pwd),
|
|
active(false),
|
|
reg_begin(0),
|
|
reg_expires(0),
|
|
remove(false),
|
|
sess_link(sess_link),
|
|
reg_send_begin(0),
|
|
waiting_result(false),
|
|
unregistering(false),
|
|
seh(NULL),
|
|
expires_interval(3600)
|
|
{
|
|
req.user = info.user;
|
|
req.method = "REGISTER";
|
|
req.r_uri = "sip:"+info.domain;
|
|
req.from = info.name+" <sip:"+info.user+"@"+info.domain+">";
|
|
req.from_uri = "sip:"+info.user+"@"+info.domain;
|
|
req.from_tag = handle;
|
|
req.to = req.from;
|
|
req.to_tag = "";
|
|
req.callid = AmSession::getNewId();
|
|
//
|
|
|
|
// clear dlg.callid? ->reregister?
|
|
dlg.initFromLocalRequest(req);
|
|
dlg.cseq = 50;
|
|
}
|
|
|
|
AmSIPRegistration::~AmSIPRegistration() {
|
|
setSessionEventHandler(NULL);
|
|
}
|
|
|
|
void AmSIPRegistration::setRegistrationInfo(const SIPRegistrationInfo& _info) {
|
|
DBG("updating registration info for '%s@%s'\n",
|
|
_info.user.c_str(), _info.domain.c_str());
|
|
info = _info;
|
|
|
|
cred.realm = info.domain;
|
|
cred.user = info.auth_user.empty() ? info.user : info.auth_user;
|
|
cred.pwd = info.pwd;
|
|
|
|
DBG("REGISTER: new credentials for registrations are: user=<%s>, pass=<%s>, realm=<%s>\n",
|
|
cred.user.c_str(), cred.pwd.c_str(), cred.realm.c_str());
|
|
|
|
req.user = info.user;
|
|
req.r_uri = "sip:"+info.domain;
|
|
req.from = info.name+" <sip:"+info.user+"@"+info.domain+">";
|
|
req.from_uri = "sip:"+info.user+"@"+info.domain;
|
|
req.to = req.from;
|
|
req.to_tag = "";
|
|
|
|
// to trigger setting dlg identifiers
|
|
dlg.setCallid(string());
|
|
|
|
dlg.initFromLocalRequest(req);
|
|
}
|
|
|
|
void AmSIPRegistration::setSessionEventHandler(AmSessionEventHandler* new_seh) {
|
|
if (seh)
|
|
delete seh;
|
|
seh = new_seh;
|
|
}
|
|
|
|
void AmSIPRegistration::setExpiresInterval(unsigned int desired_expires) {
|
|
expires_interval = desired_expires;
|
|
}
|
|
|
|
bool AmSIPRegistration::doRegistration()
|
|
{
|
|
bool res = true;
|
|
|
|
waiting_result = true;
|
|
unregistering = false;
|
|
|
|
req.to_tag = "";
|
|
req.r_uri = "sip:"+info.domain;
|
|
|
|
dlg.setRemoteTag(string());
|
|
dlg.setRemoteUri(req.r_uri);
|
|
|
|
// set outbound proxy as next hop
|
|
if (!info.proxy.empty()) {
|
|
dlg.outbound_proxy = info.proxy;
|
|
} else if (!AmConfig::OutboundProxy.empty()) {
|
|
dlg.outbound_proxy = AmConfig::OutboundProxy;
|
|
}
|
|
|
|
string hdrs = SIP_HDR_COLSP(SIP_HDR_EXPIRES) +
|
|
int2str(expires_interval) + CRLF;
|
|
|
|
int flags=0;
|
|
if(!info.contact.empty()) {
|
|
hdrs += SIP_HDR_COLSP(SIP_HDR_CONTACT) "<"
|
|
+ info.contact + ">" + CRLF;
|
|
flags = SIP_FLAGS_NOCONTACT;
|
|
}
|
|
|
|
if (dlg.sendRequest(req.method, NULL, hdrs, flags) < 0) {
|
|
ERROR("failed to send registration.\n");
|
|
res = false;
|
|
waiting_result = false;
|
|
}
|
|
|
|
// save TS
|
|
reg_send_begin = time(NULL);
|
|
return res;
|
|
}
|
|
|
|
bool AmSIPRegistration::doUnregister()
|
|
{
|
|
bool res = true;
|
|
|
|
waiting_result = true;
|
|
unregistering = true;
|
|
|
|
req.to_tag = "";
|
|
req.r_uri = "sip:"+info.domain;
|
|
dlg.setRemoteTag(string());
|
|
dlg.setRemoteUri(req.r_uri);
|
|
|
|
// set outbound proxy as next hop
|
|
if (!info.proxy.empty()) {
|
|
dlg.outbound_proxy = info.proxy;
|
|
} else if (!AmConfig::OutboundProxy.empty())
|
|
dlg.outbound_proxy = AmConfig::OutboundProxy;
|
|
|
|
int flags=0;
|
|
string hdrs = SIP_HDR_COLSP(SIP_HDR_EXPIRES) "0" CRLF;
|
|
if(!info.contact.empty()) {
|
|
hdrs = SIP_HDR_COLSP(SIP_HDR_CONTACT) "<";
|
|
hdrs += info.contact + ">" + CRLF;
|
|
flags = SIP_FLAGS_NOCONTACT;
|
|
}
|
|
|
|
if (dlg.sendRequest(req.method, NULL, hdrs, flags) < 0) {
|
|
ERROR("failed to send deregistration.\n");
|
|
res = false;
|
|
waiting_result = false;
|
|
}
|
|
|
|
// save TS
|
|
reg_send_begin = time(NULL);
|
|
return res;
|
|
}
|
|
|
|
void AmSIPRegistration::onSendRequest(AmSipRequest& req, int& flags)
|
|
{
|
|
if (seh)
|
|
seh->onSendRequest(req,flags);
|
|
}
|
|
|
|
void AmSIPRegistration::onSendReply(const AmSipRequest& req, AmSipReply& reply,
|
|
int& flags) {
|
|
if (seh)
|
|
seh->onSendReply(req,reply,flags);
|
|
}
|
|
|
|
AmSIPRegistration::RegistrationState AmSIPRegistration::getState() {
|
|
if (active)
|
|
return RegisterActive;
|
|
if (waiting_result)
|
|
return RegisterPending;
|
|
|
|
return RegisterExpired;
|
|
}
|
|
|
|
bool AmSIPRegistration::getUnregistering() {
|
|
return unregistering;
|
|
}
|
|
|
|
unsigned int AmSIPRegistration::getExpiresLeft() {
|
|
long diff = reg_begin + reg_expires - time(NULL);
|
|
if (diff < 0)
|
|
return 0;
|
|
else
|
|
return diff;
|
|
}
|
|
|
|
time_t AmSIPRegistration::getExpiresTS() {
|
|
return reg_begin + reg_expires;
|
|
}
|
|
|
|
void AmSIPRegistration::onRegisterExpired() {
|
|
if (sess_link.length()) {
|
|
AmSessionContainer::instance()->postEvent(sess_link,
|
|
new SIPRegistrationEvent(SIPRegistrationEvent::RegisterTimeout,
|
|
req.from_tag));
|
|
}
|
|
DBG("Registration '%s' expired.\n", (info.user+"@"+info.domain).c_str());
|
|
active = false;
|
|
remove = true;
|
|
}
|
|
|
|
void AmSIPRegistration::onRegisterSendTimeout() {
|
|
if (sess_link.length()) {
|
|
AmSessionContainer::instance()->
|
|
postEvent(sess_link,
|
|
new SIPRegistrationEvent(SIPRegistrationEvent::RegisterSendTimeout,
|
|
req.from_tag));
|
|
}
|
|
DBG("Registration '%s' REGISTER request timeout.\n",
|
|
(info.user+"@"+info.domain).c_str());
|
|
active = false;
|
|
remove = true;
|
|
}
|
|
|
|
bool AmSIPRegistration::registerSendTimeout(time_t now_sec) {
|
|
return now_sec > reg_send_begin + REGISTER_SEND_TIMEOUT;
|
|
}
|
|
|
|
bool AmSIPRegistration::timeToReregister(time_t now_sec) {
|
|
// if (active)
|
|
// DBG("compare %lu with %lu\n",(reg_begin+reg_expires), (unsigned long)now_sec);
|
|
return (((unsigned long)reg_begin+ reg_expires/2) < (unsigned long)now_sec);
|
|
}
|
|
|
|
bool AmSIPRegistration::registerExpired(time_t now_sec) {
|
|
return ((reg_begin+reg_expires) < now_sec);
|
|
}
|
|
|
|
void AmSIPRegistration::onSipReply(const AmSipRequest& req,
|
|
const AmSipReply& reply,
|
|
AmBasicSipDialog::Status old_dlg_status)
|
|
{
|
|
if ((seh!=NULL) && seh->onSipReply(req,reply, old_dlg_status))
|
|
return;
|
|
|
|
if (reply.code>=200)
|
|
waiting_result = false;
|
|
|
|
if ((reply.code>=200)&&(reply.code<300)) {
|
|
|
|
string contacts = reply.contact;
|
|
if (contacts.empty())
|
|
contacts = getHeader(reply.hdrs, "Contact", "m", true);
|
|
|
|
if (unregistering) {
|
|
DBG("received positive reply to De-REGISTER\n");
|
|
|
|
active = false;
|
|
remove = true;
|
|
if (!contacts.length()) {
|
|
DBG("no contacts registered any more\n");
|
|
}
|
|
if (sess_link.length()) {
|
|
AmSessionContainer::instance()->
|
|
postEvent(sess_link,
|
|
new SIPRegistrationEvent(SIPRegistrationEvent::RegisterNoContact,
|
|
req.from_tag,
|
|
reply.code, reply.reason));
|
|
}
|
|
|
|
} else {
|
|
DBG("positive reply to REGISTER!\n");
|
|
|
|
size_t end = 0;
|
|
string local_contact_hdr = info.contact.empty() ?
|
|
dlg.getContactUri() : info.contact;
|
|
local_contact.parse_contact(local_contact_hdr, (size_t)0, end);
|
|
local_contact.dump();
|
|
|
|
bool found = false;
|
|
|
|
if (!contacts.length()) {
|
|
// should not happen - positive reply without contact
|
|
DBG("no contacts registered any more\n");
|
|
active = false;
|
|
remove = true;
|
|
} else {
|
|
end = 0;
|
|
while (!found) {
|
|
if (contacts.length() == end)
|
|
break;
|
|
|
|
if (!server_contact.parse_contact(contacts, end, end)) {
|
|
ERROR("while parsing contact\n");
|
|
break;
|
|
}
|
|
server_contact.dump();
|
|
|
|
if (server_contact.isEqual(local_contact)) {
|
|
DBG("contact found\n");
|
|
found = active = true;
|
|
|
|
if (str2int(server_contact.params["expires"], reg_expires)) {
|
|
ERROR("could not extract expires value, default to 300.\n");
|
|
reg_expires = 300;
|
|
}
|
|
DBG("got an expires of %d\n", reg_expires);
|
|
// save TS
|
|
reg_begin = time(0);
|
|
|
|
if (sess_link.length()) {
|
|
DBG("posting SIPRegistrationEvent to '%s'\n", sess_link.c_str());
|
|
AmSessionContainer::instance()->
|
|
postEvent(sess_link,
|
|
new SIPRegistrationEvent(SIPRegistrationEvent::RegisterSuccess,
|
|
req.from_tag,
|
|
reply.code, reply.reason));
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
if (!found) {
|
|
if (sess_link.length()) {
|
|
AmSessionContainer::instance()->
|
|
postEvent(sess_link,
|
|
new SIPRegistrationEvent(SIPRegistrationEvent::RegisterNoContact,
|
|
req.from_tag,
|
|
reply.code, reply.reason));
|
|
}
|
|
DBG("no matching Contact - deregistered.\n");
|
|
active = false;
|
|
remove = true;
|
|
}
|
|
}
|
|
|
|
} else if (reply.code >= 300) {
|
|
DBG("Registration failed.\n");
|
|
if (sess_link.length()) {
|
|
AmSessionContainer::instance()->
|
|
postEvent(sess_link,
|
|
new SIPRegistrationEvent(SIPRegistrationEvent::RegisterFailed,
|
|
req.from_tag,
|
|
reply.code, reply.reason));
|
|
}
|
|
active = false;
|
|
remove = true;
|
|
}
|
|
}
|
|
|