mirror of https://github.com/sipwise/kamailio.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.
833 lines
18 KiB
833 lines
18 KiB
/*
|
|
* Copyright (C) 2001-2003 FhG Fokus
|
|
*
|
|
* This file is part of Kamailio, a free SIP server.
|
|
*
|
|
* Kamailio 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
|
|
*
|
|
* Kamailio 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*
|
|
*/
|
|
|
|
/*! \file
|
|
* \brief USRLOC - Usrloc record structure
|
|
* \ingroup usrloc
|
|
*
|
|
* - Module \ref usrloc
|
|
*/
|
|
|
|
|
|
#include "urecord.h"
|
|
#include <string.h>
|
|
#include "../../mem/shm_mem.h"
|
|
#include "../../dprint.h"
|
|
#include "../../ut.h"
|
|
#include "../../hashes.h"
|
|
#include "../../tcp_conn.h"
|
|
#include "../../pass_fd.h"
|
|
#include "ul_mod.h"
|
|
#include "usrloc.h"
|
|
#include "utime.h"
|
|
#include "ul_callback.h"
|
|
#include "usrloc.h"
|
|
|
|
/*! contact matching mode */
|
|
int matching_mode = CONTACT_ONLY;
|
|
/*! retransmission detection interval in seconds */
|
|
int cseq_delay = 20;
|
|
|
|
/*!
|
|
* \brief Create and initialize new record structure
|
|
* \param _dom domain name
|
|
* \param _aor address of record
|
|
* \param _r pointer to the new record
|
|
* \return 0 on success, negative on failure
|
|
*/
|
|
int new_urecord(str* _dom, str* _aor, urecord_t** _r)
|
|
{
|
|
*_r = (urecord_t*)shm_malloc(sizeof(urecord_t));
|
|
if (*_r == 0) {
|
|
LM_ERR("no more share memory\n");
|
|
return -1;
|
|
}
|
|
memset(*_r, 0, sizeof(urecord_t));
|
|
|
|
(*_r)->aor.s = (char*)shm_malloc(_aor->len);
|
|
if ((*_r)->aor.s == 0) {
|
|
LM_ERR("no more share memory\n");
|
|
shm_free(*_r);
|
|
*_r = 0;
|
|
return -2;
|
|
}
|
|
memcpy((*_r)->aor.s, _aor->s, _aor->len);
|
|
(*_r)->aor.len = _aor->len;
|
|
(*_r)->domain = _dom;
|
|
(*_r)->aorhash = ul_get_aorhash(_aor);
|
|
return 0;
|
|
}
|
|
|
|
|
|
/*!
|
|
* \brief Free all memory used by the given structure
|
|
*
|
|
* Free all memory used by the given structure.
|
|
* The structure must be removed from all linked
|
|
* lists first
|
|
* \param _r freed record list
|
|
*/
|
|
void free_urecord(urecord_t* _r)
|
|
{
|
|
ucontact_t* ptr;
|
|
|
|
while(_r->contacts) {
|
|
ptr = _r->contacts;
|
|
_r->contacts = _r->contacts->next;
|
|
free_ucontact(ptr);
|
|
}
|
|
|
|
/* if mem cache is not used, the urecord struct is static*/
|
|
if (db_mode!=DB_ONLY) {
|
|
if (_r->aor.s) shm_free(_r->aor.s);
|
|
shm_free(_r);
|
|
}
|
|
}
|
|
|
|
|
|
/*!
|
|
* \brief Print a record, useful for debugging
|
|
* \param _f print output
|
|
* \param _r printed record
|
|
*/
|
|
void print_urecord(FILE* _f, urecord_t* _r)
|
|
{
|
|
ucontact_t* ptr;
|
|
|
|
fprintf(_f, "...Record(%p)...\n", _r);
|
|
fprintf(_f, "domain : '%.*s'\n", _r->domain->len, ZSW(_r->domain->s));
|
|
fprintf(_f, "aor : '%.*s'\n", _r->aor.len, ZSW(_r->aor.s));
|
|
fprintf(_f, "aorhash: '%u'\n", (unsigned)_r->aorhash);
|
|
fprintf(_f, "slot: '%d'\n", _r->aorhash&(_r->slot->d->size-1));
|
|
|
|
if (_r->contacts) {
|
|
ptr = _r->contacts;
|
|
while(ptr) {
|
|
print_ucontact(_f, ptr);
|
|
ptr = ptr->next;
|
|
}
|
|
}
|
|
|
|
fprintf(_f, ".../Record...\n");
|
|
}
|
|
|
|
|
|
/*!
|
|
* \brief Add a new contact in memory
|
|
*
|
|
* Add a new contact in memory, contacts are ordered by:
|
|
* 1) q value, 2) descending modification time
|
|
* \param _r record this contact belongs to
|
|
* \param _c contact
|
|
* \param _ci contact information
|
|
* \return pointer to new created contact on success, 0 on failure
|
|
*/
|
|
ucontact_t* mem_insert_ucontact(urecord_t* _r, str* _c, ucontact_info_t* _ci)
|
|
{
|
|
ucontact_t* ptr, *prev = 0;
|
|
ucontact_t* c;
|
|
|
|
if ( (c=new_ucontact(_r->domain, &_r->aor, _c, _ci)) == 0) {
|
|
LM_ERR("failed to create new contact\n");
|
|
return 0;
|
|
}
|
|
if_update_stat( _r->slot, _r->slot->d->contacts, 1);
|
|
|
|
ptr = _r->contacts;
|
|
|
|
if (!desc_time_order) {
|
|
while(ptr) {
|
|
if (ptr->q < c->q) break;
|
|
prev = ptr;
|
|
ptr = ptr->next;
|
|
}
|
|
}
|
|
|
|
if (ptr) {
|
|
if (!ptr->prev) {
|
|
ptr->prev = c;
|
|
c->next = ptr;
|
|
_r->contacts = c;
|
|
} else {
|
|
c->next = ptr;
|
|
c->prev = ptr->prev;
|
|
ptr->prev->next = c;
|
|
ptr->prev = c;
|
|
}
|
|
} else if (prev) {
|
|
prev->next = c;
|
|
c->prev = prev;
|
|
} else {
|
|
_r->contacts = c;
|
|
}
|
|
|
|
return c;
|
|
}
|
|
|
|
|
|
/*!
|
|
* \brief Remove the contact from lists in memory
|
|
* \param _r record this contact belongs to
|
|
* \param _c removed contact
|
|
*/
|
|
void mem_remove_ucontact(urecord_t* _r, ucontact_t* _c)
|
|
{
|
|
if (_c->prev) {
|
|
_c->prev->next = _c->next;
|
|
if (_c->next) {
|
|
_c->next->prev = _c->prev;
|
|
}
|
|
} else {
|
|
_r->contacts = _c->next;
|
|
if (_c->next) {
|
|
_c->next->prev = 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/*!
|
|
* \brief Remove contact in memory from the list and delete it
|
|
* \param _r record this contact belongs to
|
|
* \param _c deleted contact
|
|
*/
|
|
void mem_delete_ucontact(urecord_t* _r, ucontact_t* _c)
|
|
{
|
|
mem_remove_ucontact(_r, _c);
|
|
if_update_stat( _r->slot, _r->slot->d->contacts, -1);
|
|
free_ucontact(_c);
|
|
}
|
|
|
|
static inline int is_valid_tcpconn(ucontact_t *c)
|
|
{
|
|
if (c->tcpconn_id == -1)
|
|
return 0; /* tcpconn_id is not present */
|
|
else
|
|
return 1; /* valid tcpconn_id */
|
|
}
|
|
|
|
static inline int is_tcp_alive(ucontact_t *c)
|
|
{
|
|
struct tcp_connection *con = NULL;
|
|
int rc = 0;
|
|
|
|
if ((con = tcpconn_get(c->tcpconn_id, 0, 0, 0, 0))) {
|
|
tcpconn_put(con); /* refcnt-- */
|
|
rc = 1;
|
|
}
|
|
|
|
return rc;
|
|
}
|
|
|
|
/*!
|
|
* \brief Close a TCP connection
|
|
*
|
|
* Requests the TCP main process to close the specified TCP connection
|
|
* \param conid the internal connection ID
|
|
*/
|
|
static inline int close_connection(int conid) {
|
|
struct tcp_connection *con;
|
|
long msg[2];
|
|
int n;
|
|
if ((con = tcpconn_get(conid, 0, 0, 0, 0))) {
|
|
msg[0] = (long)con;
|
|
msg[1] = CONN_EOF;
|
|
|
|
con->send_flags.f |= SND_F_CON_CLOSE;
|
|
con->flags |= F_CONN_FORCE_EOF;
|
|
|
|
n = send_all(unix_tcp_sock, msg, sizeof(msg));
|
|
if (unlikely(n <= 0)){
|
|
LM_ERR("failed to send close request: %s (%d)\n", strerror(errno), errno);
|
|
return 0;
|
|
}
|
|
return 1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/*!
|
|
* \brief Expires timer for NO_DB db_mode
|
|
*
|
|
* Expires timer for NO_DB db_mode, process all contacts from
|
|
* the record, delete the expired ones from memory.
|
|
* \param _r processed record
|
|
*/
|
|
static inline void nodb_timer(urecord_t* _r)
|
|
{
|
|
ucontact_t* ptr, *t;
|
|
|
|
|
|
ptr = _r->contacts;
|
|
|
|
while(ptr) {
|
|
if (handle_lost_tcp && is_valid_tcpconn(ptr) && !is_tcp_alive(ptr)) {
|
|
LM_DBG("tcp connection has been lost, expiring contact %.*s\n", ptr->c.len, ptr->c.s);
|
|
ptr->expires = UL_EXPIRED_TIME;
|
|
}
|
|
|
|
if (!VALID_CONTACT(ptr, act_time)) {
|
|
/* run callbacks for EXPIRE event */
|
|
if (exists_ulcb_type(UL_CONTACT_EXPIRE))
|
|
run_ul_callbacks( UL_CONTACT_EXPIRE, ptr);
|
|
|
|
LM_DBG("Binding '%.*s','%.*s' has expired\n",
|
|
ptr->aor->len, ZSW(ptr->aor->s),
|
|
ptr->c.len, ZSW(ptr->c.s));
|
|
|
|
if (close_expired_tcp && is_valid_tcpconn(ptr)) {
|
|
close_connection(ptr->tcpconn_id);
|
|
}
|
|
|
|
t = ptr;
|
|
ptr = ptr->next;
|
|
|
|
mem_delete_ucontact(_r, t);
|
|
update_stat( _r->slot->d->expires, 1);
|
|
} else {
|
|
ptr = ptr->next;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/*!
|
|
* \brief Write through timer, used for WRITE_THROUGH db_mode
|
|
*
|
|
* Write through timer, used for WRITE_THROUGH db_mode. Process all
|
|
* contacts from the record, delete all expired ones from the DB.
|
|
* \param _r processed record
|
|
* \note currently unused, this mode is also handled by the wb_timer
|
|
*/
|
|
static inline void wt_timer(urecord_t* _r)
|
|
{
|
|
ucontact_t* ptr, *t;
|
|
|
|
ptr = _r->contacts;
|
|
|
|
while(ptr) {
|
|
if (!VALID_CONTACT(ptr, act_time)) {
|
|
/* run callbacks for EXPIRE event */
|
|
if (exists_ulcb_type(UL_CONTACT_EXPIRE)) {
|
|
run_ul_callbacks( UL_CONTACT_EXPIRE, ptr);
|
|
}
|
|
|
|
LM_DBG("Binding '%.*s','%.*s' has expired\n",
|
|
ptr->aor->len, ZSW(ptr->aor->s),
|
|
ptr->c.len, ZSW(ptr->c.s));
|
|
|
|
if (close_expired_tcp && is_valid_tcpconn(ptr)) {
|
|
close_connection(ptr->tcpconn_id);
|
|
}
|
|
|
|
t = ptr;
|
|
ptr = ptr->next;
|
|
|
|
if (db_delete_ucontact(t) < 0) {
|
|
LM_ERR("deleting contact from database failed\n");
|
|
}
|
|
mem_delete_ucontact(_r, t);
|
|
update_stat( _r->slot->d->expires, 1);
|
|
} else {
|
|
ptr = ptr->next;
|
|
}
|
|
}
|
|
}
|
|
|
|
/*!
|
|
* \brief Write-back timer, used for WRITE_BACK db_mode
|
|
*
|
|
* Write-back timer, used for WRITE_BACK db_mode. Process
|
|
* all contacts from the record, delete expired ones from the DB.
|
|
* Furthermore it updates changed contacts, and also insert new
|
|
* ones in the DB.
|
|
* \param _r processed record
|
|
*/
|
|
static inline void wb_timer(urecord_t* _r)
|
|
{
|
|
ucontact_t* ptr, *t;
|
|
cstate_t old_state;
|
|
int op;
|
|
int res;
|
|
|
|
ptr = _r->contacts;
|
|
|
|
while(ptr) {
|
|
if (handle_lost_tcp && is_valid_tcpconn(ptr) && !is_tcp_alive(ptr)) {
|
|
LM_DBG("tcp connection has been lost, expiring contact %.*s\n", ptr->c.len, ptr->c.s);
|
|
ptr->expires = UL_EXPIRED_TIME;
|
|
}
|
|
|
|
if (!VALID_CONTACT(ptr, act_time)) {
|
|
/* run callbacks for EXPIRE event */
|
|
if (exists_ulcb_type(UL_CONTACT_EXPIRE)) {
|
|
run_ul_callbacks( UL_CONTACT_EXPIRE, ptr);
|
|
}
|
|
|
|
LM_DBG("Binding '%.*s','%.*s' has expired\n",
|
|
ptr->aor->len, ZSW(ptr->aor->s),
|
|
ptr->c.len, ZSW(ptr->c.s));
|
|
update_stat( _r->slot->d->expires, 1);
|
|
|
|
if (close_expired_tcp && is_valid_tcpconn(ptr)) {
|
|
close_connection(ptr->tcpconn_id);
|
|
}
|
|
|
|
t = ptr;
|
|
ptr = ptr->next;
|
|
|
|
/* Should we remove the contact from the database ? */
|
|
if (st_expired_ucontact(t) == 1) {
|
|
if (db_delete_ucontact(t) < 0) {
|
|
LM_ERR("failed to delete contact from the database"
|
|
" (aor: %.*s)\n",
|
|
t->aor->len, ZSW(t->aor->s));
|
|
}
|
|
}
|
|
|
|
mem_delete_ucontact(_r, t);
|
|
} else {
|
|
/* Determine the operation we have to do */
|
|
old_state = ptr->state;
|
|
op = st_flush_ucontact(ptr);
|
|
|
|
switch(op) {
|
|
case 0: /* do nothing, contact is synchronized */
|
|
break;
|
|
|
|
case 1: /* insert */
|
|
if (db_insert_ucontact(ptr) < 0) {
|
|
LM_ERR("inserting contact into database failed"
|
|
" (aor: %.*s)\n",
|
|
ptr->aor->len, ZSW(ptr->aor->s));
|
|
ptr->state = old_state;
|
|
}
|
|
break;
|
|
|
|
case 2: /* update */
|
|
if (ul_db_update_as_insert)
|
|
res = db_insert_ucontact(ptr);
|
|
else
|
|
res = db_update_ucontact(ptr);
|
|
if (res < 0) {
|
|
LM_ERR("updating contact in db failed (aor: %.*s)\n",
|
|
ptr->aor->len, ZSW(ptr->aor->s));
|
|
ptr->state = old_state;
|
|
}
|
|
break;
|
|
}
|
|
|
|
ptr = ptr->next;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/*!
|
|
* \brief Run timer functions depending on the db_mode setting.
|
|
*
|
|
* Helper function that run the appropriate timer function, depending
|
|
* on the db_mode setting.
|
|
* \param _r processed record
|
|
*/
|
|
void timer_urecord(urecord_t* _r)
|
|
{
|
|
switch(db_mode) {
|
|
case DB_READONLY:
|
|
case NO_DB: nodb_timer(_r);
|
|
break;
|
|
/* use also the write_back timer routine to handle the failed
|
|
* realtime inserts/updates */
|
|
case WRITE_THROUGH: wb_timer(_r); /*wt_timer(_r);*/
|
|
break;
|
|
case WRITE_BACK: wb_timer(_r);
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
/*!
|
|
* \brief Delete a record from the database
|
|
* \param _r deleted record
|
|
* \return 0 on success, -1 on failure
|
|
*/
|
|
int db_delete_urecord(urecord_t* _r)
|
|
{
|
|
db_key_t keys[2];
|
|
db_val_t vals[2];
|
|
char* dom;
|
|
|
|
keys[0] = &user_col;
|
|
keys[1] = &domain_col;
|
|
vals[0].type = DB1_STR;
|
|
vals[0].nul = 0;
|
|
vals[0].val.str_val.s = _r->aor.s;
|
|
vals[0].val.str_val.len = _r->aor.len;
|
|
|
|
if (use_domain) {
|
|
dom = memchr(_r->aor.s, '@', _r->aor.len);
|
|
vals[0].val.str_val.len = dom - _r->aor.s;
|
|
|
|
vals[1].type = DB1_STR;
|
|
vals[1].nul = 0;
|
|
vals[1].val.str_val.s = dom + 1;
|
|
vals[1].val.str_val.len = _r->aor.s + _r->aor.len - dom - 1;
|
|
}
|
|
|
|
if (ul_dbf.use_table(ul_dbh, _r->domain) < 0) {
|
|
LM_ERR("use_table failed\n");
|
|
return -1;
|
|
}
|
|
|
|
if (ul_dbf.delete(ul_dbh, keys, 0, vals, (use_domain) ? (2) : (1)) < 0) {
|
|
LM_ERR("failed to delete from database\n");
|
|
return -1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/*!
|
|
* \brief Delete a record from the database based on ruid
|
|
* \return 0 on success, -1 on failure
|
|
*/
|
|
int db_delete_urecord_by_ruid(str *_table, str *_ruid)
|
|
{
|
|
db_key_t keys[1];
|
|
db_val_t vals[1];
|
|
|
|
keys[0] = &ruid_col;
|
|
vals[0].type = DB1_STR;
|
|
vals[0].nul = 0;
|
|
vals[0].val.str_val.s = _ruid->s;
|
|
vals[0].val.str_val.len = _ruid->len;
|
|
|
|
if (ul_dbf.use_table(ul_dbh, _table) < 0) {
|
|
LM_ERR("use_table failed\n");
|
|
return -1;
|
|
}
|
|
|
|
if (ul_dbf.delete(ul_dbh, keys, 0, vals, 1) < 0) {
|
|
LM_ERR("failed to delete from database\n");
|
|
return -1;
|
|
}
|
|
|
|
if (ul_dbf.affected_rows(ul_dbh) == 0) {
|
|
return -2;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/*!
|
|
* \brief Release urecord previously obtained through get_urecord
|
|
* \warning Failing to calls this function after get_urecord will
|
|
* result in a memory leak when the DB_ONLY mode is used. When
|
|
* the records is later deleted, e.g. with delete_urecord, then
|
|
* its not necessary, as this function already releases the record.
|
|
* \param _r released record
|
|
*/
|
|
void release_urecord(urecord_t* _r)
|
|
{
|
|
if (db_mode==DB_ONLY) {
|
|
free_urecord(_r);
|
|
} else if (_r->contacts == 0) {
|
|
mem_delete_urecord(_r->slot->d, _r);
|
|
}
|
|
}
|
|
|
|
|
|
/*!
|
|
* \brief Create and insert new contact into urecord
|
|
* \param _r record into the new contact should be inserted
|
|
* \param _contact contact string
|
|
* \param _ci contact information
|
|
* \param _c new created contact
|
|
* \return 0 on success, -1 on failure
|
|
*/
|
|
int insert_ucontact(urecord_t* _r, str* _contact, ucontact_info_t* _ci,
|
|
ucontact_t** _c)
|
|
{
|
|
if ( ((*_c)=mem_insert_ucontact(_r, _contact, _ci)) == 0) {
|
|
LM_ERR("failed to insert contact\n");
|
|
return -1;
|
|
}
|
|
|
|
if (db_mode==DB_ONLY) {
|
|
if (db_insert_ucontact(*_c) < 0) {
|
|
LM_ERR("failed to insert in database\n");
|
|
return -1;
|
|
} else {
|
|
(*_c)->state = CS_SYNC;
|
|
}
|
|
}
|
|
|
|
if (exists_ulcb_type(UL_CONTACT_INSERT)) {
|
|
run_ul_callbacks( UL_CONTACT_INSERT, *_c);
|
|
}
|
|
|
|
if (db_mode == WRITE_THROUGH) {
|
|
if (db_insert_ucontact(*_c) < 0) {
|
|
LM_ERR("failed to insert in database\n");
|
|
return -1;
|
|
} else {
|
|
(*_c)->state = CS_SYNC;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/*!
|
|
* \brief Delete ucontact from urecord
|
|
* \param _r record where the contact belongs to
|
|
* \param _c deleted contact
|
|
* \return 0 on success, -1 on failure
|
|
*/
|
|
int delete_ucontact(urecord_t* _r, struct ucontact* _c)
|
|
{
|
|
int ret = 0;
|
|
|
|
if (exists_ulcb_type(UL_CONTACT_DELETE)) {
|
|
run_ul_callbacks( UL_CONTACT_DELETE, _c);
|
|
}
|
|
|
|
if (st_delete_ucontact(_c) > 0) {
|
|
if (db_mode == WRITE_THROUGH || db_mode==DB_ONLY) {
|
|
if (db_delete_ucontact(_c) < 0) {
|
|
LM_ERR("failed to remove contact from database\n");
|
|
ret = -1;
|
|
}
|
|
}
|
|
|
|
mem_delete_ucontact(_r, _c);
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
|
|
int delete_urecord_by_ruid(udomain_t* _d, str *_ruid)
|
|
{
|
|
if (db_mode != DB_ONLY) {
|
|
LM_ERR("delete_urecord_by_ruid currently available only in db_mode=3\n");
|
|
return -1;
|
|
}
|
|
|
|
return db_delete_urecord_by_ruid(_d->name, _ruid);
|
|
}
|
|
|
|
|
|
/*!
|
|
* \brief Match a contact record to a contact string
|
|
* \param ptr contact record
|
|
* \param _c contact string
|
|
* \return ptr on successfull match, 0 when they not match
|
|
*/
|
|
static inline struct ucontact* contact_match( ucontact_t* ptr, str* _c)
|
|
{
|
|
while(ptr) {
|
|
if ((_c->len == ptr->c.len) && !memcmp(_c->s, ptr->c.s, _c->len)) {
|
|
return ptr;
|
|
}
|
|
|
|
ptr = ptr->next;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
|
|
/*!
|
|
* \brief Match a contact record to a contact string and callid
|
|
* \param ptr contact record
|
|
* \param _c contact string
|
|
* \param _callid callid
|
|
* \return ptr on successfull match, 0 when they not match
|
|
*/
|
|
static inline struct ucontact* contact_callid_match( ucontact_t* ptr,
|
|
str* _c, str *_callid)
|
|
{
|
|
while(ptr) {
|
|
if ( (_c->len==ptr->c.len) && (_callid->len==ptr->callid.len)
|
|
&& !memcmp(_c->s, ptr->c.s, _c->len)
|
|
&& !memcmp(_callid->s, ptr->callid.s, _callid->len)
|
|
) {
|
|
return ptr;
|
|
}
|
|
|
|
ptr = ptr->next;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/*!
|
|
+ * \brief Match a contact record to a contact string and path
|
|
+ * \param ptr contact record
|
|
+ * \param _c contact string
|
|
+ * \param _path path
|
|
+ * \return ptr on successfull match, 0 when they not match
|
|
+ */
|
|
static inline struct ucontact* contact_path_match( ucontact_t* ptr, str* _c, str *_path)
|
|
{
|
|
/* if no path is preset (in REGISTER request) or use_path is not configured
|
|
in registrar module, default to contact_match() */
|
|
if( _path == NULL) return contact_match(ptr, _c);
|
|
|
|
while(ptr) {
|
|
if ( (_c->len==ptr->c.len) && (_path->len==ptr->path.len)
|
|
&& !memcmp(_c->s, ptr->c.s, _c->len)
|
|
&& !memcmp(_path->s, ptr->path.s, _path->len)
|
|
) {
|
|
return ptr;
|
|
}
|
|
|
|
ptr = ptr->next;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
|
|
|
|
/*!
|
|
* \brief Match a contact record to a Call-ID only
|
|
* \param ptr contact record
|
|
* \param _c contact string
|
|
* \return ptr on successfull match, 0 when they not match
|
|
*/
|
|
static inline struct ucontact* contact_match_callidonly( ucontact_t* ptr, str* _callid)
|
|
{
|
|
while(ptr) {
|
|
if ((_callid->len == ptr->callid.len) && !memcmp(_callid->s, ptr->callid.s, _callid->len)) {
|
|
return ptr;
|
|
}
|
|
|
|
ptr = ptr->next;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
|
|
/*!
|
|
* \brief Get pointer to ucontact with given contact
|
|
* \param _r record where to search the contacts
|
|
* \param _c contact string
|
|
* \param _callid callid
|
|
* \param _path path
|
|
* \param _cseq CSEQ number
|
|
* \param _co found contact
|
|
* \return 0 - found, 1 - not found, -1 - invalid found,
|
|
* -2 - found, but to be skipped (same cseq)
|
|
*/
|
|
int get_ucontact(urecord_t* _r, str* _c, str* _callid, str* _path, int _cseq,
|
|
struct ucontact** _co)
|
|
{
|
|
ucontact_t* ptr;
|
|
int no_callid;
|
|
|
|
ptr = 0;
|
|
no_callid = 0;
|
|
*_co = 0;
|
|
|
|
switch (matching_mode) {
|
|
case CONTACT_ONLY:
|
|
ptr = contact_match( _r->contacts, _c);
|
|
break;
|
|
case CONTACT_CALLID:
|
|
ptr = contact_callid_match( _r->contacts, _c, _callid);
|
|
no_callid = 1;
|
|
break;
|
|
case CONTACT_PATH:
|
|
ptr = contact_path_match( _r->contacts, _c, _path);
|
|
break;
|
|
case CONTACT_CALLID_ONLY:
|
|
ptr = contact_match_callidonly( _r->contacts, _callid);
|
|
break;
|
|
default:
|
|
LM_CRIT("unknown matching_mode %d\n", matching_mode);
|
|
return -1;
|
|
}
|
|
|
|
if (ptr) {
|
|
/* found -> check callid and cseq */
|
|
if ( no_callid || (ptr->callid.len==_callid->len
|
|
&& memcmp(_callid->s, ptr->callid.s, _callid->len)==0 ) ) {
|
|
if (_cseq<ptr->cseq)
|
|
return -1;
|
|
if (_cseq==ptr->cseq) {
|
|
get_act_time();
|
|
return (ptr->last_modified+cseq_delay>act_time)?-2:-1;
|
|
}
|
|
}
|
|
*_co = ptr;
|
|
return 0;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* Get pointer to ucontact with given info (by address or sip.instance)
|
|
*/
|
|
int get_ucontact_by_instance(urecord_t* _r, str* _c, ucontact_info_t* _ci,
|
|
ucontact_t** _co)
|
|
{
|
|
ucontact_t* ptr;
|
|
str i1;
|
|
str i2;
|
|
|
|
if (_ci->instance.s == NULL || _ci->instance.len <= 0) {
|
|
return get_ucontact(_r, _c, _ci->callid, _ci->path, _ci->cseq, _co);
|
|
}
|
|
|
|
/* find by instance */
|
|
ptr = _r->contacts;
|
|
while(ptr) {
|
|
if (ptr->instance.len>0 && _ci->reg_id==ptr->reg_id)
|
|
{
|
|
i1 = _ci->instance;
|
|
i2 = ptr->instance;
|
|
if(i1.s[0]=='<' && i1.s[i1.len-1]=='>') {
|
|
i1.s++;
|
|
i1.len-=2;
|
|
}
|
|
if(i2.s[0]=='<' && i2.s[i2.len-1]=='>') {
|
|
i2.s++;
|
|
i2.len-=2;
|
|
}
|
|
if(i1.len==i2.len && memcmp(i1.s, i2.s, i2.len)==0) {
|
|
*_co = ptr;
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
ptr = ptr->next;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
unsigned int ul_get_aorhash(str *_aor)
|
|
{
|
|
return core_hash(_aor, 0, 0);
|
|
}
|