📄 sip_utils.c
字号:
if ( (memcmp(&addr_myself, &addr_uri, sizeof(addr_myself))==0) && (port == configuration.sip_listen_port) ) { DEBUG("address match [%s]", utils_inet_ntoa(addr_uri)); found=1; break; } } DEBUGC(DBCLASS_DNS, "SIP URI is %slocal", found? "":"not "); return (found)? STS_TRUE : STS_FALSE;}/* * check if a given request (outbound -> inbound) shall its * request URI get rewritten based upon our UA knowledge * * RETURNS * STS_TRUE if to be rewritten * STS_FALSE otherwise */int check_rewrite_rq_uri (osip_message_t *sip) { int i, j, sts; int dflidx; osip_header_t *ua_hdr; /* get index of default entry */ dflidx=(sizeof(RQ_rewrite)/sizeof(RQ_rewrite[0])) - 1; /* check fort existence of method */ if ((sip==NULL) || (sip->sip_method==NULL)) { ERROR("check_rewrite_rq_uri: got NULL method"); return STS_FALSE; } /* extract UA string */ osip_message_get_user_agent (sip, 0, &ua_hdr); if ((ua_hdr==NULL) || (ua_hdr->hvalue==NULL)) { DEBUGC(DBCLASS_SIP, "check_rewrite_rq_uri: NULL UA in Header, " "using default"); i=dflidx; } else { /* loop through the knowledge base */ for (i=0; RQ_rewrite[i].UAstring; i++) { if (strncmp(RQ_rewrite[i].UAstring, ua_hdr->hvalue, sizeof(RQ_rewrite[i].UAstring))==0) { DEBUGC(DBCLASS_SIP, "got knowledge entry for [%s]", ua_hdr->hvalue); break; } } /* for i */ } /* if ua_hdr */ for (j=0; RQ_method[j].name; j++) { if (strncmp(RQ_method[j].name, sip->sip_method, RQ_method[j].size)==0) { if (RQ_rewrite[i].action[j] >= 0) { sts = (RQ_rewrite[i].action[j])? STS_TRUE: STS_FALSE; } else { sts = (RQ_rewrite[dflidx].action[j])? STS_TRUE: STS_FALSE; } DEBUGC(DBCLASS_SIP, "check_rewrite_rq_uri: [%s:%s, i=%i, j=%i] " "got action %s", (sip && sip->sip_method) ? sip->sip_method : "*NULL*", (ua_hdr && ua_hdr->hvalue)? ua_hdr->hvalue:"*NULL*", i, j, (sts==STS_TRUE)? "rewrite":"norewrite"); return sts; } } /* for j */ WARN("check_rewrite_rq_uri: didn't get a hit of the method [%s]", sip->sip_method); return STS_FALSE;}/* * SIP_GEN_RESPONSE * * send an proxy generated response back to the client. * Only errors are reported from the proxy itself. * code = SIP result code to deliver * * RETURNS * STS_SUCCESS on success * STS_FAILURE on error */int sip_gen_response(osip_message_t *request, int code) { osip_message_t *response; int sts; osip_via_t *via; int port; char *buffer; struct in_addr addr; /* create the response template */ if ((response=msg_make_template_reply(request, code))==NULL) { ERROR("proxy_response: error in msg_make_template_reply"); return STS_FAILURE; } /* we must check if first via has x.x.x.x address. If not, we must resolve it */ osip_message_get_via (response, 0, &via); if (via == NULL) { ERROR("proxy_response: Cannot send response - no via field"); return STS_FAILURE; } /* name resolution */ if (utils_inet_aton(via->host, &addr) == 0) { /* need name resolution */ DEBUGC(DBCLASS_DNS,"resolving name:%s",via->host); sts = get_ip_by_host(via->host, &addr); if (sts == STS_FAILURE) { DEBUGC(DBCLASS_PROXY, "sip_gen_response: cannot resolve via [%s]", via->host); return STS_FAILURE; } } sts = osip_message_to_str(response, &buffer); if (sts != 0) { ERROR("proxy_response: msg_2char failed"); return STS_FAILURE; } if (via->port) { port=atoi(via->port); } else { port=SIP_PORT; } /* send to destination */ sipsock_send_udp(&sip_socket, addr, port, buffer, strlen(buffer), 1); /* free the resources */ osip_message_free(response); osip_free(buffer); return STS_SUCCESS;}/* * SIP_ADD_MYVIA * * interface == IF_OUTBOUND, IF_INBOUND * * RETURNS * STS_SUCCESS on success * STS_FAILURE on error */int sip_add_myvia (osip_message_t *request, int interface) { struct in_addr addr; char tmp[URL_STRING_SIZE]; osip_via_t *via; int sts; char branch_id[VIA_BRANCH_SIZE]; if (interface == IF_OUTBOUND) { sts = get_ip_by_ifname(configuration.outbound_if, &addr); if (sts == STS_FAILURE) { ERROR("can't find outbound interface %s - configuration error?", configuration.outbound_if); return STS_FAILURE; } } else { sts = get_ip_by_ifname(configuration.inbound_if, &addr); if (sts == STS_FAILURE) { ERROR("can't find inbound interface %s - configuration error?", configuration.inbound_if); return STS_FAILURE; } } sts = sip_calculate_branch_id(request, branch_id); sprintf(tmp, "SIP/2.0/UDP %s:%i;branch=%s;", utils_inet_ntoa(addr), configuration.sip_listen_port, branch_id); DEBUGC(DBCLASS_BABBLE,"adding VIA:%s",tmp); sts = osip_via_init(&via); if (sts!=0) return STS_FAILURE; /* allocation failed */ sts = osip_via_parse(via, tmp); if (sts!=0) return STS_FAILURE; osip_list_add(request->vias,via,0); return STS_SUCCESS;}/* * SIP_DEL_MYVIA * * RETURNS * STS_SUCCESS on success * STS_FAILURE on error */int sip_del_myvia (osip_message_t *response) { osip_via_t *via; int sts; DEBUGC(DBCLASS_PROXY,"deleting topmost VIA"); via = osip_list_get (response->vias, 0); if ( is_via_local(via) == STS_FALSE ) { ERROR("I'm trying to delete a VIA but it's not mine! host=%s",via->host); return STS_FAILURE; } sts = osip_list_remove(response->vias, 0); osip_via_free (via); return STS_SUCCESS;}/* * SIP_REWRITE_CONTACT * * rewrite the Contact header * * RETURNS * STS_SUCCESS on success * STS_FAILURE on error */int sip_rewrite_contact (osip_message_t *sip_msg, int direction) { osip_contact_t *contact; int i; if (sip_msg == NULL) return STS_FAILURE; osip_message_get_contact(sip_msg, 0, &contact); if (contact == NULL) return STS_FAILURE; for (i=0;i<URLMAP_SIZE;i++){ if (urlmap[i].active == 0) continue; if ((direction == DIR_OUTGOING) && (compare_url(contact->url, urlmap[i].true_url)==STS_SUCCESS)) break; if ((direction == DIR_INCOMING) && (compare_url(contact->url, urlmap[i].masq_url)==STS_SUCCESS)) break; } /* found a mapping entry */ if (i<URLMAP_SIZE) { char *tmp; DEBUGC(DBCLASS_PROXY, "rewrote Contact header %s@%s -> %s@%s", (contact->url->username)? contact->url->username : "*NULL*", (contact->url->host)? contact->url->host : "*NULL*", urlmap[i].masq_url->username, urlmap[i].masq_url->host); /* remove old entry */ osip_list_remove(sip_msg->contacts,0); osip_contact_to_str(contact, &tmp); osip_contact_free(contact); /* clone the url from urlmap*/ osip_contact_init(&contact); osip_contact_parse(contact,tmp); osip_free(tmp); osip_uri_free(contact->url); if (direction == DIR_OUTGOING) { /* outgoing, use masqueraded url */ osip_uri_clone(urlmap[i].masq_url, &contact->url); } else { /* incoming, use true url */ osip_uri_clone(urlmap[i].true_url, &contact->url); } osip_list_add(sip_msg->contacts,contact,-1); } else { return STS_FAILURE; } return STS_SUCCESS;}/* * SIP_CALCULATE_BRANCH * * Calculates a branch parameter according to RFC3261 section 16.11 * * The returned 'id' will be HASHHEXLEN + strlen(magic_cookie) * characters (32 + 7) long. The caller must supply at least this * amount of space in 'id'. * * RETURNS * STS_SUCCESS on success * STS_FAILURE on error */int sip_calculate_branch_id (osip_message_t *sip_msg, char *id) {/* RFC3261 section 16.11 recommends the following procedure: * The stateless proxy MAY use any technique it likes to guarantee * uniqueness of its branch IDs across transactions. However, the * following procedure is RECOMMENDED. The proxy examines the * branch ID in the topmost Via header field of the received * request. If it begins with the magic cookie, the first * component of the branch ID of the outgoing request is computed * as a hash of the received branch ID. Otherwise, the first * component of the branch ID is computed as a hash of the topmost * Via, the tag in the To header field, the tag in the From header * field, the Call-ID header field, the CSeq number (but not * method), and the Request-URI from the received request. One of * these fields will always vary across two different * transactions. * * The branch value will consist of: * - magic cookie "z9hG4bK" * - 1st part (unique calculated ID * - 2nd part (value for loop detection) <<- not yet used by siproxd */ static char *magic_cookie="z9hG4bK"; osip_via_t *via; osip_uri_param_t *param=NULL; osip_call_id_t *call_id=NULL; HASHHEX hashstring; hashstring[0]='\0'; /* * Examine topmost via and look for a magic cookie. * If it is there, I use THIS branch parameter as input for * our hash calculation */ via = osip_list_get (sip_msg->vias, 0); if (via == NULL) { ERROR("have a SIP message without any via header"); return STS_FAILURE; } param=NULL; osip_via_param_get_byname(via, "branch", ¶m); if (param && param->gvalue) { if (strncmp(param->gvalue, magic_cookie, strlen(magic_cookie))) { /* calculate MD5 hash */ MD5_CTX Md5Ctx; HASH HA1; MD5Init(&Md5Ctx); MD5Update(&Md5Ctx, param->gvalue, strlen(param->gvalue)); MD5Final(HA1, &Md5Ctx); CvtHex(HA1, hashstring); DEBUGC(DBCLASS_BABBLE, "existing branch -> branch hash [%s]", hashstring); } } /* * If I don't have a branch parameter in the existing topmost via, * then I need: * - the topmost via * - the tag in the To header field * - the tag in the From header field * - the Call-ID header field * - the CSeq number (but not method) * - the Request-URI from the received request */ if (hashstring[0] == '\0') { /* calculate MD5 hash */ MD5_CTX Md5Ctx; HASH HA1; char *tmp; MD5Init(&Md5Ctx); /* topmost via */ osip_via_to_str(via, &tmp); if (tmp) { MD5Update(&Md5Ctx, tmp, strlen(tmp)); osip_free(tmp); } /* Tag in To header */ osip_to_get_tag(sip_msg->to, ¶m); if (param && param->gvalue) { MD5Update(&Md5Ctx, param->gvalue, strlen(param->gvalue)); } /* Tag in From header */ osip_from_get_tag(sip_msg->from, ¶m); if (param && param->gvalue) { MD5Update(&Md5Ctx, param->gvalue, strlen(param->gvalue)); } /* Call-ID */ call_id = osip_message_get_call_id(sip_msg); osip_call_id_to_str(call_id, &tmp); if (tmp) { MD5Update(&Md5Ctx, tmp, strlen(tmp)); osip_free(tmp); } /* CSeq number (but not method) */ tmp = osip_cseq_get_number(sip_msg->cseq); if (tmp) { MD5Update(&Md5Ctx, tmp, strlen(tmp)); } /* Request URI */ osip_uri_to_str(sip_msg->req_uri, &tmp); if (tmp) { MD5Update(&Md5Ctx, tmp, strlen(tmp)); osip_free(tmp); } MD5Final(HA1, &Md5Ctx); CvtHex(HA1, hashstring); DEBUGC(DBCLASS_BABBLE, "non-existing branch -> branch hash [%s]", hashstring); } /* include the magic cookie */ sprintf(id, "%s%s", magic_cookie, hashstring); return STS_SUCCESS;}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -