freeswitch/libs/sofia-sip/libsofia-sip-ua/msg/msg_parser_util.c

2006 lines
46 KiB
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
*
*/
/**@ingroup msg_parser
* @CFILE msg_parser_util.c
*
* Text-message parser helper functions.
*
* @author Pekka Pessi <Pekka.Pessi@nokia.com>
*
* @date Created: Tue Aug 28 16:26:34 2001 ppessi
*
*/
#include "config.h"
#include <stddef.h>
#include <stdlib.h>
#include <string.h>
#include <stdio.h>
#include <assert.h>
#include <limits.h>
#include <stdarg.h>
#include <sofia-sip/su_tagarg.h>
#include <sofia-sip/su.h>
#include <sofia-sip/su_alloc.h>
#include <sofia-sip/su_string.h>
#include "msg_internal.h"
#include "sofia-sip/msg_parser.h"
#include "sofia-sip/bnf.h"
#include "sofia-sip/url.h"
static issize_t msg_comma_scanner(char *start);
/**
* Parse first line.
*
* Splits the first line from a message into three whitespace-separated
* parts.
*/
int msg_firstline_d(char *s, char **return_part2, char **return_part3)
{
char *s1 = s, *s2, *s3;
size_t n;
/* Split line into three segments separated by whitespace */
if (s1[n = span_non_ws(s1)]) {
s1[n] = '\0';
s2 = s1 + n + 1;
while (IS_WS(*s2))
s2++;
}
else {
/* Hopeless - no WS in first line */
return -1;
}
n = span_non_ws(s2);
if (s2[n]) {
s2[n++] = '\0';
while (IS_WS(s2[n]))
n++;
}
s3 = s2 + n;
*return_part2 = s2;
*return_part3 = s3;
return 0;
}
/**Parse a token.
*
* Parses a token from string pointed by @a *ss. It stores the token value
* in @a return_token, and updates the @a ss to the end of token and
* possible whitespace.
*/
issize_t msg_token_d(char **ss, char const **return_token)
{
char *s = *ss;
size_t n = span_token(s);
if (n) {
for (; IS_LWS(s[n]); n++)
s[n] = '\0';
*return_token = s;
*ss = s + n;
return n;
}
else
return -1;
}
/** Parse a 32-bit unsigned int.
*
* The function msg_uint32_d() parses a 32-bit unsigned integer in string
* pointed by @a *ss. It stores the value in @a return_token and updates the
* @a ss to the end of integer and possible whitespace.
*
* @retval length of parsed integer, or -1 upon an error.
*/
issize_t msg_uint32_d(char **ss, uint32_t *return_value)
{
char const *s = *ss, *s0 = s;
uint32_t value;
unsigned digit;
if (!IS_DIGIT(*s))
return -1;
for (value = 0; IS_DIGIT(*s); s++) {
digit = *s - '0';
if (value > 429496729U)
return -1;
else if (value == 429496729U && digit > 5)
return -1;
value = 10 * value + digit;
}
if (*s) {
if (!IS_LWS(*s))
return (issize_t)-1;
skip_lws(&s);
}
*ss = (char *)s;
*return_value = value;
return s - s0;
}
/** Parse any list.
*
* Parses a list of items, separated by @a sep. The parsed string is passed
* in @a *ss, which is updated to point to the first non-linear-whitespace
* character after the list. The function modifies the string as it parses
* it.
*
* The parsed items are appended to the list @a *append_list. If there the
* list in @a *append_list is NULL, allocate a new list and return it in @a
* *append_list. Empty list items are ignored, and are not appended to the
* list.
*
* The function @b must be passed a scanning function @a scanner. The
* scanning function scans for a legitimate list item, for example, a token.
* It should also compact the list item, for instance, if the item consists
* of @c name=value parameter definitions it should remove whitespace around
* "=" sign. The scanning function returns the length of the scanned item,
* including any linear whitespace after it.
*
* @param[in] home memory home for allocating list pointers
* @param[in,out] ss pointer to pointer to string to be parsed
* @param[in,out] append_list pointer to list
* where parsed list items are appended
* @param[in] sep separator character
* @param[in] scanner pointer to function for scanning a single item
*
* @retval 0 if successful.
* @retval -1 upon an error.
*/
issize_t msg_any_list_d(su_home_t *home,
char **ss,
msg_param_t **append_list,
issize_t (*scanner)(char *s),
int sep)
{
char const *stack[MSG_N_PARAMS];
char const **list = stack, **re_list;
size_t N = MSG_N_PARAMS, n = 0;
issize_t tlen;
char *s = *ss;
char const **start;
if (!scanner)
return -1;
if (*append_list) {
list = *append_list;
while (list[n])
n++;
N = MSG_PARAMS_NUM(n + 1);
}
start = &list[n];
skip_lws(&s);
while (*s) {
tlen = scanner(s);
if (tlen < 0 || (s[tlen] && s[tlen] != sep && s[tlen] != ','))
goto error;
if (tlen > 0) {
if (n + 1 == N) { /* Reallocate list? */
N = MSG_PARAMS_NUM(N + 1);
if (list == stack || list == *append_list) {
re_list = su_alloc(home, N * sizeof(*list));
if (re_list)
memcpy(re_list, list, n * sizeof(*list));
}
else
re_list = su_realloc(home, list, N * sizeof(*list));
if (!re_list)
goto error;
list = re_list;
}
list[n++] = s;
s += tlen;
}
if (*s == sep) {
*s++ = '\0';
skip_lws(&s);
}
else if (*s)
break;
}
*ss = s;
if (n == 0) {
*append_list = NULL;
return 0;
}
if (list == stack) {
size_t size = sizeof(*list) * MSG_PARAMS_NUM(n + 1);
list = su_alloc(home, size);
if (!list) return -1;
memcpy((void *)list, stack, n * sizeof(*list));
}
list[n] = NULL;
*append_list = list;
return 0;
error:
*start = NULL;
if (list != stack && list != *append_list)
su_free(home, list);
return -1;
}
/** Scan an attribute (name [= value]) pair.
*
* The attribute consists of name (a token) and optional value, separated by
* equal sign. The value can be a token or quoted string.
*
* This function compacts the scanned value. It removes the
* whitespace around equal sign "=" by moving the equal sign character and
* value towards name.
*
* If there is whitespace within the scanned value or after it,
* NUL-terminates the scanned attribute.
*
* @retval > 0 number of characters scanned,
* including the whitespace within the value
* @retval -1 upon an error
*/
issize_t msg_attribute_value_scanner(char *s)
{
char *p = s;
size_t tlen;
skip_token(&s);
if (s == p) /* invalid parameter name */
return -1;
tlen = s - p;
if (IS_LWS(*s)) { *s++ = '\0'; skip_lws(&s); }
if (*s == '=') {
char *v;
s++;
skip_lws(&s);
/* get value */
if (*s == '"') {
size_t qlen = span_quoted(s);
if (!qlen)
return -1;
v = s; s += qlen;
}
else {
v = s;
skip_param(&s);
if (s == v)
return -1;
}
if (p + tlen + 1 != v) {
memmove(p + tlen + 1, v, s - v);
p[tlen] = '=';
p[tlen + 1 + (s - v)] = '\0';
}
}
if (IS_LWS(*s)) { *s++ = '\0'; skip_lws(&s); }
return s - p;
}
/**Parse an attribute-value list.
*
* Parses an attribute-value list, which has syntax as follows:
* @code
* av-list = (av-pair *(";" av-pair)
* av-pair = token ["=" ( value / quoted-string) ] ; optional value
* @endcode
*
* @param[in] home pointer to a memory home
* @param[in,out] ss pointer to string at the start of parameter list
* @param[in,out] append_list pointer to list
* where parsed list items are appended
*
* @retval >= 0 if successful
* @retval -1 upon an error
*/
issize_t msg_avlist_d(su_home_t *home,
char **ss,
msg_param_t const **append_list)
{
char const *stack[MSG_N_PARAMS];
char const **params;
size_t n = 0, N;
char *s = *ss;
if (!*s)
return -1;
if (*append_list) {
params = (char const **)*append_list;
for (n = 0; params[n]; n++)
;
N = MSG_PARAMS_NUM(n + 1);
}
else {
params = stack;
N = MSG_PARAMS_NUM(1);
}
for (;;) {
char *p;
size_t tlen;
/* XXX - we should handle also quoted parameters */
skip_lws(&s);
p = s;
skip_token(&s);
tlen = s - p;
if (!tlen) /* invalid parameter name */
goto error;
if (IS_LWS(*s)) { *s++ = '\0'; skip_lws(&s); }
if (*s == '=') {
char *v;
s++;
skip_lws(&s);
/* get value */
if (*s == '"') {
size_t qlen = span_quoted(s);
if (!qlen)
goto error;
v = s; s += qlen;
}
else {
v = s;
skip_param(&s);
if (s == v)
goto error;
}
if (p + tlen + 1 != v) {
p = memmove(v - tlen - 1, p, tlen);
p[tlen] = '=';
}
}
if (IS_LWS(*s)) { *s++ = '\0'; skip_lws(&s); }
if (n == N) {
/* Reallocate params */
char const **nparams = su_realloc(home, (void*)(params != stack ? params : NULL),
(N = MSG_PARAMS_NUM(N + 1)) * sizeof(*params));
if (!nparams) {
goto error;
}
if (params == stack)
memcpy(nparams, stack, n * sizeof(*params));
params = nparams;
}
params[n++] = p;
if (*s != ';')
break;
*s++ = '\0';
}
*ss = s;
if (params == stack) {
size_t size = sizeof(*params) * MSG_PARAMS_NUM(n + 1);
params = su_alloc(home, size);
if (!params) return -1;
memcpy((void *)params, stack, n * sizeof(*params));
}
else if (n == N) {
/* Reallocate params */
char const **nparams = su_realloc(home, (void*)(params != stack ? params : NULL),
(N = MSG_PARAMS_NUM(N + 1)) * sizeof(*params));
if (!nparams) {
goto error;
}
if (params == stack)
memcpy(nparams, stack, n * sizeof(*params));
params = nparams;
}
params[n] = NULL;
*append_list = params;
return 0;
error:
if (params != stack)
su_free(home, params);
return -1;
}
/**Parse a semicolon-separated parameter list starting with semicolon.
*
* Parse a parameter list, which has syntax as follows:
* @code
* *(";" token [ "=" (token | quoted-string)]).
* @endcode
*
* @param[in] home pointer to a memory home
* @param[in,out] ss pointer to string at the start of parameter list
* @param[in,out] append_list pointer to list
* where parsed list items are appended
*
* @retval >= 0 if successful
* @retval -1 upon an error
*
* @sa msg_avlist_d()
*/
issize_t msg_params_d(su_home_t *home,
char **ss,
msg_param_t const **append_list)
{
if (**ss == ';') {
*(*ss)++ = '\0';
*append_list = NULL;
return msg_avlist_d(home, ss, append_list);
}
if (IS_LWS(**ss)) {
*(*ss)++ = '\0'; skip_lws(ss);
}
return 0;
}
/** Encode a list of parameters */
isize_t msg_params_e(char b[], isize_t bsiz, msg_param_t const pparams[])
{
int i;
char *end = b + bsiz, *b0 = b;
msg_param_t p;
if (pparams)
for (i = 0; (p = pparams[i]); i++) {
if (p[0]) {
MSG_CHAR_E(b, end, ';');
MSG_STRING_E(b, end, p);
}
}
return b - b0;
}
/** Duplicate a parameter list */
char *msg_params_dup(msg_param_t const **d, msg_param_t const s[],
char *b, isize_t xtra)
{
char *end = b + xtra;
char **pp;
int i;
isize_t n;
n = msg_params_count(s);
if (n == 0) {
*d = NULL;
return b;
}
MSG_STRUCT_ALIGN(b);
pp = (char **)b;
b += sizeof(*pp) * MSG_PARAMS_NUM(n + 1);
for (i = 0; s[i]; i++) {
MSG_STRING_DUP(b, pp[i], s[i]);
}
pp[i] = NULL;
assert(b <= end); (void)end;
*d = (msg_param_t const *)pp;
return b;
}
/** Parse a comma-separated list.
*
* Parses a comma-separated list. The parsed string is passed in @a *ss,
* which is updated to point to the first non-linear-whitespace character
* after the list. The function modifies the string as it parses it.
*
* A pointer to the resulting list is returned in @a *retval. If there
* already is a list in @a *retval, new items are appended. Empty list items
* are ignored, and are not included in the list.
*
* The function can be passed an optional scanning function. The scanning
* function scans for a legitimate list item, for example, a token. It also
* compacts the list item, for instance, if the item consists of @c
* name=value parameter definitions. The scanning function returns the
* length of the scanned item, including any linear whitespace after it.
*
* By default, the scanning function accepts tokens, quoted strings or
* separators (except comma, of course).
*
* @param[in] home memory home for allocating list pointers
* @param[in,out] ss pointer to pointer to string to be parsed
* @param[in,out] append_list pointer to list
* where parsed list items are appended
* @param[in] scanner pointer to function scanning a single item
* (optional)
*
* @retval 0 if successful.
* @retval -1 upon an error.
*/
issize_t msg_commalist_d(su_home_t *home,
char **ss,
msg_param_t **append_list,
issize_t (*scanner)(char *s))
{
scanner = scanner ? scanner : msg_comma_scanner;
return msg_any_list_d(home, ss, append_list, scanner, ',');
}
/** Token scanner for msg_commalist_d() accepting also empty entries. */
issize_t msg_token_scan(char *start)
{
char *s = start;
skip_token(&s);
if (IS_LWS(*s))
*s++ = '\0';
skip_lws(&s);
return s - start;
}
/** Scan and compact a comma-separated item */
static
issize_t msg_comma_scanner(char *start)
{
size_t tlen;
char *s, *p;
s = p = start;
if (s[0] == ',')
return 0;
for (;;) {
/* Grab next section - token, quoted string, or separator character */
char c = *s;
if (IS_TOKEN(c))
tlen = span_token(s);
else if (c == '"')
tlen = span_quoted(s);
else /* if (IS_SEPARATOR(c)) */
tlen = 1;
if (tlen == 0)
return -1;
if (p != s)
memmove(p, s, tlen); /* Move section to end of paramexter */
p += tlen; s += tlen;
skip_lws(&s); /* Skip possible LWS */
if (*s == '\0' || *s == ',') { /* Test for possible end */
if (p != s) *p = '\0';
return s - start;
}
if (IS_TOKEN(c) && IS_TOKEN(*s))
*p++ = ' '; /* Two tokens must be separated by LWS */
}
}
/** Parse a comment.
*
* Parses a multilevel comment. The comment assigned to return-value
* parameter @a return_comment is NUL-terminated. The string at return-value
* parameter @a ss is updated to point to first non-linear-whitespace
* character after the comment.
*/
issize_t msg_comment_d(char **ss, char const **return_comment)
{
/* skip comment */
int level = 1;
char *s = *ss;
assert(s[0] == '(');
if (*s != '(')
return -1;
*s++ = '\0';
if (return_comment)
*return_comment = s;
while (level) switch (*s++) {
case '(': level++; break;
case ')': level--; break;
case '\0': /* ERROR */ return -1;
}
assert(s[-1] == ')');
s[-1] = '\0';
skip_lws(&s);
*ss = s;
return 0;
}
/** Parse a quoted string */
issize_t msg_quoted_d(char **ss, char **return_quoted)
{
char *s= *ss, *s0 = s;
ssize_t n = span_quoted(s);
if (n <= 0)
return -1;
*return_quoted = s;
s += n;
if (IS_LWS(*s)) {
*s++ = '\0';
skip_lws(&s); /* skip linear whitespace */
}
*ss = s;
return s - s0;
}
#if 0
/** Calculate length of string when quoted. */
int msg_quoted_len(char const *u)
{
int rv;
if (!u)
return 0;
rv = span_token_lws(u);
if (u[rv]) {
/* We need to quote string */
int n;
int extra = 2; /* quote chars */
/* Find all characters to quote */
for (n = strcspn(u + rv, "\\\""); u[rv + n]; rv += n)
extra++;
rv += extra;
}
return rv;
}
#endif
/**Parse @e host[":"port] pair.
*
* Parses a @e host[":"port] pair. The caller passes function a pointer to a
* string via @a ss, and pointers to which parsed host and port are assigned
* via @a hhost and @a pport, respectively. The value-result parameter @a
* *pport must be initialized to default value (e.g., NULL).
*
* @param ss pointer to pointer to string to be parsed
* @param return_host value-result parameter for @e host
* @param return_port value-result parameter for @e port
* @return
* Returns zero when successful, and a negative value upon an error. The
* parsed values for host and port are assigned via @a return_host and @a
* return_port, respectively. The function also updates the pointer @a *ss,
* so if call is successful, the @a *ss points to first
* non-linear-whitespace character after @e host[":"port] pair.
*
* @note
* If there is no whitespace after @e port, the value in @a *pport may not be
* NUL-terminated. The calling function @b must NUL terminate the value by
* setting the @a **ss to NUL after first examining its value.
*/
int msg_hostport_d(char **ss,
char const **return_host,
char const **return_port)
{
char *host, *s = *ss;
char *port = NULL;
/* Host name */
host = s;
if (s[0] != '[') {
skip_token(&s); if (host == s) return -1;
}
else {
/* IPv6 */
size_t n = strspn(++s, HEX ":.");
if (s[n] != ']') return -1;
s += n + 1;
}
if (IS_LWS(*s)) { *s++ = '\0'; skip_lws(&s); }
if (s[0] == ':') {
unsigned long nport;
*s++ = '\0'; skip_lws(&s);
if (!IS_DIGIT(*s))
return -1;
port = s;
nport = strtoul(s, &s, 10);
if (nport > 65535)
return -1;
if (IS_LWS(*s)) {
*s++ = '\0';
skip_lws(&s);
}
}
*return_host = host;
*return_port = port;
*ss = s;
return 0;
}
/** Find a header parameter.
*
* Searches for given parameter @a name from the header. If parameter is
* found, it returns a non-NULL pointer to the parameter value. If there is
* no value for the name (in form "name" or "name=value"), the returned pointer
* points to a NUL character.
*
* @param h pointer to header structure
* @param name parameter name (with or without "=" sign)
*
* @return
* A pointer to parameter value, or NULL if parameter was not found.
*/
char const *msg_header_find_param(msg_common_t const *h, char const *name)
{
if (h && h->h_class->hc_params) {
msg_param_t const **params = (msg_param_t const **)
((char *)h + h->h_class->hc_params);
return msg_params_find(*params, name);
}
return NULL;
}
/**Modify a parameter value or list item in a header.
*
* A header parameter @a param can be just a C-string (@a is_item > 0), or
* it can have internal format <i>name [ "=" value]</i>. In the latter case,
* the value part following = is ignored when replacing or removing the
* parameter.
*
* @param home memory home used to re-allocate parameter list in header
* @param h pointer to a header
* @param param parameter to be replaced or added
* @param is_item how to interpret @a param:
* - 1 case-sensitive, no structure
* - 0 case-insensitive, <i>name [ "=" value ]</i>
* @param remove_replace_add what operation to do:
* - -1 remove
* - 0 replace
* - 1 add
*
* @retval 1 if parameter was replaced or removed successfully
* @retval 0 if parameter was added successfully,
* or there was nothing to remove
* @retval -1 upon an error
*/
static
int msg_header_param_modify(su_home_t *home, msg_common_t *h,
char const *param,
int is_item,
int remove_replace_add)
{
msg_param_t *params, **pointer_to_params;
size_t plen, n;
if (!h || !h->h_class->hc_params || !param)
return -1;
pointer_to_params = (msg_param_t **)((char *)h + h->h_class->hc_params);
params = *pointer_to_params;
plen = is_item > 0 ? strlen(param) : strcspn(param, "=");
n = 0;
if (params) {
/* Existing list, try to replace or remove */
for (; params[n]; n++) {
char const *maybe = params[n];
if (remove_replace_add > 0)
continue;
if (is_item > 0) {
if (strcmp(maybe, param) == 0) {
if (remove_replace_add == 0)
return 1;
}
}
else {
if (su_casenmatch(maybe, param, plen) &&
(maybe[plen] == '=' || maybe[plen] == 0))
break;
}
}
}
/* Not found? */
if (!params || !params[n]) {
if (remove_replace_add < 0)
return 0; /* Nothing to remove */
else
remove_replace_add = 1; /* Add instead of replace */
}
if (remove_replace_add < 0) { /* Remove */
for (; params[n]; n++)
params[n] = params[n + 1];
}
else {
if (remove_replace_add > 0) { /* Add */
size_t m_before = MSG_PARAMS_NUM(n + 1);
size_t m_after = MSG_PARAMS_NUM(n + 2);
assert(!params || !params[n]);
if (m_before != m_after || !params) {
msg_param_t *p;
/* XXX - we should know when to do realloc */
p = su_alloc(home, m_after * sizeof(*p));
if (!p) return -1;
if (n > 0)
memcpy(p, params, n * sizeof(p[0]));
*pointer_to_params = params = p;
}
params[n + 1] = NULL;
}
params[n] = param; /* Add .. or replace */
}
msg_fragment_clear(h);
if (h->h_class->hc_update) {
/* Update shortcuts */
size_t namelen;
char const *name, *value;
name = param;
namelen = strcspn(name, "=");
if (remove_replace_add < 0)
value = NULL;
else
value = param + namelen + (name[namelen] == '=');
h->h_class->hc_update(h, name, namelen, value);
}
return remove_replace_add <= 0; /* 0 when added, 1 otherwise */
}
/** Add a parameter to a header.
*
* You should use this function only when the header accepts multiple
* parameters (or list items) with the same name. If that is not the case,
* you should use msg_header_replace_param().
*
* @note This function @b does @b not duplicate @p param. The caller should
* have allocated the @a param from the memory home associated with header
* @a h.
*
* The possible shortcuts to parameter values are updated. For example, the
* "received" parameter in @Via header has shortcut in structure #sip_via_t,
* the @ref sip_via_s::v_received "v_received" field. The shortcut is usully
* a pointer to the parameter value. If the parameter was
* "received=127.0.0.1" the @ref sip_via_s::v_received "v_received" field
* would be a pointer to "127.0.0.1". If the parameter was "received=" or
* "received", the shortcut would be a pointer to an empty string, "".
*
* @param home memory home used to re-allocate parameter list in header
* @param h pointer to a header
* @param param parameter to be replaced or added
*
* @retval 0 if parameter was added
* @retval -1 upon an error
*
* @sa msg_header_replace_param(), msg_header_remove_param(),
* msg_header_update_params(),
* #msg_common_t, #msg_header_t,
* #msg_hclass_t, msg_hclass_t::hc_params, msg_hclass_t::hc_update
*/
int msg_header_add_param(su_home_t *home, msg_common_t *h, char const *param)
{
return msg_header_param_modify(home, h, param,
0 /* case-insensitive name=value */,
1 /* add */);
}
/** Replace or add a parameter to a header.
*
* A header parameter @a param is a string of format <i>name "=" value</i>
* or just name. The value part following "=" is ignored when selecting a
* parameter to replace.
*
* @note This function @b does @b not duplicate @p param. The caller should
* have allocated the @a param from the memory home associated with header
* @a h.
*
* The possible shortcuts to parameter values are updated. For example, the
* "received" parameter in @Via header has shortcut in structure #sip_via_t,
* the @ref sip_via_s::v_received "v_received" field.
*
* @param home memory home used to re-allocate parameter list in header
* @param h pointer to a header
* @param param parameter to be replaced or added
*
* @retval 0 if parameter was added
* @retval 1 if parameter was replaced
* @retval -1 upon an error
*
* @sa msg_header_add_param(), msg_header_remove_param(),
* msg_header_update_params(),
* #msg_common_t, #msg_header_t,
* #msg_hclass_t, msg_hclass_t::hc_params, msg_hclass_t::hc_update
*/
int msg_header_replace_param(su_home_t *home,
msg_common_t *h,
char const *param)
{
return msg_header_param_modify(home, h, param,
0 /* case-insensitive name=value */,
0 /* replace */);
}
/** Remove a parameter from header.
*
* The parameter name is given as token optionally followed by "=" sign and
* value. The "=" and value after it are ignored when selecting a parameter
* to remove.
*
* The possible shortcuts to parameter values are updated. For example, the
* "received" parameter in @Via header has shortcut in structure #sip_via_t,
* the @ref sip_via_s::v_received "v_received" field. The shortcut to
* removed parameter would be set to NULL.
*
* @param h pointer to a header
* @param name name of parameter to be removed
*
* @retval 1 if a parameter was removed
* @retval 0 if no parameter was not removed
* @retval -1 upon an error
*
* @sa msg_header_add_param(), msg_header_replace_param(),
* msg_header_update_params(),
* #msg_common_t, #msg_header_t,
* #msg_hclass_t, msg_hclass_t::hc_params, msg_hclass_t::hc_update
*/
int msg_header_remove_param(msg_common_t *h, char const *name)
{
return msg_header_param_modify(NULL, h, name,
0 /* case-insensitive name=value */,
-1 /* remove */);
}
/** Update shortcuts to parameter values.
*
* Update the shortcuts to parameter values in parameter list. For example,
* the "received" parameter in @Via header has shortcut in structure
* #sip_via_t, the @ref sip_via_s::v_received "v_received" field. The
* shortcut is usully a pointer to the parameter value. If the parameter was
* "received=127.0.0.1" the @ref sip_via_s::v_received "v_received" field
* would be a pointer to "127.0.0.1". If the parameter was "received=" or
* "received", the shortcut would be a pointer to an empty string, "".
*
* @retval 0 when successful
* @retval -1 upon an error
*
* @sa msg_header_add_param(), msg_header_replace_param(),
* msg_header_update_params(),
* #msg_common_t, #msg_header_t,
* #msg_hclass_t, msg_hclass_t::hc_params, msg_hclass_t::hc_update
*/
int msg_header_update_params(msg_common_t *h, int clear)
{
msg_hclass_t *hc;
unsigned char offset;
msg_update_f *update;
int retval;
msg_param_t const *params;
size_t n;
char const *p, *v;
if (h == NULL)
return errno = EFAULT, -1;
hc = h->h_class; offset = hc->hc_params; update = hc->hc_update;
if (offset == 0 || update == NULL)
return 0;
if (clear)
update(h, NULL, 0, NULL);
params = *(msg_param_t **)((char *)h + offset);
if (params == NULL)
return 0;
retval = 0;
for (p = *params; p; p = *++params) {
n = strcspn(p, "=");
v = p + n + (p[n] == '=');
if (update(h, p, n, v) < 0)
retval = -1;
}
return retval;
}
/** Find a header item.
*
* Searches for given item @a name from the header. If item is found, the
* function returns a non-NULL pointer to the item.
*
* @param h pointer to header structure
* @param item item
*
* @return
* A pointer to item, or NULL if it was not found.
*
* @since New in @VERSION_1_12_4
*
* @sa msg_header_replace_item(), msg_header_remove_item(),
* @Allow, @AllowEvents
*/
char const *msg_header_find_item(msg_common_t const *h, char const *item)
{
if (h && h->h_class->hc_params) {
char const * const * items =
*(char const * const * const *)
((char *)h + h->h_class->hc_params);
if (items)
for (; *items; items++) {
if (strcmp(item, *items) == 0) {
return *items;
}
}
}
return NULL;
}
/**Add an item to a header.
*
* This function treats a #msg_header_t as set of C strings. The @a item is
* a C string. If no identical string is found from the list, it is added to
* the list.
*
* The shortcuts, if any, to item values are updated accordingly.
*
* @param home memory home used to re-allocate list in header
* @param h pointer to a header
* @param item item to be removed
*
* @retval 0 if item was added
* @retval 1 if item was replaced
* @retval -1 upon an error
*
* @since New in @VERSION_1_12_4.
*
* @sa msg_header_remove_item(), @Allow, @AllowEvents,
* msg_header_replace_param(), msg_header_remove_param(),
* #msg_common_t, #msg_header_t, #msg_list_t
* #msg_hclass_t, msg_hclass_t::hc_params
*/
int msg_header_replace_item(su_home_t *home,
msg_common_t *h,
char const *item)
{
return msg_header_param_modify(home, h, item,
1 /* string item */,
0 /* replace */);
}
/**Remove an item from a header.
*
* This function treats a #msg_header_t as set of C strings. The @a item is a
* C string. If identical string is found from the list, it is removed.
*
* The shortcuts, if any, to item values are updated accordingly.
*
* @param h pointer to a header
* @param name item to be removed
*
* @retval 0 if item was added
* @retval 1 if item was replaced
* @retval -1 upon an error
*
* @since New in @VERSION_1_12_4.
*
* @sa msg_header_replace_item(), @Allow, @AllowEvents,
* msg_header_replace_param(), msg_header_remove_param(),
* #msg_common_t, #msg_header_t, #msg_list_t
* #msg_hclass_t, msg_hclass_t::hc_params
*/
int msg_header_remove_item(msg_common_t *h, char const *name)
{
return msg_header_param_modify(NULL, h, name,
1 /* item */,
-1 /* remove */);
}
/** Find a parameter from a parameter list.
*
* Searches for given parameter @a token from the parameter list. If
* parameter is found, it returns a non-NULL pointer to the parameter value.
* If there is no value for the parameter (the parameter is of form "name"
* or "name="), the returned pointer points to a NUL character.
*
* @param params list (or vector) of parameters
* @param token parameter name (with or without "=" sign)
*
* @return
* A pointer to parameter value, or NULL if parameter was not found.
*/
msg_param_t msg_params_find(msg_param_t const params[], msg_param_t token)
{
if (params && token) {
size_t i, n = strcspn(token, "=");
assert(n > 0);
for (i = 0; params[i]; i++) {
msg_param_t param = params[i];
if (su_casenmatch(param, token, n)) {
if (param[n] == '=')
return param + n + 1;
else if (param[n] == 0)
return param + n;
}
}
}
return NULL;
}
/** Find a slot for parameter from a parameter list.
*
* Searches for given parameter @a token from the parameter list. If
* parameter is found, it returns a non-NULL pointer to the item containing
* the parameter.
*
* @param params list (or vector) of parameters
* @param token parameter name (with or without "=" sign)
*
* @return
* A pointer to parameter slot, or NULL if parameter was not found.
*/
msg_param_t *msg_params_find_slot(msg_param_t params[], msg_param_t token)
{
if (params && token) {
int i;
size_t n = strlen(token);
assert(n > 0);
for (i = 0; params[i]; i++) {
msg_param_t param = params[i];
if (su_casenmatch(param, token, n)) {
if (param[n] == '=')
return params + i;
else if (param[n] == 0 || token[n - 1] == '=')
return params + i;
}
}
}
return NULL;
}
/** Replace or add a parameter from a list.
*
* A non-NULL parameter list must have been created by msg_params_d()
* or by msg_params_dup().
*
* @note This function does not duplicate @p param.
*
* @param home memory home
* @param inout_params pointer to pointer to parameter list
* @param param parameter to be replaced or added
*
* @retval 0 if parameter was added
* @retval 1 if parameter was replaced
* @retval -1 upon an error
*/
int msg_params_replace(su_home_t *home,
msg_param_t **inout_params,
msg_param_t param)
{
msg_param_t *params;
size_t i, n;
assert(inout_params);
if (param == NULL || param[0] == '=' || param[0] == '\0')
return -1;
params = *inout_params;
n = strcspn(param, "=");
if (params) {
/* Existing list, try to replace or remove */
for (i = 0; params[i]; i++) {
msg_param_t maybe = params[i];
if (su_casenmatch(maybe, param, n)) {
if (maybe[n] == '=' || maybe[n] == 0) {
params[i] = param;
return 1;
}
}
}
}
/* Not found on list */
return msg_params_add(home, inout_params, param);
}
/** Remove a parameter from a list.
*
* @retval 1 if parameter was removed
* @retval 0 if parameter was not found
* @retval -1 upon an error
*/
int msg_params_remove(msg_param_t *params, msg_param_t param)
{
size_t i, n;
if (!params || !param || !param[0])
return -1;
n = strcspn(param, "=");
assert(n > 0);
for (i = 0; params[i]; i++) {
msg_param_t maybe = params[i];
if (su_casenmatch(maybe, param, n)) {
if (maybe[n] == '=' || maybe[n] == 0) {
/* Remove */
do {
params[i] = params[i + 1];
} while (params[i++]);
return 1;
}
}
}
return 0;
}
/** Calculate number of parameters in a parameter list */
size_t msg_params_length(char const * const * params)
{
size_t len;
if (!params)
return 0;
for (len = 0; params[len]; len++)
;
return len;
}
/**
* Add a parameter to a list.
*
* Add a parameter to the list; the list must have been created by @c
* msg_params_d() or by @c msg_params_dup() (or it may contain only @c
* NULL).
*
* @note This function does not duplicate @p param.
*
* @param home memory home
* @param inout_params pointer to pointer to parameter list
* @param param parameter to be added
*
* @retval 0 if parameter was added
* @retval -1 upon an error
*/
int msg_params_add(su_home_t *home,
msg_param_t **inout_params,
msg_param_t param)
{
size_t n, m_before, m_after;
msg_param_t *p = *inout_params;
if (param == NULL)
return -1;
/* Count number of parameters */
for (n = 0; p && p[n]; n++)
;
m_before = MSG_PARAMS_NUM(n + 1);
m_after = MSG_PARAMS_NUM(n + 2);
if (m_before != m_after || !p) {
p = su_alloc(home, m_after * sizeof(*p));
assert(p); if (!p) return -1;
if (n)
memcpy(p, *inout_params, n * sizeof(*p));
*inout_params = p;
}
p[n] = param;
p[n + 1] = NULL;
return 0;
}
static
int msg_param_prune(msg_param_t const d[], msg_param_t p, unsigned prune)
{
size_t i, nlen;
if (prune == 1)
nlen = strcspn(p, "=");
else
nlen = 0;
for (i = 0; d[i]; i++) {
if ((prune == 1 &&
su_casenmatch(p, d[i], nlen)
&& (d[i][nlen] == '=' || d[i][nlen] == '\0'))
||
(prune == 2 && su_casematch(p, d[i]))
||
(prune == 3 && strcmp(p, d[i]) == 0))
return 1;
}
return 0;
}
/**Join two parameter lists.
*
* The function @c msg_params_join() joins two parameter lists; the
* first list must have been created by @c msg_params_d() or by @c
* msg_params_dup() (or it may contain only @c NULL).
*
* @param home memory home
* @param dst pointer to pointer to destination parameter list
* @param src source list
* @param prune prune duplicates
* @param dup duplicate parameters in src list
*
* @par Pruning
* <table>
* <tr><td>0<td>do not prune</tr>
* <tr><td>1<td>prune parameters with identical names</tr>
* <tr><td>2<td>case-insensitive values</tr>
* <tr><td>3<td>case-sensitive values</tr>
* </table>
*
* @return
* @retval >= 0 when successful
* @retval -1 upon an error
*/
issize_t msg_params_join(su_home_t *home,
msg_param_t **dst,
msg_param_t const *src,
unsigned prune,
int dup)
{
size_t n, m, n_before, n_after, pruned, total = 0;
msg_param_t *d = *dst;
if (prune > 3)
return -1;
if (src == NULL || *src == NULL)
return 0;
/* Count number of parameters */
for (n = 0; d && d[n]; n++)
;
n_before = MSG_PARAMS_NUM(n + 1);
for (m = 0, pruned = 0; src[m]; m++) {
if (n > 0 && prune > 0 && msg_param_prune(d, src[m], prune)) {
pruned++;
if (prune > 1)
continue;
}
if (dup)
total += strlen(src[m]) + 1;
}
n_after = MSG_PARAMS_NUM(n + m - pruned + 1);
if (n_before != n_after || !d) {
d = su_alloc(home, n_after * sizeof(*d));
assert(d); if (!d) return -1;
if (n)
memcpy(d, *dst, n * sizeof(*d));
*dst = d;
}
for (m = 0; src[m]; m++) {
if (pruned && msg_param_prune(d, src[m], prune)) {
pruned--;
if (prune > 1)
continue;
}
if (dup)
d[n++] = su_strdup(home, src[m]); /* XXX */
else
d[n++] = src[m];
}
d[n] = NULL;
return 0;
}
/**Join header item lists.
*
* Join items from a source header to the destination header. The item are
* compared with the existing ones. If a match is found, it is not added to
* the list. If @a duplicate is true, the entries are duplicated while they
* are added to the list.
*
* @param home memory home
* @param dst destination header
* @param src source header
* @param duplicate if true, allocate and copy items that are added
*
* @return
* @retval >= 0 when successful
* @retval -1 upon an error
*
* @NEW_1_12_5.
*/
int msg_header_join_items(su_home_t *home,
msg_common_t *dst,
msg_common_t const *src,
int duplicate)
{
size_t N, m, M, i, n_before, n_after, total;
char *dup = NULL;
msg_param_t *d, **dd, *s;
msg_param_t t, *temp, temp0[16];
size_t *len, len0[(sizeof temp0)/(sizeof temp0[0])];
msg_update_f *update = NULL;
if (dst == NULL || dst->h_class->hc_params == 0 ||
src == NULL || src->h_class->hc_params == 0)
return -1;
s = *(msg_param_t **)((char *)src + src->h_class->hc_params);
if (s == NULL)
return 0;
for (M = 0; s[M]; M++)
{}
if (M == 0)
return 0;
if (M <= (sizeof temp0) / (sizeof temp0[0])) {
temp = temp0, len = len0;
}
else {
temp = malloc(M * (sizeof *temp) + M * (sizeof *len));
if (!temp) return -1;
len = (void *)(temp + M);
}
dd = (msg_param_t **)((char *)dst + dst->h_class->hc_params);
d = *dd;
for (N = 0; d && d[N]; N++)
{}
for (m = 0, M = 0, total = 0; s[m]; m++) {
t = s[m];
for (i = 0; i < N; i++)
if (strcmp(t, d[i]) == 0)
break;
if (i < N)
continue;
for (i = 0; i < M; i++)
if (strcmp(t, temp[i]) == 0)
break;
if (i < M)
continue;
temp[M] = t;
len[M] = strlen(t);
total += len[M++] + 1;
}
if (M == 0)
goto success;
dup = su_alloc(home, total); if (!dup) goto error;
n_before = MSG_PARAMS_NUM(N + 1);
n_after = MSG_PARAMS_NUM(N + M + 1);
if (d == NULL || n_before != n_after) {
d = su_alloc(home, n_after * sizeof(*d)); if (!d) goto error;
if (N)
memcpy(d, *dd, N * sizeof(*d));
*dd = d;
}
update = dst->h_class->hc_update;
for (m = 0; m < M; m++) {
d[N++] = memcpy(dup, temp[m], len[m] + 1);
if (update)
update(dst, dup, len[m], dup + len[m]);
dup += len[m] + 1;
}
d[N] = NULL;
success:
if (temp != temp0)
free(temp);
return 0;
error:
if (temp != temp0)
free(temp);
su_free(home, dup);
return -1;
}
/**Compare parameter lists.
*
* Compares parameter lists.
*
* @param a pointer to a parameter list
* @param b pointer to a parameter list
*
* @retval an integer less than zero if @a is less than @a b
* @retval an integer zero if @a match with @a b
* @retval an integer greater than zero if @a is greater than @a b
*/
int msg_params_cmp(char const * const a[], char const * const b[])
{
int c;
size_t nlen;
if (a == NULL || b == NULL)
return (a != NULL) - (b != NULL);
for (;;) {
if (*a == NULL || *b == NULL)
return (*a != NULL) - (*b != NULL);
nlen = strcspn(*a, "=");
if ((c = su_strncasecmp(*a, *b, nlen)))
return c;
if ((c = strcmp(*a + nlen, *b + nlen)))
return c;
a++, b++;
}
}
/** Unquote string
*
* Duplicates the string @a q in unquoted form.
*/
char *msg_unquote_dup(su_home_t *home, char const *q)
{
char *d;
size_t total, n, m;
/* First, easy case */
if (q[0] == '"')
q++;
n = strcspn(q, "\"\\");
if (q[n] == '\0' || q[n] == '"')
return su_strndup(home, q, n);
/* Hairy case - backslash-escaped chars */
total = n;
for (;;) {
if (q[n] == '\0' || q[n] == '"' || q[n + 1] == '\0')
break;
m = strcspn(q + n + 2, "\"\\");
total += m + 1;
n += m + 2;
}
if (!(d = su_alloc(home, total + 1)))
return NULL;
for (n = 0;;) {
m = strcspn(q, "\"\\");
memcpy(d + n, q, m);
n += m, q += m;
if (q[0] == '\0' || q[0] == '"' || q[1] == '\0')
break;
d[n++] = q[1];
q += 2;
}
assert(total == n);
d[n] = '\0';
return d;
}
/** Unquote string */
char *msg_unquote(char *dst, char const *s)
{
int copy = dst != NULL;
char *d = dst;
if (*s++ != '"')
return NULL;
for (;;) {
size_t n = strcspn(s, "\"\\");
if (copy)
memmove(d, s, n);
s += n;
d += n;
if (*s == '\0')
return NULL;
else if (*s == '"') {
if (copy) *d = '\0';
return dst;
}
else {
/* Copy quoted char */
if ((copy ? (*d++ = *++s) : *++s) == '\0')
return NULL;
s++;
}
}
}
/** Quote string */
issize_t msg_unquoted_e(char *b, isize_t bsiz, char const *s)
{
isize_t e = 0;
if (b == NULL)
bsiz = 0;
if (0 < bsiz)
*b = '"';
e++;
for (;*s;) {
size_t n = strcspn(s, "\"\\");
if (n == 0) {
if (e + 2 <= bsiz)
b[e] = '\\', b[e + 1] = s[0];
e += 2;
s++;
}
else {
if (e + n <= bsiz)
memcpy(b + e, s, n);
e += n;
s += n;
}
}
if (e < bsiz)
b[e] = '"';
e++;
return e;
}
/** Calculate a simple hash over a string. */
unsigned long msg_hash_string(char const *id)
{
unsigned long hash = 0;
if (id)
for (; *id; id++) {
hash += (unsigned)*id;
hash *= 38501U;
}
else
hash *= 38501U;
if (hash == 0)
hash = (unsigned long)-1;
return hash;
}
/** Calculate the size of a duplicate of a header structure. */
isize_t msg_header_size(msg_header_t const *h)
{
if (h == NULL || h == MSG_HEADER_NONE)
return 0;
else
return h->sh_class->hc_dxtra(h, h->sh_class->hc_size);
}
/** Encode a message to the buffer.
*
* The function msg_encode_e encodes a message to a given buffer.
* It returns the length of the message to be encoded, even if the
* buffer is too small (just like snprintf() is supposed to do).
*
* @param b buffer (may be NULL)
* @param size size of buffer
* @param mo public message structure (#sip_t, #http_t)
* @param flags see #
*/
issize_t msg_object_e(char b[], isize_t size, msg_pub_t const *mo, int flags)
{
size_t rv = 0;
ssize_t n;
msg_header_t const *h;
if (mo->msg_request)
h = mo->msg_request;
else
h = mo->msg_status;
for (; h; h = h->sh_succ) {
n = msg_header_e(b, size, h, flags);
if (n < 0)
return -1;
if ((size_t)n < size)
b += n, size -= n;
else
b = NULL, size = 0;
rv += n;
}
return rv;
}
/** Encode header contents. */
issize_t msg_header_field_e(char b[], isize_t bsiz, msg_header_t const *h, int flags)
{
assert(h); assert(h->sh_class);
return h->sh_class->hc_print(b, bsiz, h, flags);
}
/** Get offset of header @a h from structure @a mo. */
msg_header_t **
msg_header_offset(msg_t const *msg, msg_pub_t const *mo, msg_header_t const *h)
{
if (h == NULL || h->sh_class == NULL)
return NULL;
return msg_hclass_offset(msg->m_class, mo, h->sh_class);
}
/**Get a header from the public message structure.
*
* Gets a pointer to header from a message structure.
*
* @param pub public message structure from which header is obtained
* @param hc header class
*/
msg_header_t *
msg_header_access(msg_pub_t const *pub, msg_hclass_t *hc)
{
msg_header_t * const * hh;
if (pub == NULL || hc == NULL)
return NULL;
hh = msg_hclass_offset((void *)pub->msg_ident, (msg_pub_t *)pub, hc);
if (hh)
return *hh;
else
return NULL;
}
#include <sofia-sip/su_uniqueid.h>
/** Generates a random token.
*
*/
issize_t msg_random_token(char token[], isize_t tlen,
void const *rmemp, isize_t rsize)
{
uint32_t random = 0, rword;
uint8_t rbyte;
uint8_t const *rmem = rmemp;
size_t i;
ssize_t n;
static char const token_chars[33] =
/* Create aesthetically pleasing raNDom capS LooK */
"aBcDeFgHjKmNpQrStUvXyZ0123456789";
if (rmem == NULL && rsize == 0)
rsize = UINT_MAX;
if (rsize == 0) {
if (token && tlen > 0)
strcpy(token, "+");
return 1;
}
if (token == NULL) {
if (rsize >= tlen * 5 / 8)
return tlen;
else
return rsize / 5 * 8;
}
for (i = 0, n = 0; i < tlen;) {
if (n < 5) {
if (rsize == 0)
;
else if (rmem) {
rbyte = *rmem++, rsize--;
random = random | (rbyte << n);
n += 8;
} else {
rword = su_random();
random = (rword >> 13) & 31;
n = 6;
}
}
token[i] = token_chars[random & 31];
random >>= 5;
i++, n -= 5;
if (n < 0 || (n == 0 && rsize == 0))
break;
}
token[i] = 0;
return i;
}
/** Parse a message.
*
* Parse a text message with parser @a mc. The @a data is copied and it is
* not modified or referenced by the parsed message.
*
* @par Example
* Parse a SIP message fragment (e.g., payload of NOTIFY sent in response to
* REFER):
* @code
* msg_t *m = msg_make(sip_default_mclass(), 0, pl->pl_data, pl->pl_len);
* sip_t *frag = sip_object(m);
* @endcode
*
* @param mc message class (parser table)
* @param flags message flags (see #msg_flg_user)
* @param data message text
* @param len size of message text (if -1, use strlen(data))
*
* @retval A pointer to a freshly allocated and parsed message.
*
* Upon parsing error, the header structure may be left incomplete. The
* #MSG_FLG_ERROR is set in @a msg_object(msg)->msg_flags.
*
* @since New in @VERSION_1_12_4
*
* @sa msg_as_string(), msg_extract()
*/
msg_t *msg_make(msg_mclass_t const *mc, int flags,
void const *data, ssize_t len)
{
msg_t *msg;
msg_iovec_t iovec[2];
if (len == -1)
len = strlen(data);
if (len == 0)
return NULL;
msg = msg_create(mc, flags);
if (msg == NULL)
return NULL;
su_home_preload(msg_home(msg), 1, len + 1024);
if (msg_recv_iovec(msg, iovec, 2, len, 1) < 0) {
perror("msg_recv_iovec");
}
assert((ssize_t)iovec->mv_len == len);
memcpy(iovec->mv_base, data, len);
msg_recv_commit(msg, len, 1);
if (msg_extract(msg) < 0)
msg->m_object->msg_flags |= MSG_FLG_ERROR;
return msg;
}