⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 sip_utils.c

📁 Siproxd is a proxy/masquerading daemon for the SIP protocol. It handles registrations of SIP client
💻 C
📖 第 1 页 / 共 3 页
字号:
 * RETURNS *	STS_TRUE if the request is addressed local *	STS_FALSE otherwise */int is_sipuri_local (sip_ticket_t *ticket) {   osip_message_t *sip=ticket->sipmsg;   int found;   struct in_addr addr_uri, addr_myself;   int port;   int i;   if (sip==NULL) {      ERROR("called is_sipuri_local with NULL sip");      return STS_FALSE;   }   if (!sip || !sip->req_uri) {      ERROR("is_sipuri_local: no request URI present");      return STS_FALSE;   }   DEBUGC(DBCLASS_DNS,"check for local SIP URI %s:%s",          sip->req_uri->host? sip->req_uri->host : "*NULL*",          sip->req_uri->port? sip->req_uri->port : "*NULL*");   if (utils_inet_aton(sip->req_uri->host, &addr_uri) == 0) {      /* need name resolution */      get_ip_by_host(sip->req_uri->host, &addr_uri);   }      found=0;   for (i=0; i<2; i++) {      /*       * search my in/outbound interfaces       */      DEBUGC(DBCLASS_BABBLE,"resolving IP of interface %s",             (i==IF_INBOUND)? "inbound":"outbound");      if (get_interface_ip(i, &addr_myself) != STS_SUCCESS) {         continue;      }      /* check the extracted HOST against my own host addresses */      if (sip->req_uri->port) {         port=atoi(sip->req_uri->port);      } else {         port=SIP_PORT;      }      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(sip_ticket_t *ticket, int code) {   osip_message_t *response;   int sts;   osip_via_t *via;   int port;   char *buffer;   size_t buflen;   struct in_addr addr;   /* create the response template */   if ((response=msg_make_template_reply(ticket, code))==NULL) {      ERROR("sip_gen_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("sip_gen_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 = sip_message_to_str(response, &buffer, &buflen);   if (sts != 0) {      ERROR("sip_gen_response: msg_2char failed");      return STS_FAILURE;   }   if (via->port) {      port=atoi(via->port);   } else {      port=SIP_PORT;   }   /* send to destination */   sipsock_send(addr, port, ticket->protocol, buffer, buflen);   /* 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 (sip_ticket_t *ticket, int interface) {   struct in_addr addr;   char tmp[URL_STRING_SIZE];   osip_via_t *via;   int sts;   char branch_id[VIA_BRANCH_SIZE];   char *myaddr;   if (get_interface_ip(interface, &addr) != STS_SUCCESS) {      return STS_FAILURE;   }   sts = sip_calculate_branch_id(ticket, branch_id);   myaddr=utils_inet_ntoa(addr);   sprintf(tmp, "SIP/2.0/UDP %s:%i;branch=%s",           myaddr, 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(ticket->sipmsg->vias,via,0);   return STS_SUCCESS;}/* * SIP_DEL_MYVIA * * RETURNS *	STS_SUCCESS on success *	STS_FAILURE on error */int sip_del_myvia (sip_ticket_t *ticket) {   osip_via_t *via;   int sts;   DEBUGC(DBCLASS_PROXY,"deleting topmost VIA");   via = osip_list_get (ticket->sipmsg->vias, 0);      if ( via == NULL ) {      ERROR("Got empty VIA list - is your UA configured properly?");      return STS_FAILURE;   }   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(ticket->sipmsg->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 (sip_ticket_t *ticket, int direction) {   osip_message_t *sip_msg=ticket->sipmsg;   osip_contact_t *contact;   int i, j;   int replaced=0;   if (sip_msg == NULL) return STS_FAILURE;   /* at least one contact header present? */   osip_message_get_contact(sip_msg, 0, &contact);   if (contact == NULL) return STS_FAILURE;   /* loop for all existing contact headers in message */   for (j=0; contact != NULL; j++) {      osip_message_get_contact(sip_msg, j, &contact);      if (contact == NULL) break;      if (contact->url == NULL) continue;      /* search for an entry */      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;         if (direction == DIR_OUTGOING) {            DEBUGC(DBCLASS_PROXY, "rewriting 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);         } else {            DEBUGC(DBCLASS_PROXY, "rewriting Contact header %s@%s -> %s@%s",                   (contact->url->username)? contact->url->username : "*NULL*",                   (contact->url->host)? contact->url->host : "*NULL*",                   urlmap[i].true_url->username, urlmap[i].true_url->host);         }         /* remove old entry */         osip_list_remove(sip_msg->contacts,j);         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,j);         replaced=1;      }   }   if (replaced == 0) {      DEBUGC(DBCLASS_PROXY, "no Contact header rewritten");      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 (sip_ticket_t *ticket, 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:

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -