2 * Copyright (C) 2008 Daniel-Constantin Mierla (asipto.com)
4 * This file is part of Kamailio, a free SIP server.
6 * Kamailio is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation; either version 2 of the License, or
9 * (at your option) any later version
11 * Kamailio is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
22 #include "../../core/parser/parse_uri.h"
23 #include "../../core/dset.h"
24 #include "../../core/onsend.h"
25 #include "../../core/socket_info.h"
28 #include "pv_branch.h"
30 static branch_t _pv_sbranch;
32 void pv_init_sbranch(void)
34 memset(&_pv_sbranch, 0, sizeof(branch_t));
37 int pv_get_branchx_helper(sip_msg_t *msg, pv_param_t *param,
38 pv_value_t *res, int btype)
48 if(pv_get_spec_index(msg, param, &idx, &idxf)!=0)
50 LM_ERR("invalid index\n");
51 return pv_get_null(msg, param, res);
53 br = get_sip_branch(idx);
55 return pv_get_null(msg, param, res);
59 /* branch(count) doesn't need a valid branch, everything else does */
60 if(br->len == 0 && ( param->pvn.u.isname.name.n != 5/* count*/ ))
62 LM_ERR("error accessing branch [%d]\n", idx);
63 return pv_get_null(msg, param, res);
66 switch(param->pvn.u.isname.name.n)
69 if(br->dst_uri_len==0)
70 return pv_get_null(msg, param, res);
71 return pv_get_strlval(msg, param, res, br->dst_uri, br->dst_uri_len);
74 return pv_get_null(msg, param, res);
75 return pv_get_strlval(msg, param, res, br->path, br->path_len);
77 if(br->q == Q_UNSPECIFIED)
78 return pv_get_null(msg, param, res);
79 return pv_get_sintval(msg, param, res, br->q);
80 case 4: /* send socket */
81 if(br->force_send_socket!=0)
82 return pv_get_strval(msg, param, res, &br->force_send_socket->sock_str);
83 return pv_get_null(msg, param, res);
85 return pv_get_uintval(msg, param, res, nr_branches);
87 return pv_get_uintval(msg, param, res, br->flags);
90 return pv_get_null(msg, param, res);
91 return pv_get_strlval(msg, param, res, br->ruid, br->ruid_len);
92 case 8: /* location_ua */
93 if(br->location_ua_len==0)
94 return pv_get_null(msg, param, res);
95 return pv_get_strlval(msg, param, res, br->location_ua, br->location_ua_len);
98 return pv_get_strlval(msg, param, res, br->uri, br->len);
104 int pv_get_branchx(sip_msg_t *msg, pv_param_t *param,
107 return pv_get_branchx_helper(msg, param, res, 0);
110 int pv_set_branchx_helper(sip_msg_t *msg, pv_param_t *param,
111 int op, pv_value_t *val, int btype)
116 struct socket_info *si;
121 if(msg==NULL || param==NULL)
123 LM_ERR("bad parameters\n");
131 if(pv_get_spec_index(msg, param, &idx, &idxf)!=0)
133 LM_ERR("invalid index\n");
138 if((int)nr_branches + idx >= 0) {
141 LM_ERR("index too low: %d (%u)\n", idx, nr_branches);
145 LM_DBG("managing branch index %d (%u)\n", idx, nr_branches);
146 br = get_sip_branch(idx);
151 LM_DBG("no branch to operate on\n");
155 switch(param->pvn.u.isname.name.n)
157 case 1: /* dst uri */
158 if(val==NULL || (val->flags&PV_VAL_NULL))
160 br->dst_uri[0] = '\0';
164 if(!(val->flags&PV_VAL_STR))
166 LM_ERR("str value required to set branch dst uri\n");
171 br->dst_uri[0] = '\0';
176 if (unlikely(val->rs.len > MAX_URI_SIZE - 1))
178 LM_ERR("too long dst uri: %.*s\n",
179 val->rs.len, val->rs.s);
182 memcpy(br->dst_uri, val->rs.s, val->rs.len);
183 br->dst_uri[val->rs.len] = 0;
184 br->dst_uri_len = val->rs.len;
187 if(val==NULL || (val->flags&PV_VAL_NULL))
193 if(!(val->flags&PV_VAL_STR))
195 LM_ERR("str value required to set branch path\n");
205 if (unlikely(val->rs.len > MAX_PATH_SIZE - 1))
207 LM_ERR("path too long: %.*s\n",
208 val->rs.len, val->rs.s);
211 memcpy(br->path, val->rs.s, val->rs.len);
212 br->path[val->rs.len] = 0;
213 br->path_len = val->rs.len;
216 if(val==NULL || (val->flags&PV_VAL_NULL))
218 br->q = Q_UNSPECIFIED;
221 if(!(val->flags&PV_VAL_INT))
223 LM_ERR("int value required to set branch q\n");
228 case 4: /* send socket */
229 if(val==NULL || (val->flags&PV_VAL_NULL))
231 br->force_send_socket = NULL;
234 if(!(val->flags&PV_VAL_STR))
236 LM_ERR("str value required to set branch send sock\n");
241 br->force_send_socket = NULL;
244 STR_VTOZ(val->rs.s[val->rs.len], backup);
245 if (parse_phostport(val->rs.s, &host.s, &host.len, &port,
248 LM_ERR("invalid socket specification\n");
249 STR_ZTOV(val->rs.s[val->rs.len], backup);
252 STR_ZTOV(val->rs.s[val->rs.len], backup);
253 si = grep_sock_info(&host, (unsigned short)port,
254 (unsigned short)proto);
257 br->force_send_socket = si;
259 LM_WARN("no socket found to match [%.*s]\n",
260 val->rs.len, val->rs.s);
261 br->force_send_socket = NULL;
265 /* do nothing - cannot set the branch counter */
268 if(val==NULL || (val->flags&PV_VAL_NULL))
273 if(!(val->flags&PV_VAL_INT))
275 LM_ERR("int value required to set branch flags\n");
281 /* do nothing - cannot set the ruid */
283 case 8: /* location_ua */
284 /* do nothing - cannot set the location_ua */
288 if(val==NULL || (val->flags&PV_VAL_NULL))
291 memset(br, 0, sizeof(branch_t));
293 drop_sip_branch(idx);
296 if(!(val->flags&PV_VAL_STR))
298 LM_ERR("str value required to set branch uri\n");
304 memset(br, 0, sizeof(branch_t));
306 drop_sip_branch(idx);
309 if (unlikely(val->rs.len > MAX_URI_SIZE - 1))
311 LM_ERR("too long r-uri: %.*s\n",
312 val->rs.len, val->rs.s);
315 memcpy(br->uri, val->rs.s, val->rs.len);
316 br->uri[val->rs.len] = 0;
317 br->len = val->rs.len;
325 int pv_set_branchx(sip_msg_t *msg, pv_param_t *param,
326 int op, pv_value_t *val)
328 return pv_set_branchx_helper(msg, param, op, val, 0);
331 int pv_parse_branchx_name(pv_spec_p sp, str *in)
333 if(sp==NULL || in==NULL || in->len<=0)
339 if(strncmp(in->s, "uri", 3)==0)
340 sp->pvp.pvn.u.isname.name.n = 0;
344 if(strncmp(in->s, "dst_uri", 7)==0)
345 sp->pvp.pvn.u.isname.name.n = 1;
349 if(strncmp(in->s, "path", 4)==0)
350 sp->pvp.pvn.u.isname.name.n = 2;
351 else if (strncmp(in->s, "ruid", 4)==0)
352 sp->pvp.pvn.u.isname.name.n = 7;
356 if(*in->s=='q' || *in->s=='Q')
357 sp->pvp.pvn.u.isname.name.n = 3;
361 if(strncmp(in->s, "send_socket", 11)==0)
362 sp->pvp.pvn.u.isname.name.n = 4;
363 else if(strncmp(in->s, "location_ua", 11)==0)
364 sp->pvp.pvn.u.isname.name.n = 8;
368 if(strncmp(in->s, "count", 5)==0)
369 sp->pvp.pvn.u.isname.name.n = 5;
370 else if(strncmp(in->s, "flags", 5)==0)
371 sp->pvp.pvn.u.isname.name.n = 6;
377 sp->pvp.pvn.type = PV_NAME_INTSTR;
378 sp->pvp.pvn.u.isname.type = 0;
383 LM_ERR("unknown PV branch name %.*s\n", in->len, in->s);
387 int pv_get_sbranch(sip_msg_t *msg, pv_param_t *param,
390 return pv_get_branchx_helper(msg, param, res, 1);
393 int pv_set_sbranch(sip_msg_t *msg, pv_param_t *param,
394 int op, pv_value_t *val)
396 return pv_set_branchx_helper(msg, param, op, val, 1);
399 int pv_get_sndfrom(struct sip_msg *msg, pv_param_t *param,
402 struct onsend_info* snd_inf;
405 snd_inf=get_onsend_info();
406 if (! likely(snd_inf && snd_inf->send_sock))
407 return pv_get_null(msg, param, res);
409 switch(param->pvn.u.isname.name.n)
412 return pv_get_uintval(msg, param, res,
413 (int)snd_inf->send_sock->address.af);
415 return pv_get_uintval(msg, param, res,
416 (int)snd_inf->send_sock->port_no);
418 return pv_get_uintval(msg, param, res,
419 (int)snd_inf->send_sock->proto);
422 s.len = snd_inf->len;
423 return pv_get_strval(msg, param, res, &s);
425 return pv_get_uintval(msg, param, res,
428 if(get_valid_proto_string((int)snd_inf->send_sock->proto,
430 return pv_get_null(msg, param, res);
431 return pv_get_strval(msg, param, res, &s);
434 return pv_get_strval(msg, param, res,
435 &snd_inf->send_sock->address_str);
441 int pv_get_sndto(struct sip_msg *msg, pv_param_t *param,
444 struct onsend_info* snd_inf;
448 snd_inf=get_onsend_info();
449 if (! likely(snd_inf && snd_inf->send_sock))
450 return pv_get_null(msg, param, res);
452 switch(param->pvn.u.isname.name.n)
455 return pv_get_uintval(msg, param, res,
456 (int)snd_inf->send_sock->address.af);
458 return pv_get_uintval(msg, param, res,
459 (int)su_getport(snd_inf->to));
461 return pv_get_uintval(msg, param, res,
462 (int)snd_inf->send_sock->proto);
465 s.len = snd_inf->len;
466 return pv_get_strval(msg, param, res, &s);
468 return pv_get_uintval(msg, param, res,
471 if(get_valid_proto_string((int)snd_inf->send_sock->proto,
473 return pv_get_null(msg, param, res);
474 return pv_get_strval(msg, param, res, &s);
477 su2ip_addr(&ip, snd_inf->to);
478 s.s = ip_addr2a(&ip);
480 return pv_get_strval(msg, param, res, &s);
486 int pv_parse_snd_name(pv_spec_p sp, str *in)
488 if(sp==NULL || in==NULL || in->len<=0)
494 if(strncmp(in->s, "ip", 2)==0)
495 sp->pvp.pvn.u.isname.name.n = 0;
496 else if(strncmp(in->s, "af", 2)==0)
497 sp->pvp.pvn.u.isname.name.n = 1;
501 if(strncmp(in->s, "buf", 3)==0)
502 sp->pvp.pvn.u.isname.name.n = 4;
503 else if(strncmp(in->s, "len", 3)==0)
504 sp->pvp.pvn.u.isname.name.n = 5;
508 if(strncmp(in->s, "port", 4)==0)
509 sp->pvp.pvn.u.isname.name.n = 2;
513 if(strncmp(in->s, "proto", 5)==0)
514 sp->pvp.pvn.u.isname.name.n = 3;
518 if(strncmp(in->s, "sproto", 6)==0)
519 sp->pvp.pvn.u.isname.name.n = 6;
525 sp->pvp.pvn.type = PV_NAME_INTSTR;
526 sp->pvp.pvn.u.isname.type = 0;
531 LM_ERR("unknown PV snd name %.*s\n", in->len, in->s);
535 int pv_get_rcv(struct sip_msg *msg, pv_param_t *param,
538 sr_net_info_t *neti = NULL;
541 neti = ksr_evrt_rcvnetinfo_get();
543 if (neti==NULL || neti->rcv==NULL || neti->rcv->bind_address==NULL)
544 return pv_get_null(msg, param, res);
546 switch(param->pvn.u.isname.name.n)
550 s.len = neti->data.len;
551 return pv_get_strval(msg, param, res, &s);
553 return pv_get_uintval(msg, param, res,
554 (int)neti->data.len);
556 return pv_get_uintval(msg, param, res, (int)neti->rcv->proto);
558 s.s = ip_addr2a(&neti->rcv->src_ip);
560 return pv_get_strval(msg, param, res, &s);
562 s.s = ip_addr2a(&neti->rcv->dst_ip);
564 return pv_get_strval(msg, param, res, &s);
566 if(get_valid_proto_string((int)neti->rcv->proto,
568 return pv_get_null(msg, param, res);
570 return pv_get_strval(msg, param, res, &s);
571 case 7: /* srcport */
572 return pv_get_uintval(msg, param, res,
573 (int)neti->rcv->src_port);
574 case 8: /* rcvport */
575 return pv_get_uintval(msg, param, res,
576 (int)neti->rcv->dst_port);
579 return pv_get_uintval(msg, param, res,
580 (int)neti->rcv->bind_address->address.af);
586 int pv_parse_rcv_name(pv_spec_p sp, str *in)
588 if(sp==NULL || in==NULL || in->len<=0)
594 if(strncmp(in->s, "af", 2)==0)
595 sp->pvp.pvn.u.isname.name.n = 0;
599 if(strncmp(in->s, "buf", 3)==0)
600 sp->pvp.pvn.u.isname.name.n = 1;
601 else if(strncmp(in->s, "len", 3)==0)
602 sp->pvp.pvn.u.isname.name.n = 2;
606 if(strncmp(in->s, "proto", 5)==0)
607 sp->pvp.pvn.u.isname.name.n = 3;
608 else if(strncmp(in->s, "srcip", 5)==0)
609 sp->pvp.pvn.u.isname.name.n = 4;
610 else if(strncmp(in->s, "rcvip", 5)==0)
611 sp->pvp.pvn.u.isname.name.n = 5;
615 if(strncmp(in->s, "sproto", 6)==0)
616 sp->pvp.pvn.u.isname.name.n = 6;
620 if(strncmp(in->s, "srcport", 7)==0)
621 sp->pvp.pvn.u.isname.name.n = 7;
622 else if(strncmp(in->s, "rcvport", 7)==0)
623 sp->pvp.pvn.u.isname.name.n = 8;
629 sp->pvp.pvn.type = PV_NAME_INTSTR;
630 sp->pvp.pvn.u.isname.type = 0;
635 LM_ERR("unknown PV rcv name %.*s\n", in->len, in->s);
639 int pv_get_nh(struct sip_msg *msg, pv_param_t *param,
642 struct sip_uri parsed_uri;
645 if(msg==NULL || res==NULL)
648 if(msg->first_line.type == SIP_REPLY) /* REPLY doesnt have r/d-uri */
649 return pv_get_null(msg, param, res);
651 if (msg->dst_uri.s != NULL && msg->dst_uri.len>0)
655 if (msg->new_uri.s!=NULL && msg->new_uri.len>0)
659 uri = msg->first_line.u.request.uri;
662 if(param->pvn.u.isname.name.n==0) /* uri */
664 return pv_get_strval(msg, param, res, &uri);
666 if(parse_uri(uri.s, uri.len, &parsed_uri)!=0)
668 LM_ERR("failed to parse nh uri [%.*s]\n", uri.len, uri.s);
669 return pv_get_null(msg, param, res);
671 if(param->pvn.u.isname.name.n==1) /* username */
673 if(parsed_uri.user.s==NULL || parsed_uri.user.len<=0)
674 return pv_get_null(msg, param, res);
675 return pv_get_strval(msg, param, res, &parsed_uri.user);
676 } else if(param->pvn.u.isname.name.n==2) /* domain */ {
677 if(parsed_uri.host.s==NULL || parsed_uri.host.len<=0)
678 return pv_get_null(msg, param, res);
679 return pv_get_strval(msg, param, res, &parsed_uri.host);
680 } else if(param->pvn.u.isname.name.n==3) /* port */ {
681 if(parsed_uri.port.s==NULL)
682 return pv_get_5060(msg, param, res);
683 return pv_get_strintval(msg, param, res, &parsed_uri.port,
684 (int)parsed_uri.port_no);
685 } else if(param->pvn.u.isname.name.n==4) /* protocol */ {
686 if(parsed_uri.transport_val.s==NULL)
687 return pv_get_udp(msg, param, res);
688 return pv_get_strintval(msg, param, res, &parsed_uri.transport_val,
689 (int)parsed_uri.proto);
691 LM_ERR("unknown specifier\n");
692 return pv_get_null(msg, param, res);
695 int pv_parse_nh_name(pv_spec_p sp, str *in)
697 if(sp==NULL || in==NULL || in->len<=0)
703 if(strncmp(in->s, "u", 1)==0)
704 sp->pvp.pvn.u.isname.name.n = 0;
705 else if(strncmp(in->s, "U", 1)==0)
706 sp->pvp.pvn.u.isname.name.n = 1;
707 else if(strncmp(in->s, "d", 1)==0)
708 sp->pvp.pvn.u.isname.name.n = 2;
709 else if(strncmp(in->s, "p", 1)==0)
710 sp->pvp.pvn.u.isname.name.n = 3;
711 else if(strncmp(in->s, "P", 1)==0)
712 sp->pvp.pvn.u.isname.name.n = 4;
718 sp->pvp.pvn.type = PV_NAME_INTSTR;
719 sp->pvp.pvn.u.isname.type = 0;
724 LM_ERR("unknown PV nh name %.*s\n", in->len, in->s);
731 int sbranch_set_ruri(sip_msg_t *msg)
746 if (rewrite_uri(msg, &sv) < 0) {
747 LM_ERR("unable to rewrite Request-URI\n");
752 /* reset next hop address */
754 if(br->dst_uri_len>0) {
756 sv.len = br->dst_uri_len;
757 if (set_dst_uri(msg, &sv) < 0) {
763 reset_path_vector(msg);
764 if(br->path_len==0) {
766 sv.len = br->path_len;
767 if(set_path_vector(msg, &sv) < 0) {
774 if (br->instance_len) {
776 sv.len = br->instance_len;
777 if (set_instance(msg, &sv) < 0) {
786 sv.len = br->ruid_len;
787 if (set_ruid(msg, &sv) < 0) {
794 if (br->location_ua_len) {
795 sv.s = br->location_ua;
796 sv.len = br->location_ua_len;
797 if (set_ua(msg, &sv) < 0) {
803 if (br->force_send_socket)
804 set_force_socket(msg, br->force_send_socket);
806 msg->reg_id = br->reg_id;
809 getbflagsval(0, &old_bflags);
810 setbflagsval(0, old_bflags|br->flags);
820 int sbranch_append(sip_msg_t *msg)
826 str location_ua = {0};
836 if(br->dst_uri_len) {
837 duri.s = br->dst_uri;
838 duri.len = br->dst_uri_len;
842 path.len = br->path_len;
846 ruid.len = br->ruid_len;
848 if(br->location_ua_len) {
849 location_ua.s = br->location_ua;
850 location_ua.len = br->location_ua_len;
853 if (append_branch(msg, &uri, &duri, &path, br->q, br->flags,
854 br->force_send_socket, 0 /*instance*/, br->reg_id,
857 LM_ERR("failed to append static branch\n");
866 int sbranch_reset(void)
868 memset(&_pv_sbranch, 0, sizeof(branch_t));