/* * This file is part of the Sofia-SIP package * * Copyright (C) 2005 Nokia Corporation. * * Contact: Pekka Pessi * * 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 * */ /**@CFILE sip_event.c * @brief Event SIP headers. * * Implementation of header classes for event-related SIP headers @Event, * @AllowEvents, and @SubscriptionState. * * @author Pekka Pessi . * * @date Created: Thu Sep 13 21:24:15 EEST 2001 ppessi */ #include "config.h" /* Avoid casting sip_t to msg_pub_t and sip_header_t to msg_header_t */ #define MSG_PUB_T struct sip_s #define MSG_HDR_T union sip_header_u #include "sofia-sip/sip_parser.h" #include #include #include #include #include #include /* ====================================================================== */ /**@SIP_HEADER sip_event Event Header * * The Event header is used to indicate the which event or class of events * the message contains or subscribes. Its syntax is defined in @RFC3265 as * follows: * * @code * Event = ( "Event" / "o" ) HCOLON event-type * *( SEMI event-param ) * event-type = event-package *( "." event-template ) * event-package = token-nodot * event-template = token-nodot * token-nodot = 1*( alphanum / "-" / "!" / "%" / "*" * / "_" / "+" / "`" / "'" / "~" ) * event-param = generic-param / ( "id" EQUAL token ) * @endcode * * The parsed Event header is stored in #sip_event_t structure. */ /**@ingroup sip_event * @typedef struct sip_event_s sip_event_t; * * The structure #sip_event_t contains representation of an @Event header. * * The #sip_event_t is defined as follows: * @code * typedef struct sip_event_s * { * sip_common_t o_common; // Common fragment info * sip_error_t *o_next; // Link to next (dummy) * char const * o_type; // Event type * msg_param_t const *o_params; // List of parameters * char const *o_id; // Event ID * } sip_event_t; * @endcode */ static msg_xtra_f sip_event_dup_xtra; static msg_dup_f sip_event_dup_one; static msg_update_f sip_event_update; msg_hclass_t sip_event_class[] = SIP_HEADER_CLASS(event, "Event", "o", o_params, single, event); issize_t sip_event_d(su_home_t *home, sip_header_t *h, char *s, isize_t slen) { sip_event_t *o = h->sh_event; size_t n; n = span_token(s); if (n == 0) return -1; o->o_type = s; s += n; while (IS_LWS(*s)) { *s++ = '\0'; } if (*s == ';') { if (msg_params_d(home, &s, &o->o_params) < 0 || *s) return -1; msg_header_update_params(o->o_common, 0); } return 0; } issize_t sip_event_e(char b[], isize_t bsiz, sip_header_t const *h, int f) { char *end = b + bsiz, *b0 = b; sip_event_t const *o = h->sh_event; assert(sip_is_event(h)); MSG_STRING_E(b, end, o->o_type); MSG_PARAMS_E(b, end, o->o_params, flags); return b - b0; } isize_t sip_event_dup_xtra(sip_header_t const *h, isize_t offset) { sip_event_t const *o = h->sh_event; MSG_PARAMS_SIZE(offset, o->o_params); offset += MSG_STRING_SIZE(o->o_type); return offset; } /** Duplicate one #sip_event_t object */ char *sip_event_dup_one(sip_header_t *dst, sip_header_t const *src, char *b, isize_t xtra) { sip_event_t *o_dst = dst->sh_event; sip_event_t const *o_src = src->sh_event; char *end = b + xtra; b = msg_params_dup(&o_dst->o_params, o_src->o_params, b, xtra); MSG_STRING_DUP(b, o_dst->o_type, o_src->o_type); assert(b <= end); (void)end; return b; } /** Update parameters in @Event header. */ static int sip_event_update(msg_common_t *h, char const *name, isize_t namelen, char const *value) { sip_event_t *o = (sip_event_t *)h; if (name == NULL) { o->o_id = NULL; } else if (namelen == strlen("id") && su_casenmatch(name, "id", namelen)) { o->o_id = value; } return 0; } /* ====================================================================== */ /**@SIP_HEADER sip_allow_events Allow-Events Header * * The Allow-Events header is used to indicate which events or classes of * events the notifier supports. Its syntax is defined in @RFC3265 as * follows: * * @code * Allow-Events = ( "Allow-Events" / "u" ) HCOLON event-type * *(COMMA event-type) * @endcode * * The parsed Allow-Events header is stored in #sip_allow_events_t structure. * * Note that the event name is case-sensitive. The event "Presence" is * different from "presence". However, it is very unwise to use such event * names. * * @sa @Event, @RFC3265, msg_header_find_item(), msg_header_replace_item(), * msg_header_remove_item() */ /**@ingroup sip_allow_events * @typedef struct msg_list_s sip_allow_events_t; * * The structure #sip_allow_events_t contains representation of an * @AllowEvents header. * * The #sip_allow_events_t is defined as follows: * @code * typedef struct msg_list_s * { * msg_common_t k_common[1]; // Common fragment info * msg_list_t *k_next; // Link to next header * msg_param_t *k_items; // List of items * } sip_allow_events_t; * @endcode */ msg_hclass_t sip_allow_events_class[] = SIP_HEADER_CLASS_LIST(allow_events, "Allow-Events", "u", list); issize_t sip_allow_events_d(su_home_t *home, sip_header_t *h, char *s, isize_t slen) { return msg_list_d(home, h, s, slen); } issize_t sip_allow_events_e(char b[], isize_t bsiz, sip_header_t const *h, int f) { assert(sip_is_allow_events(h)); return msg_list_e(b, bsiz, h, f); } /** Append an event to a @AllowEvents header. * * @note This function @b does @b duplicate @p event. * * @deprecated Use msg_header_replace_item() directly. */ int sip_allow_events_add(su_home_t *home, sip_allow_events_t *ae, char const *event) { event = su_strdup(home, event); if (!event) return -1; return msg_header_replace_item(home, ae->k_common, event); } /* ====================================================================== */ /**@SIP_HEADER sip_subscription_state Subscription-State Header * * The Subscription-State header is used to indicate in which state a * subscription is. Its syntax is defined in @RFC3265 section 4.2.4 as * follows: * * @code * Subscription-State = "Subscription-State" HCOLON substate-value * *( SEMI subexp-params ) * substate-value = "active" / "pending" / "terminated" * / extension-substate * extension-substate = token * subexp-params = ("reason" EQUAL event-reason-value) * / ("expires" EQUAL delta-seconds) * / ("retry-after" EQUAL delta-seconds) * / generic-param * event-reason-value = "deactivated" * / "probation" * / "rejected" * / "timeout" * / "giveup" * / "noresource" * / event-reason-extension * event-reason-extension = token * @endcode * * The parsed Subscription-State header * is stored in #sip_subscription_state_t structure. */ /**@ingroup sip_subscription_state * @typedef struct sip_subscription_state_s sip_subscription_state_t; * * The structure #sip_subscription_state_t contains representation of an * @SubscriptionState header. * * The #sip_subscription_state_t is defined as follows: * @code * typedef struct sip_subscription_state_s * { * sip_common_t ss_common[1]; * sip_unknown_t *ss_next; * // Subscription state: "pending", "active" or "terminated" * char const *ss_substate; * msg_param_t const *ss_params; // List of parameters * char const *ss_reason; // Reason of terminating * char const *ss_expires; // Subscription lifetime in seconds * char const *ss_retry_after; // Value of retry-after parameter * } sip_subscription_state_t; * @endcode */ static msg_xtra_f sip_subscription_state_dup_xtra; static msg_dup_f sip_subscription_state_dup_one; static msg_update_f sip_subscription_state_update; msg_hclass_t sip_subscription_state_class[] = SIP_HEADER_CLASS(subscription_state, "Subscription-State", "", ss_params, single, subscription_state); issize_t sip_subscription_state_d(su_home_t *home, sip_header_t *h, char *s, isize_t slen) { sip_subscription_state_t *ss = h->sh_subscription_state; ss->ss_substate = s; s += span_token(s); /* forwards the pointer to the end of substate-value */ if (s == ss->ss_substate) return -1; if (IS_LWS(*s)) { *s = '\0'; s += span_lws(s + 1) + 1; } /* check if parameters are present and if so parse them */ if (*s == ';') { if ( msg_params_d(home, &s, &ss->ss_params) < 0) return -1; if (msg_header_update_params(ss->ss_common, 0) < 0) return -1; } return 0; } issize_t sip_subscription_state_e(char b[], isize_t bsiz, sip_header_t const *h, int flags) { char *end = b + bsiz, *b0 = b; sip_subscription_state_t const *ss = h->sh_subscription_state; assert(sip_is_subscription_state(h)); MSG_STRING_E(b, end, ss->ss_substate); MSG_PARAMS_E(b, end, ss->ss_params, flags); return b - b0; } isize_t sip_subscription_state_dup_xtra(sip_header_t const *h, isize_t offset) { sip_subscription_state_t const *ss = h->sh_subscription_state; /* Calculates memory size occupied */ MSG_PARAMS_SIZE(offset, ss->ss_params); offset += MSG_STRING_SIZE(ss->ss_substate); return offset; } /** Duplicate one #sip_subscription_state_t object */ char *sip_subscription_state_dup_one(sip_header_t *dst, sip_header_t const *src, char *b, isize_t xtra) { sip_subscription_state_t *ss_dst = dst->sh_subscription_state; sip_subscription_state_t const *ss_src = src->sh_subscription_state; char *end = b + xtra; b = msg_params_dup(&ss_dst->ss_params, ss_src->ss_params, b, xtra); MSG_STRING_DUP(b, ss_dst->ss_substate, ss_src->ss_substate); assert(b <= end); (void)end; return b; } static int sip_subscription_state_update(msg_common_t *h, char const *name, isize_t namelen, char const *value) { sip_subscription_state_t *ss = (sip_subscription_state_t *)h; if (name == NULL) { ss->ss_reason = NULL; ss->ss_retry_after = NULL; ss->ss_expires = NULL; } #define MATCH(s) (namelen == strlen(#s) && su_casenmatch(name, #s, strlen(#s))) else if (MATCH(reason)) { ss->ss_reason = value; } else if (MATCH(retry-after)) { ss->ss_retry_after = value; } else if (MATCH(expires)) { ss->ss_expires = value; } #undef MATCH return 0; } #if 0 /* More dead headers */ /* ====================================================================== */ /**@SIP_HEADER sip_publication Publication Header * * The Publication header is used to indicate the which publication or class * of publications the message contains. Its syntax is defined * in (draft-niemi-simple-publish-00.txt) as follows: * * @code * Publication = ( "Publication") HCOLON publish-package * *( SEMI publish-param ) * publish-package = token-nodot * token-nodot = 1*( alphanum / "-" / "!" / "%" / "*" * / "_" / "+" / "`" / "'" / "~" ) * publish-param = generic-param / pstream / ptype * pstream = "stream" EQUAL token * ptype = "type" EQUAL token * @endcode * * * The parsed Publication header is stored in #sip_publication_t structure. */ /**@ingroup sip_publication * @brief Structure for Publication header. */ struct sip_publication_s { sip_common_t pub_common; /**< Common fragment info */ sip_error_t *pub_next; /**< Link to next (dummy) */ char const * pub_package; /**< Publication packaage */ msg_param_t const *pub_params; /**< List of parameters */ msg_param_t pub_type; /**< Publication type */ msg_param_t pub_stream; /**< Publication stream */ }; static msg_xtra_f sip_publication_dup_xtra; static msg_dup_f sip_publication_dup_one; msg_hclass_t sip_publication_class[] = SIP_HEADER_CLASS(publication, "Publication", "", pub_params, single, publication); su_inline void sip_publication_update(sip_publication_t *pub); issize_t sip_publication_d(su_home_t *home, sip_header_t *h, char *s, isize_t slen) { sip_publication_t *pub = h->sh_publication; size_t n; n = span_token(s); if (n == 0) return -1; pub->pub_package = s; s += n; while (IS_LWS(*s)) { *s++ = '\0'; } if (*s == ';') { if (msg_params_d(home, &s, &pub->pub_params) < 0 || *s) return -1; sip_publication_update(pub); } return 0; } issize_t sip_publication_e(char b[], isize_t bsiz, sip_header_t const *h, int f) { char *end = b + bsiz, *b0 = b; sip_publication_t const *pub = h->sh_publication; assert(sip_is_publication(h)); MSG_STRING_E(b, end, pub->pub_package); MSG_PARAMS_E(b, end, pub->pub_params, flags); return b - b0; } isize_t sip_publication_dup_xtra(sip_header_t const *h, isize_t offset) { sip_publication_t const *pub = h->sh_publication; MSG_PARAMS_SIZE(offset, pub->pub_params); offset += MSG_STRING_SIZE(pub->pub_package); return offset; } /** Duplicate one #sip_publication_t object */ char *sip_publication_dup_one(sip_header_t *dst, sip_header_t const *src, char *b, isize_t xtra) { sip_publication_t *pub_dst = dst->sh_publication; sip_publication_t const *pub_src = src->sh_publication; char *end = b + xtra; b = msg_params_dup(&pub_dst->pub_params, pub_src->pub_params, b, xtra); MSG_STRING_DUP(b, pub_dst->pub_package, pub_src->pub_package); if (pub_dst->pub_params) sip_publication_update(pub_dst); assert(b <= end); return b; } su_inline void sip_publication_update(sip_publication_t *pub) { size_t i; if (pub->pub_params) for (i = 0; pub->pub_params[i]; i++) { if (su_casenmatch(pub->pub_params[i], "stream=", strlen("stream="))) pub->pub_stream = pub->pub_params[i] + strlen("stream="); else if (su_casenmatch(pub->pub_params[i], "type=", strlen("type="))) pub->pub_type = pub->pub_params[i] + strlen("type="); } } /* ====================================================================== */ /**@SIP_HEADER sip_allow_publications Allow-Publication Header * * The Allow-Publication header is used to indicate which publications or classes of * publications the server supports. Its syntax is defined in [niemi] * (draft-niemi-simple-publish-00.txt) as follows: * * @code * Allow-Publications = "Allow-Publications" HCOLON publish-type * * ( COMMA publish-type ) * @endcode * * * The parsed Allow-Publication Header * is stored in #sip_allow_publications_t structure. */ msg_hclass_t sip_allow_publications_class[] = SIP_HEADER_CLASS_LIST(allow_publications, "Allow-Publications", "", list); issize_t sip_allow_publications_d(su_home_t *home, sip_header_t *h, char *s, isize_t slen) { return msg_list_d(home, h, s, slen); } issize_t sip_allow_publications_e(char b[], isize_t bsiz, sip_header_t const *h, int f) { assert(sip_is_allow_publications(h)); return msg_list_e(b, bsiz, h, f); } /** Append an publication to a Allow-Publications header. */ int sip_allow_publications_add(su_home_t *home, sip_allow_publications_t *ae, char const *e) { e = su_strdup(home, e); if (!e) return -1; return msg_params_replace(home, (msg_param_t **)&ae->k_items, e); } #endif