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

📄 sip_tag_class.c

📁 Sofia SIP is an open-source SIP User-Agent library, compliant with the IETF RFC3261 specification.
💻 C
字号:
/* * This file is part of the Sofia-SIP package * * Copyright (C) 2005 Nokia Corporation. * * Contact: Pekka Pessi <pekka.pessi@nokia.com> * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public License * as published by the Free Software Foundation; either version 2.1 of * the License, or (at your option) any later version. * * This library 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 * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA * 02110-1301 USA * *//**@SIP_TAG *  * @CFILE sip_tag_class.c  SIP Tag classes * * @author Pekka Pessi <Pekka.Pessi@nokia.com>. * * @date Created: Fri Feb 23 12:46:42 2001 ppessi */#include "config.h"#include "sofia-sip/sip_parser.h"#include <sofia-sip/su_tag_class.h>#include <sofia-sip/su_tag_inline.h>#include <sofia-sip/sip_tag_class.h>#include <sofia-sip/sip_tag.h>#include <sofia-sip/su_tagarg.h>#include <sofia-sip/su_strlst.h>#include <stdio.h>#include <ctype.h>#include <assert.h>#include <stddef.h>#include <string.h>#include <limits.h>/** Tag class for tags containing SIP headers. @HIDE * * Tags in this class are not automatically added to the message with * sip_add_tl() or sip_add_tagis(). */tag_class_t sipexthdrtag_class[1] =   {{    sizeof(siphdrtag_class),    /* tc_next */     NULL,    /* tc_len */      NULL,    /* tc_move */     NULL,    /* tc_xtra */     msghdrtag_xtra,    /* tc_dup */      msghdrtag_dup,    /* tc_free */     NULL,    /* tc_find */     NULL,    /* tc_snprintf */ msghdrtag_snprintf,    /* tc_filter */   siptag_filter,    /* tc_ref_set */  t_ptr_ref_set,    /* tc_scan */     msghdrtag_scan,  }};/** Tag class for SIP header tags. @HIDE */tag_class_t siphdrtag_class[1] =   {{    sizeof(siphdrtag_class),    /* tc_next */     NULL,    /* tc_len */      NULL,    /* tc_move */     NULL,    /* tc_xtra */     msghdrtag_xtra,    /* tc_dup */      msghdrtag_dup,    /* tc_free */     NULL,    /* tc_find */     NULL,    /* tc_snprintf */ msghdrtag_snprintf,    /* tc_filter */   siptag_filter,    /* tc_ref_set */  t_ptr_ref_set,    /* tc_scan */     msghdrtag_scan,  }};/** Tag class for SIP header string tags. @HIDE */tag_class_t sipstrtag_class[1] =   {{    sizeof(sipstrtag_class),    /* tc_next */     NULL,    /* tc_len */      NULL,    /* tc_move */     NULL,    /* tc_xtra */     t_str_xtra,    /* tc_dup */      t_str_dup,    /* tc_free */     NULL,    /* tc_find */     NULL,    /* tc_snprintf */ t_str_snprintf,    /* tc_filter */   NULL /* msgtag_str_filter */,    /* tc_ref_set */  t_ptr_ref_set,    /* tc_scan */     t_str_scan  }};/** Tag class for SIP message tags. @HIDE */tag_class_t sipmsgtag_class[1] =   {{    sizeof(sipmsgtag_class),    /* tc_next */     NULL,    /* tc_len */      NULL,    /* tc_move */     NULL,    /* tc_xtra */     msgobjtag_xtra,    /* tc_dup */      msgobjtag_dup,    /* tc_free */     NULL,    /* tc_find */     NULL,    /* tc_snprintf */ msgobjtag_snprintf,    /* tc_filter */   NULL /* siptag_sip_filter */,    /* tc_ref_set */  t_ptr_ref_set,  }};/** Filter a for SIP header tag. * * @param[in] dst tag list for filtering result. May be NULL. * @param[in] f   filter tag  * @param[in] src tag item from source list.  * @param[in,out] bb pointer to pointer of mempory area used to dup  *                   the filtering result * * This function is also used to calculate size for filtering result. */tagi_t *siptag_filter(tagi_t *dst,		      tagi_t const f[],		      tagi_t const *src, 		      void **bb){  tagi_t stub[2] = {{ NULL }};  tag_type_t srctt, tt = f->t_tag;  msg_hclass_t *hc = (msg_hclass_t *)tt->tt_magic;  assert(src);  srctt = src->t_tag;  /* Match filtered header with a header from a SIP message */  if (srctt && srctt->tt_class == sipmsgtag_class) {    sip_t const *sip = (sip_t const *)src->t_value;    sip_header_t const **hh, *h;    if (sip == NULL)      return dst;    hh = (sip_header_t const **)      msg_hclass_offset((msg_mclass_t *)sip->sip_common->h_class, 			(msg_pub_t *)sip, hc);    /* Is header present in the SIP message? */    if (hh == NULL ||	(char *)hh >= ((char *)sip + sip->sip_size) ||	(char *)hh < (char *)&sip->sip_request)      return dst;    h = *hh;    if (h == NULL)      return dst;    stub[0].t_tag = tt;    stub[0].t_value = (tag_value_t)h;    src = stub; srctt = tt;  }  if (tt != srctt)    return dst;  if (!src->t_value)    return dst;  else if (dst) {    return t_dup(dst, src, bb);  }  else {    *bb = (char *)*bb + t_xtra(src, (size_t)*bb);    return dst + 1;  }}/** Duplicate headers from taglist and add them to the SIP message. */int sip_add_tl(msg_t *msg, sip_t *sip,	       tag_type_t tag, tag_value_t value, ...){  tagi_t const *t;  ta_list ta;  int retval;  ta_start(ta, tag, value);  t = ta_args(ta);  retval = sip_add_tagis(msg, sip, &t);  ta_end(ta);  return retval;}/** Add duplicates of headers from taglist to the SIP message. */int sip_add_tagis(msg_t *msg, sip_t *sip, tagi_t const **inout_list){  tagi_t const *t;  tag_type_t tag;  tag_value_t value;  if (!msg || !inout_list)    return -1;  if (sip == NULL)    sip = sip_object(msg);  for (t = *inout_list; t; t = t_next(t)) {    tag = t->t_tag, value = t->t_value;    if (tag == NULL || tag == siptag_end) {      t = t_next(t);      break;    }    if (!value)      continue;    if (SIPTAG_P(tag)) {      msg_hclass_t *hc = (msg_hclass_t *)tag->tt_magic;      msg_header_t *h = (msg_header_t *)value, **hh;      if (h == SIP_NONE) {	/* Remove header */	hh = msg_hclass_offset(msg_mclass(msg), (msg_pub_t *)sip, hc);	if (hh != NULL &&	    (char *)hh < ((char *)sip + sip->sip_size) &&	    (char *)hh >= (char *)&sip->sip_request) {	  while (*hh)	    msg_header_remove(msg, (msg_pub_t *)sip, *hh);	}	continue;      }       if (tag == siptag_header)	hc = h->sh_class;      if (msg_header_add_dup_as(msg, (msg_pub_t *)sip, hc, h) < 0)	break;    }    else if (SIPTAG_STR_P(tag)) {      msg_hclass_t *hc = (msg_hclass_t *)tag->tt_magic;      char const *s = (char const *)value;      if (s && msg_header_add_make(msg, (msg_pub_t *)sip, hc, s) < 0)	return -1;    }    else if (tag == siptag_header_str) {      if (msg_header_add_str(msg, (msg_pub_t *)sip, (char const *)value) < 0)	return -1;    }  }  *inout_list = t;  return 0;}static char const *append_escaped(su_strlst_t *l,				  msg_hclass_t *hc,				  char const *s);/** Convert tagged SIP headers to a URL-encoded headers list. * * The SIP URI can contain a query part separated with the "?", which * specifies SIP headers that are included in the request constructed * from the URI. For example, using URI @code <sip:example.com?subject=test> * would include @Subject header with value "test" in the request. * * @param home memory home used to allocate query string (if NULL, use malloc) * @param tag, value, ... list of tagged arguments * * @bug This function returns NULL if SIPTAG_REQUEST(), SIPTAG_STATUS(), * SIPTAG_HEADER(), SIPTAG_UNKNOWN(), SIPTAG_ERROR(), SIPTAG_SEPARATOR(), or * any corresponding string tag is included in the tag list. It ignores * SIPTAG_SIP(). * * @par Example * @code * url->url_headers =  *   sip_headers_as_url_query(home, SIPTAG_REPLACES(replaces), TAG_END()); * @endcode *  * @since New in @VERSION_1_12_4. * * @sa  * url_query_as_header_string(), sip_url_query_as_taglist(), * nta_msg_request_complete(), * @RFC3261 section 19.1.1 "Headers", #url_t, url_s#url_headers */char *sip_headers_as_url_query(su_home_t *home,			       tag_type_t tag, tag_value_t value,			       ...){  ta_list ta;  tagi_t const *t;  su_strlst_t *l = su_strlst_create(home);  su_home_t *lhome = su_strlst_home(l);  char const *retval = "";  if (!l)    return NULL;  ta_start(ta, tag, value);  for (t = ta_args(ta); t && retval; t = t_next(t)) {    msg_hclass_t *hc;    if (t->t_value == 0 || t->t_value == -1)      continue;    hc = (msg_hclass_t *)t->t_tag->tt_magic;    if (SIPTAG_P(t->t_tag)) {      sip_header_t const *h = (sip_header_t const *)t->t_value;      char *s = sip_header_as_string(lhome, h);      retval = append_escaped(l, hc, s);      if (retval != s)	su_free(lhome, s);    }    else if (SIPTAG_STR_P(t->t_tag)) {      retval = append_escaped(l, hc, (char const *)t->t_value);    }  }  ta_end(ta);  if (retval)    retval = su_strlst_join(l, home, "");  su_strlst_destroy(l);  return (char *)retval;}/* "[" / "]" / "/" / "?" / ":" / "+" / "$" */#define HNV_UNRESERVED "[]/?+$"#define HNV_RESERVED ":=,;"/* Append a string to list and url-escape it if needed */staticchar const *append_escaped(su_strlst_t *l,			   msg_hclass_t *hc,			   char const *s){  char const *name;  if (hc == NULL)    return NULL;  if (hc->hc_hash == sip_payload_hash)    name = "body";  else				/* XXX - could we use short form? */    name = hc->hc_name;  if (name == NULL)    return NULL;  if (s) {    su_home_t *lhome = su_strlst_home(l);    size_t slen;    isize_t elen;    char *n, *escaped;    char *sep = su_strlst_len(l) > 0 ? "&" : "";    n = su_sprintf(lhome, "%s%s=", sep, name);    if (!su_strlst_append(l, n))      return NULL;    for (;*n; n++)      if (isupper(*n))	*n = tolower(*n);        slen = strlen(s); elen = url_esclen(s, HNV_RESERVED);    if ((size_t)elen == slen)      return su_strlst_append(l, s);        escaped = su_alloc(lhome, elen + 1);    if (escaped)      return su_strlst_append(l, url_escape(escaped, s, HNV_RESERVED));  }  return NULL;}/** Convert URL query to a tag list. * * SIP headers encoded as URL @a query is parsed returned as a tag list. * Unknown headers are encoded as SIPTAG_HEADER_STR(). * * @param home memory home used to alloate string (if NULL, malloc() it) * @param query query part from SIP URL  * @param parser optional SIP parser used *  * @sa sip_add_tl(), sip_add_tagis(), SIPTAG_HEADER_STR(), * sip_headers_as_url_query(), url_query_as_header_string(), * @RFC3261 section 19.1.1 "Headers", #url_t, url_s#url_headers * * @NEW_1_12_4. * * @bug Extension headers are ignored. The @a parser parameter is not used * at the moment. */tagi_t *sip_url_query_as_taglist(su_home_t *home, char const *query,				 msg_mclass_t const *parser){  tagi_t *retval = NULL;  char *s;  su_strlst_t *l;  isize_t N;  size_t i, j, n;  if (query == NULL || query[0] == '\0' || query[0] == '&')    return NULL;  s = su_strdup(home, query); if (!s) return NULL;  l = su_strlst_split(home, s, "&");  N = su_strlst_len(l);  if (N == 0)    goto error;  retval = su_zalloc(home, (N + 1) * sizeof (*retval));  if (retval == NULL)    goto error;  for (i = 0; i < N; i++) {    char const *hnv;    char *value;    tag_type_t t;    tag_value_t v;    msg_hclass_t *hc = NULL;    hnv = su_strlst_item(l, i);    n = hnv ? strcspn(hnv, "=") : 0;    if (n == 0)      break;    if (n == 4 && strncasecmp(hnv, "body", 4) == 0)      t = siptag_payload, hc = sip_payload_class;    else {      for (j = 0; (t = sip_tag_list[j]); j++) {	hc = (msg_hclass_t *)sip_tag_list[j]->tt_magic;	if (n == 1 && strncasecmp(hnv, hc->hc_short, 1) == 0)	  break;	else if (n == (size_t)hc->hc_len &&		 strncasecmp(hnv, hc->hc_name, n) == 0)	  break;      }    }    value = (char *)hnv + n;    *value++ = ':';    n = url_unescape_to(value, value, SIZE_MAX);    value[n] = '\0';    if (t) {      msg_header_t *h = msg_header_make(home, hc, value);      if (!h)	break;      v = (tag_value_t)h;    }    else {      char *s;      s = su_alloc(home, n + 1);      if (!s)	break;      memcpy(s, value, n + 1);      t = siptag_header_str;      v = (tag_value_t)s;    }    retval[i].t_tag = t, retval[i].t_value = v;  }  retval[i].t_tag = NULL, retval[i].t_value = (tag_value_t)0;  if (i < N) {    for (j = 0; j < i; j++) {      if (retval[i].t_tag == siptag_header_str)	su_free(home, (void *)retval[i].t_value);      else	msg_header_free_all(home, (msg_header_t *)retval[i].t_value);    }    su_free(home, retval);    retval = NULL;  } error:  su_free(home, s);  su_strlst_destroy(l);  return retval;}

⌨️ 快捷键说明

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