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.
292 lines
7.2 KiB
292 lines
7.2 KiB
/*
|
|
* $Id$
|
|
*
|
|
* Copyright (C) 2006 iptelorg GmbH
|
|
*
|
|
* This file is part of ser, a free SIP server.
|
|
*
|
|
* ser 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
|
|
*
|
|
* ser 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
|
|
*
|
|
*/
|
|
|
|
#include "../../onsend.h"
|
|
#include "../../parser/parse_fline.h"
|
|
#include "../../ut.h"
|
|
#include "permissions.h"
|
|
#include "im_hash.h"
|
|
#include "im_db.h"
|
|
#include "im_locks.h"
|
|
#include "ipmatch.h"
|
|
|
|
/* initialize ipmatch table */
|
|
int init_ipmatch(void)
|
|
{
|
|
if (db_mode != ENABLE_CACHE) {
|
|
/* not an error, but ipmatch functions will not operate */
|
|
LOG(L_WARN, "WARNING: ipmatch_init(): Database cache is disabled, thus ipmatch functions cannot be used\n");
|
|
return 0;
|
|
}
|
|
|
|
if (init_im_hash()) {
|
|
LOG(L_ERR, "ERROR: ipmatch_init(): cannot init ipmatch hash table\n");
|
|
return -1;
|
|
}
|
|
|
|
if (reload_im_cache()) {
|
|
LOG(L_ERR, "ERROR: ipmatch_init(): error occured while caching ipmatch table\n");
|
|
return -1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/* destroy function */
|
|
void clean_ipmatch(void) {
|
|
LOG(L_DBG, "DEBUG: clean_ipmatch(): free shared memory required by ipmatch table\n");
|
|
destroy_im_hash();
|
|
}
|
|
|
|
/* process variable to filter on entry->mark */
|
|
static unsigned int IM_FILTER = ~0;
|
|
|
|
/* tries to find the given IP address and port number in the global hash table
|
|
* return value
|
|
* 1: found
|
|
* 0: not found
|
|
* -1: error
|
|
*/
|
|
static int ipmatch(struct ip_addr *ip, unsigned short port,
|
|
avp_ident_t *avp)
|
|
{
|
|
im_entry_t *entry;
|
|
int ret;
|
|
avp_value_t avp_val;
|
|
|
|
ret = 0;
|
|
|
|
if (!IM_HASH) {
|
|
LOG(L_CRIT, "ERROR: ipmatch(): ipmatch hash table is not initialied. "
|
|
"Have you set the database url?\n");
|
|
return 0;
|
|
}
|
|
|
|
/* lock hash table for reading */
|
|
reader_lock_imhash();
|
|
|
|
LOG(L_DBG, "DEBUG: ipmatch(): start searching... (ip=%s, port=%u, filter=%u)\n",
|
|
ip_addr2a(ip), port, IM_FILTER);
|
|
|
|
if (!IM_HASH->entries) {
|
|
LOG(L_DBG, "DEBUG: ipmatch(): cache is empty\n");
|
|
goto done;
|
|
}
|
|
|
|
entry = IM_HASH->entries[im_hash(ip)];
|
|
while (entry) {
|
|
if ((entry->mark & IM_FILTER)
|
|
&& ((entry->port == 0) || (port == 0) || (entry->port == port))
|
|
&& (ip_addr_cmp(&entry->ip, ip))) {
|
|
|
|
LOG(L_DBG, "DEBUG: ipmatch(): entry found\n");
|
|
|
|
/* shall we set the AVP? */
|
|
if (avp) {
|
|
/* delete AVP before inserting */
|
|
delete_avp(avp->flags, avp->name);
|
|
|
|
avp_val.s.s = entry->avp_val.s;
|
|
avp_val.s.len = entry->avp_val.len;
|
|
if (add_avp(avp->flags | AVP_VAL_STR, avp->name, avp_val)) {
|
|
LOG(L_ERR, "ERROR: ipmatch(): failed to add AVP\n");
|
|
ret = -1;
|
|
break;
|
|
}
|
|
}
|
|
|
|
ret = 1;
|
|
break;
|
|
}
|
|
|
|
entry = entry->next;
|
|
}
|
|
|
|
if (!entry) LOG(L_DBG, "DEBUG: ipmatch(): entry not found\n");
|
|
|
|
done:
|
|
/* release hash table */
|
|
reader_release_imhash();
|
|
|
|
/* reset filter */
|
|
IM_FILTER = ~0;
|
|
|
|
return ret;
|
|
}
|
|
|
|
/* wrapper function for ipmatch */
|
|
int ipmatch_2(struct sip_msg *msg, char *str1, char *str2)
|
|
{
|
|
int ret;
|
|
fparam_t *param1;
|
|
str s;
|
|
struct ip_addr *ip, ip_buf;
|
|
unsigned short port;
|
|
unsigned int iport;
|
|
|
|
param1 = (fparam_t *)str1;
|
|
|
|
switch(param1->type) {
|
|
case FPARAM_AVP:
|
|
case FPARAM_SELECT:
|
|
if (get_str_fparam(&s, msg, param1)) {
|
|
LOG(L_ERR, "ERROR: w_ipmatch_2(): could not get first parameter\n");
|
|
return -1;
|
|
}
|
|
if (parse_ip(&s, &ip_buf, &port)) {
|
|
LOG(L_ERR, "ERROR: w_ipmatch_2(): could not parse ip address\n");
|
|
return -1;
|
|
}
|
|
ip = &ip_buf;
|
|
break;
|
|
|
|
case FPARAM_STR:
|
|
if (param1->v.str.s[0] == 's') {
|
|
/* "src" */
|
|
ip = &msg->rcv.src_ip;
|
|
port = msg->rcv.src_port;
|
|
|
|
} else {
|
|
/* "via2" */
|
|
if (!msg->via2 &&
|
|
((parse_headers(msg, HDR_VIA2_F, 0) == -1) || !msg->via2)) {
|
|
|
|
LOG(L_ERR, "ERROR: w_ipmatch_2(): could not get 2nd VIA HF\n");
|
|
return -1;
|
|
}
|
|
if (!msg->via2->received || !msg->via2->received->value.s) {
|
|
LOG(L_ERR, "ERROR: w_ipmatch_2(): there is no received param in the 2nd VIA HF\n");
|
|
return -1;
|
|
}
|
|
if (parse_ip(&msg->via2->received->value, &ip_buf, &port)) {
|
|
LOG(L_ERR, "ERROR: w_ipmatch_2(): could not parse ip address\n");
|
|
return -1;
|
|
}
|
|
ip = &ip_buf;
|
|
if (!msg->via2->rport || !msg->via2->rport->value.s) {
|
|
LOG(L_WARN, "WARNING: w_ipmatch_2(): there is no rport param in the 2nd VIA HF\n");
|
|
port = 0;
|
|
} else {
|
|
if (str2int(&msg->via2->rport->value, &iport)) {
|
|
LOG(L_ERR, "ERROR: w_ipmatch_2(): invalid port number %.*s\n",
|
|
msg->via2->rport->value.len, msg->via2->rport->value.s);
|
|
return -1;
|
|
}
|
|
port = iport;
|
|
}
|
|
}
|
|
break;
|
|
|
|
default:
|
|
LOG(L_ERR, "ERROR: w_ipmatch_2(): unknown parameter type\n");
|
|
return -1;
|
|
}
|
|
|
|
ret = ipmatch(
|
|
ip,
|
|
port,
|
|
(str2) ? &((fparam_t *)str2)->v.avp : 0
|
|
);
|
|
|
|
|
|
return (ret == 0) ? -1 : 1;
|
|
}
|
|
|
|
/* wrapper function for ipmatch */
|
|
int ipmatch_1(struct sip_msg *msg, char *str1, char *str2)
|
|
{
|
|
return ipmatch_2(msg, str1, 0);
|
|
}
|
|
|
|
/* wrapper function for ipmatch */
|
|
int ipmatch_onsend(struct sip_msg *msg, char *str1, char *str2)
|
|
{
|
|
int ret;
|
|
struct ip_addr ip;
|
|
unsigned short port;
|
|
char *buf, *ch1, *ch2;
|
|
struct msg_start fl;
|
|
str *uri, s;
|
|
|
|
if (str1[0] == 'd') {
|
|
/* get info from destination address */
|
|
port = su_getport(get_onsend_info()->to);
|
|
su2ip_addr(&ip, get_onsend_info()->to);
|
|
|
|
} else {
|
|
/* get info from Request URI
|
|
we need to parse the first line again because the parsed uri can be
|
|
changed by another branch */
|
|
|
|
/* use another buffer pointer, because parse_first_list() modifies it */
|
|
buf = get_onsend_info()->buf;
|
|
parse_first_line(buf, get_onsend_info()->len, &fl);
|
|
if (fl.type != SIP_REQUEST) {
|
|
LOG(L_ERR, "ERROR: w_ipmatch_onsend(): message type is not request\n");
|
|
return -1;
|
|
}
|
|
uri = &(fl.u.request.uri);
|
|
|
|
/* find the host:port part in the uri */
|
|
if ((!(ch1 = memchr(uri->s, '@', uri->len))) && (!(ch1 = memchr(uri->s, ':', uri->len)))) {
|
|
LOG(L_ERR, "ERROR: w_ipmatch_onsend(): unable to get host:port part of uri: %.*s\n",
|
|
uri->len, uri->s);
|
|
return -1;
|
|
}
|
|
/* is there a parameter in the uri? */
|
|
if ((ch2 = memchr(uri->s, ';', uri->len))) {
|
|
s.s = ch1 + 1;
|
|
s.len = ch2 - ch1 - 1;
|
|
} else {
|
|
s.s = ch1 + 1;
|
|
s.len = uri->len - (ch1 - uri->s) - 1;
|
|
}
|
|
if (parse_ip(&s, &ip, &port)) {
|
|
LOG(L_ERR, "ERROR: w_ipmatch_onsend(): could not parse ip address\n");
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
ret = ipmatch(
|
|
&ip,
|
|
port,
|
|
0 /* AVP operations are unsafe in onsend_route! */
|
|
);
|
|
|
|
return (ret == 0) ? -1 : 1;
|
|
}
|
|
|
|
/* set IM_FILTER */
|
|
int ipmatch_filter(struct sip_msg *msg, char *str1, char *str2)
|
|
{
|
|
int i;
|
|
|
|
if (get_int_fparam(&i, msg, (fparam_t *)str1)) return -1;
|
|
IM_FILTER = (unsigned int)i;
|
|
return 1;
|
|
}
|