/* * Copyright (c) 2012-2024 Belledonne Communications SARL. * * This file is part of belle-sip. * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program 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 General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include "belle-sip/headers.h" #include "belle-sip/parameters.h" #include "sip/sip_parser.hh" #include #include #include #include "belle_sip_internal.h" #include "listeningpoint_internal.h" /************************ * header ***********************/ GET_SET_STRING(belle_sip_header, name); #define PROTO_SIP 0x1 #define PROTO_HTTP 0x1 << 1 typedef belle_sip_header_t *(*header_parse_func)(const char *); struct header_name_func_pair { int protocol; const char *name; header_parse_func func; }; static struct header_name_func_pair header_table[] = { {PROTO_SIP, "m", (header_parse_func)belle_sip_header_contact_parse}, {PROTO_SIP, BELLE_SIP_CONTACT, (header_parse_func)belle_sip_header_contact_parse}, {PROTO_SIP, "f", (header_parse_func)belle_sip_header_from_parse}, {PROTO_SIP, BELLE_SIP_FROM, (header_parse_func)belle_sip_header_from_parse}, {PROTO_SIP, "t", (header_parse_func)belle_sip_header_to_parse}, {PROTO_SIP, BELLE_SIP_TO, (header_parse_func)belle_sip_header_to_parse}, {PROTO_SIP, "d", (header_parse_func)belle_sip_header_diversion_parse}, {PROTO_SIP, BELLE_SIP_DIVERSION, (header_parse_func)belle_sip_header_diversion_parse}, {PROTO_SIP, "i", (header_parse_func)belle_sip_header_call_id_parse}, {PROTO_SIP, BELLE_SIP_CALL_ID, (header_parse_func)belle_sip_header_call_id_parse}, {PROTO_SIP, "r", (header_parse_func)belle_sip_header_retry_after_parse}, {PROTO_SIP, BELLE_SIP_RETRY_AFTER, (header_parse_func)belle_sip_header_retry_after_parse}, {PROTO_SIP, "l", (header_parse_func)belle_sip_header_content_length_parse}, {PROTO_SIP | PROTO_HTTP, BELLE_SIP_CONTENT_LENGTH, (header_parse_func)belle_sip_header_content_length_parse}, {PROTO_SIP, "c", (header_parse_func)belle_sip_header_content_type_parse}, {PROTO_SIP | PROTO_HTTP, BELLE_SIP_CONTENT_TYPE, (header_parse_func)belle_sip_header_content_type_parse}, {PROTO_SIP, BELLE_SIP_CSEQ, (header_parse_func)belle_sip_header_cseq_parse}, {PROTO_SIP, BELLE_SIP_ROUTE, (header_parse_func)belle_sip_header_route_parse}, {PROTO_SIP, BELLE_SIP_RECORD_ROUTE, (header_parse_func)belle_sip_header_record_route_parse}, {PROTO_SIP, "v", (header_parse_func)belle_sip_header_via_parse}, {PROTO_SIP, BELLE_SIP_VIA, (header_parse_func)belle_sip_header_via_parse}, {PROTO_SIP, "x", (header_parse_func)belle_sip_header_session_expires_parse}, {PROTO_SIP, BELLE_SIP_SESSION_EXPIRES, (header_parse_func)belle_sip_header_session_expires_parse}, {PROTO_SIP, BELLE_SIP_AUTHORIZATION, (header_parse_func)belle_sip_header_authorization_parse}, {PROTO_SIP, BELLE_SIP_PROXY_AUTHORIZATION, (header_parse_func)belle_sip_header_proxy_authorization_parse}, {PROTO_SIP | PROTO_HTTP, BELLE_SIP_WWW_AUTHENTICATE, (header_parse_func)belle_sip_header_www_authenticate_parse}, {PROTO_SIP | PROTO_HTTP, BELLE_SIP_PROXY_AUTHENTICATE, (header_parse_func)belle_sip_header_proxy_authenticate_parse}, {PROTO_SIP, BELLE_SIP_MAX_FORWARDS, (header_parse_func)belle_sip_header_max_forwards_parse}, {PROTO_SIP | PROTO_HTTP, BELLE_SIP_USER_AGENT, (header_parse_func)belle_sip_header_user_agent_parse}, {PROTO_SIP, BELLE_SIP_EXPIRES, (header_parse_func)belle_sip_header_expires_parse}, {PROTO_SIP | PROTO_HTTP, BELLE_SIP_ALLOW, (header_parse_func)belle_sip_header_allow_parse}, {PROTO_SIP, BELLE_SIP_SUBSCRIPTION_STATE, (header_parse_func)belle_sip_header_subscription_state_parse}, {PROTO_SIP, BELLE_SIP_SERVICE_ROUTE, (header_parse_func)belle_sip_header_service_route_parse}, {PROTO_SIP, BELLE_SIP_REFER_TO, (header_parse_func)belle_sip_header_refer_to_parse}, {PROTO_SIP, BELLE_SIP_REFERRED_BY, (header_parse_func)belle_sip_header_referred_by_parse}, {PROTO_SIP, BELLE_SIP_REPLACES, (header_parse_func)belle_sip_header_replaces_parse}, {PROTO_SIP | PROTO_HTTP, BELLE_SIP_DATE, (header_parse_func)belle_sip_header_date_parse}, {PROTO_SIP, BELLE_SIP_P_PREFERRED_IDENTITY, (header_parse_func)belle_sip_header_p_preferred_identity_parse}, {PROTO_SIP, BELLE_SIP_PRIVACY, (header_parse_func)belle_sip_header_privacy_parse}, {PROTO_SIP, BELLE_SIP_EVENT, (header_parse_func)belle_sip_header_event_parse}, {PROTO_SIP, "o", (header_parse_func)belle_sip_header_event_parse}, {PROTO_SIP, BELLE_SIP_SUPPORTED, (header_parse_func)belle_sip_header_supported_parse}, {PROTO_SIP, "k", (header_parse_func)belle_sip_header_supported_parse}, {PROTO_SIP, BELLE_SIP_REQUIRE, (header_parse_func)belle_sip_header_require_parse}, {PROTO_SIP, BELLE_SIP_CONTENT_DISPOSITION, (header_parse_func)belle_sip_header_content_disposition_parse}, {PROTO_SIP | PROTO_HTTP, BELLE_SIP_ACCEPT, (header_parse_func)belle_sip_header_accept_parse}, {PROTO_SIP, BELLE_SIP_REASON, (header_parse_func)belle_sip_header_reason_parse}, {PROTO_SIP, BELLE_SIP_AUTHENTICATION_INFO, (header_parse_func)belle_sip_header_authentication_info_parse}}; static belle_sip_header_t *belle_header_create(const char *name, const char *value, int protocol) { size_t i; belle_sip_header_t *ret; size_t elements = sizeof(header_table) / sizeof(struct header_name_func_pair); if (!name || name[0] == '\0') { belle_sip_error("Cannot create header without name"); return NULL; } for (i = 0; i < elements; i++) { if ((header_table[i].protocol & protocol) && strcasecmp(header_table[i].name, name) == 0) { char *raw = belle_sip_strdup_printf("%s:%s", name, value); ret = header_table[i].func(raw); belle_sip_free(raw); return ret; } } /*not a known header*/ return BELLE_SIP_HEADER(belle_sip_header_extension_create(name, value)); } belle_sip_header_t *belle_sip_header_create(const char *name, const char *value) { return belle_header_create(name, value, PROTO_SIP); } belle_sip_header_t *belle_http_header_create(const char *name, const char *value) { return belle_header_create(name, value, PROTO_HTTP); } void belle_sip_header_init(belle_sip_header_t *header) { } static void belle_sip_header_clone(belle_sip_header_t *header, const belle_sip_header_t *orig) { CLONE_STRING(belle_sip_header, name, header, orig) if (belle_sip_header_get_next(orig)) { belle_sip_header_set_next( header, BELLE_SIP_HEADER(belle_sip_object_clone(BELLE_SIP_OBJECT(belle_sip_header_get_next(orig))))); } } static void belle_sip_header_destroy(belle_sip_header_t *header) { if (header->name) belle_sip_free(header->name); if (header->unparsed_value) belle_sip_free(header->unparsed_value); if (header->next) belle_sip_object_unref(BELLE_SIP_OBJECT(header->next)); } void belle_sip_header_set_next(belle_sip_header_t *header, belle_sip_header_t *next) { if (next) belle_sip_object_ref(next); if (header->next) belle_sip_object_unref(header->next); header->next = next; } void belle_sip_header_append(belle_sip_header_t *header, belle_sip_header_t *next) { auto current = header; while (current->next) current = current->next; belle_sip_header_set_next(current, next); } belle_sip_header_t *belle_sip_header_get_next(const belle_sip_header_t *header) { return header->next; } const char *belle_sip_header_get_unparsed_value(belle_sip_header_t *obj) { if (BELLE_SIP_OBJECT_IS_INSTANCE_OF(obj, belle_sip_header_extension_t)) { return belle_sip_header_extension_get_value(BELLE_SIP_HEADER_EXTENSION(obj)); } else { char *tmp = belle_sip_object_to_string(obj); char *ret; if (obj->unparsed_value) { belle_sip_free(obj->unparsed_value); obj->unparsed_value = NULL; } obj->unparsed_value = tmp; ret = tmp; ret += strlen(obj->name) + 1; /* name + semicolon*/ for (; *ret == ' '; ret++) { }; /*skip spaces*/ return ret; } } belle_sip_error_code belle_sip_header_marshal(belle_sip_header_t *header, char *buff, size_t buff_size, size_t *offset) { if (header->name) { return belle_sip_snprintf(buff, buff_size, offset, "%s: ", header->name); } else { belle_sip_warning("no header name found"); return BELLE_SIP_OK; } } BELLE_SIP_DECLARE_NO_IMPLEMENTED_INTERFACES(belle_sip_header_t); BELLE_SIP_INSTANCIATE_VPTR(belle_sip_header_t, belle_sip_object_t, belle_sip_header_destroy, belle_sip_header_clone, belle_sip_header_marshal, TRUE); BELLE_SIP_PARSE_WITH_CONTEXT_AND_RULE(header, message_header); belle_sip_header_t *belle_sip_header_new_dummy(void) { return belle_sip_header_create("dummy", "dummy"); } void belle_sip_header_set_unparsed_value(belle_sip_header_t *obj, const char *value) { if (obj->unparsed_value) belle_sip_free(obj->unparsed_value); if (value) { obj->unparsed_value = belle_sip_strdup(value); } else { obj->unparsed_value = nullptr; } } /************************ * header_address ***********************/ struct _belle_sip_header_address { belle_sip_parameters_t base; char *displayname; belle_sip_uri_t *uri; belle_generic_uri_t *absolute_uri; unsigned char automatic; }; static void belle_sip_header_address_init(belle_sip_header_address_t *object) { belle_sip_parameters_init((belle_sip_parameters_t *)object); /*super*/ } static void belle_sip_header_address_destroy(belle_sip_header_address_t *address) { if (address->displayname) belle_sip_free(address->displayname); if (address->uri) belle_sip_object_unref(address->uri); if (address->absolute_uri) belle_sip_object_unref(address->absolute_uri); } static void _belle_sip_header_address_clone(belle_sip_header_address_t *addr, const belle_sip_header_address_t *orig) { CLONE_STRING(belle_sip_header_address, displayname, addr, orig) if (belle_sip_header_address_get_uri(orig)) { belle_sip_header_address_set_uri( addr, BELLE_SIP_URI(belle_sip_object_clone(BELLE_SIP_OBJECT(belle_sip_header_address_get_uri(orig))))); } if (belle_sip_header_address_get_absolute_uri(orig)) { belle_sip_header_address_set_absolute_uri(addr, BELLE_GENERIC_URI(belle_sip_object_clone(BELLE_SIP_OBJECT( belle_sip_header_address_get_absolute_uri(orig))))); } if (belle_sip_header_address_get_automatic(orig)) { belle_sip_header_address_set_automatic(addr, belle_sip_header_address_get_automatic(orig)); } belle_sip_parameters_copy_parameters_from(&addr->base, &orig->base); } belle_sip_header_address_t *belle_sip_header_address_clone(const belle_sip_header_address_t *orig) { belle_sip_header_address_t *new_address = belle_sip_header_address_new(); _belle_sip_header_address_clone(new_address, orig); return new_address; } static belle_sip_error_code _belle_sip_header_address_marshal( belle_sip_header_address_t *header, char *buff, size_t buff_size, size_t *offset, int force_angle_quote) { belle_sip_error_code error = BELLE_SIP_OK; bool has_display_name = header->displayname != nullptr; if (belle_sip_stack_name_addr_forced) force_angle_quote = TRUE; /*1 display name*/ if (has_display_name) { char *escaped_display_name = belle_sip_display_name_to_backslashed_escaped_string(header->displayname); error = belle_sip_snprintf(buff, buff_size, offset, "\"%s\" ", escaped_display_name); belle_sip_free(escaped_display_name); if (error != BELLE_SIP_OK) return error; } if (header->uri || header->absolute_uri) { bool has_parameters = belle_sip_parameters_get_parameter_names(&header->base); bool is_sip_uri_with_headers_or_parameters = (header->uri != nullptr) && (belle_sip_parameters_get_parameter_names((belle_sip_parameters_t *)header->uri) || belle_sip_uri_get_header_names(header->uri)); bool angle_quote_required = force_angle_quote || has_display_name || is_sip_uri_with_headers_or_parameters || has_parameters; /*cases where < is required*/ if (angle_quote_required) { error = belle_sip_snprintf(buff, buff_size, offset, "%s", "<"); if (error != BELLE_SIP_OK) return error; } if (header->uri) { error = belle_sip_uri_marshal(header->uri, buff, buff_size, offset); } else { error = belle_generic_uri_marshal(header->absolute_uri, buff, buff_size, offset); } if (error != BELLE_SIP_OK) return error; if (angle_quote_required) { error = belle_sip_snprintf(buff, buff_size, offset, "%s", ">"); if (error != BELLE_SIP_OK) return error; } } error = belle_sip_parameters_marshal(&header->base, buff, buff_size, offset); if (error != BELLE_SIP_OK) return error; return error; } belle_sip_error_code belle_sip_header_address_marshal(belle_sip_header_address_t *header, char *buff, size_t buff_size, size_t *offset) { return _belle_sip_header_address_marshal(header, buff, buff_size, offset, FALSE); } #define belle_sip_header_address_clone \ _belle_sip_header_address_clone /*because public clone function is not the one to be used internally*/ BELLE_SIP_NEW_HEADER(header_address, parameters, "header_address"); BELLE_SIP_ADDRESS_PARSE(header_address); GET_SET_STRING(belle_sip_header_address, displayname); SET_QUOTED_STRING(belle_sip_header_address, displayname); void belle_sip_header_address_set_quoted_displayname_with_slashes(belle_sip_header_address_t *address, const char *value) { char *unescaped_value = belle_sip_string_to_backslash_less_unescaped_string(value); belle_sip_header_address_set_quoted_displayname(address, unescaped_value); belle_sip_free(unescaped_value); } belle_sip_uri_t *belle_sip_header_address_get_uri(const belle_sip_header_address_t *address) { return address->uri; } void belle_sip_header_address_set_uri(belle_sip_header_address_t *address, belle_sip_uri_t *uri) { if (uri) belle_sip_object_ref(uri); if (address->uri) { belle_sip_object_unref(address->uri); } address->uri = uri; if (address->absolute_uri && uri) { belle_sip_warning("sip absolute uri [%p] already set for header_address [%p], cleaning it", address->absolute_uri, address); belle_sip_header_address_set_absolute_uri(address, NULL); } } void belle_sip_header_address_set_automatic(belle_sip_header_address_t *address, int automatic) { address->automatic = (unsigned char)automatic; } int belle_sip_header_address_get_automatic(const belle_sip_header_address_t *address) { return address->automatic; } belle_generic_uri_t *belle_sip_header_address_get_absolute_uri(const belle_sip_header_address_t *address) { return address->absolute_uri; } void belle_sip_header_address_set_absolute_uri(belle_sip_header_address_t *address, belle_generic_uri_t *absolute_uri) { belle_sip_object_ref(absolute_uri); if (address->absolute_uri) { belle_sip_object_unref(address->absolute_uri); } address->absolute_uri = absolute_uri; if (address->uri && absolute_uri) { belle_sip_warning("sip uri [%p] already set for header_address [%p], cleaning it", address->uri, address); belle_sip_header_address_set_uri(address, NULL); } } void belle_sip_header_address_set_generic_uri(belle_sip_header_address_t *obj, belle_generic_uri_t *generic_uri) { const char *scheme = belle_generic_uri_get_scheme(generic_uri); if (scheme && (strcasecmp(scheme, "sip") != 0) && (strcasecmp(scheme, "sips") != 0)) { belle_sip_header_address_set_absolute_uri(obj, generic_uri); } else { belle_sip_error("Cannot parse a sip/sips uri as a generic uri"); belle_sip_object_unref(generic_uri); } } belle_sip_header_address_t *belle_sip_header_address_create(const char *display, belle_sip_uri_t *uri) { belle_sip_header_address_t *address = belle_sip_header_address_new(); belle_sip_header_address_set_displayname(address, display); belle_sip_header_address_set_uri(address, uri); return address; } belle_sip_header_address_t *belle_sip_header_address_create2(const char *display, belle_generic_uri_t *uri) { belle_sip_header_address_t *address = belle_sip_header_address_new(); belle_sip_header_address_set_displayname(address, display); belle_sip_header_address_set_absolute_uri(address, uri); return address; } int belle_sip_header_address_equals(const belle_sip_header_address_t *addr_a, const belle_sip_header_address_t *addr_b) { // Addresses are not identical if either is NULL if (!addr_a | !addr_b) return -1; belle_sip_uri_t *uri_a = belle_sip_header_address_get_uri(addr_a); belle_sip_uri_t *uri_b = belle_sip_header_address_get_uri(addr_b); // URIs are not identical if either is NULL if (!uri_a | !uri_b) return -1; const bool_t uri_equal = (belle_sip_uri_equals(uri_a, uri_b) != 0); const char *displayname_a = belle_sip_header_address_get_displayname(addr_a); const char *displayname_b = belle_sip_header_address_get_displayname(addr_b); bool_t displayname_equal = FALSE; if (displayname_a && displayname_b) { displayname_equal = (strcmp(displayname_a, displayname_b) == 0); } else if (!displayname_a & !displayname_b) { displayname_equal = TRUE; } return ((uri_equal && displayname_equal) ? 0 : -1); } /*fast header address implementation*/ belle_sip_header_address_t *belle_sip_header_address_fast_parse(const char *address) { return belle_sip_header_address_parse(address); } /* same as belle_sip_header_address_fast_parse but with no error log. */ belle_sip_header_address_t *belle_sip_header_address_try_fast_parse(const char *address) { return belle_sip_try_header_address_parse(address); } belle_sip_header_address_t *belle_sip_header_address_try_parse(const char *address) { return belle_sip_try_header_address_parse(address); } /****************************** * Allow header inherits from header ******************************/ struct _belle_sip_header_allow { belle_sip_header_t header; const char *method; }; static void belle_sip_header_allow_clone(belle_sip_header_allow_t *allow, const belle_sip_header_allow_t *orig) { CLONE_STRING(belle_sip_header_allow, method, allow, orig) } static void belle_sip_header_allow_destroy(belle_sip_header_allow_t *allow) { if (allow->method) belle_sip_free((void *)allow->method); } belle_sip_error_code belle_sip_header_allow_marshal(belle_sip_header_allow_t *allow, char *buff, size_t buff_size, size_t *offset) { belle_sip_error_code error = belle_sip_header_marshal(BELLE_SIP_HEADER(allow), buff, buff_size, offset); if (error != BELLE_SIP_OK) return error; error = belle_sip_snprintf(buff, buff_size, offset, "%s", allow->method); if (error != BELLE_SIP_OK) return error; return error; } BELLE_SIP_NEW_HEADER(header_allow, header, "Allow"); BELLE_SIP_PARSE_FULL(header_allow); belle_sip_header_allow_t *belle_sip_header_allow_create(const char *methods) { belle_sip_header_allow_t *allow = belle_sip_header_allow_new(); belle_sip_header_allow_set_method(allow, methods); return allow; } GET_SET_STRING(belle_sip_header_allow, method); /************************ * Contact header object inherits from header_address ***********************/ struct _belle_sip_header_contact { belle_sip_header_address_t address; unsigned char wildcard; unsigned char unknown; unsigned char pad[2]; }; void belle_sip_header_contact_destroy(belle_sip_header_contact_t *contact) { } void belle_sip_header_contact_clone(belle_sip_header_contact_t *contact, const belle_sip_header_contact_t *orig) { contact->wildcard = orig->wildcard; } belle_sip_error_code belle_sip_header_contact_marshal(belle_sip_header_contact_t *contact, char *buff, size_t buff_size, size_t *offset) { belle_sip_error_code error = belle_sip_header_marshal(BELLE_SIP_HEADER(contact), buff, buff_size, offset); if (error != BELLE_SIP_OK) return error; if (contact->wildcard) { error = belle_sip_snprintf(buff, buff_size, offset, "%s", "*"); } else { error = belle_sip_header_address_marshal(&contact->address, buff, buff_size, offset); } return error; } BELLE_SIP_NEW_HEADER(header_contact, header_address, BELLE_SIP_CONTACT); BELLE_SIP_PARSE_WITH_CONTEXT(header_contact); belle_sip_header_contact_t *belle_sip_header_contact_create(const belle_sip_header_address_t *contact) { belle_sip_header_contact_t *header = belle_sip_header_contact_new(); _belle_sip_object_copy(BELLE_SIP_OBJECT(header), BELLE_SIP_OBJECT(contact)); belle_sip_header_set_next(BELLE_SIP_HEADER(header), NULL); /*make sure only one header is kept*/ belle_sip_header_set_name(BELLE_SIP_HEADER(header), BELLE_SIP_CONTACT); /*restaure header name*/ return header; } GET_SET_INT_PARAM_PRIVATE(belle_sip_header_contact, expires, int, _); GET_SET_INT_PARAM_PRIVATE(belle_sip_header_contact, q, float, _); GET_SET_BOOL(belle_sip_header_contact, wildcard, is); void belle_sip_header_contact_set_string_wildcard(belle_sip_header_contact_t *contact, const char *wildcard) { if (strcmp("*", wildcard) == 0) { belle_sip_header_contact_set_wildcard(contact, 1); } } int belle_sip_header_contact_set_expires(belle_sip_header_contact_t *contact, int expires) { if (expires < 0) { belle_sip_error("bad expires value [%i] for contact", expires); return -1; } _belle_sip_header_contact_set_expires(contact, expires); return 0; } int belle_sip_header_contact_set_qvalue(belle_sip_header_contact_t *contact, float qValue) { if (qValue != -1 && qValue < 0 && qValue > 1) { belle_sip_error("bad q value [%f] for contact", qValue); return -1; } _belle_sip_header_contact_set_q(contact, qValue); return 0; } float belle_sip_header_contact_get_qvalue(const belle_sip_header_contact_t *contact) { return belle_sip_header_contact_get_q(contact); } unsigned int belle_sip_header_contact_equals(const belle_sip_header_contact_t *a, const belle_sip_header_contact_t *b) { if (!a | !b) return 0; return belle_sip_uri_equals(belle_sip_header_address_get_uri(BELLE_SIP_HEADER_ADDRESS(a)), belle_sip_header_address_get_uri(BELLE_SIP_HEADER_ADDRESS(b))); } unsigned int belle_sip_header_contact_not_equals(const belle_sip_header_contact_t *a, const belle_sip_header_contact_t *b) { return !belle_sip_header_contact_equals(a, b); } unsigned int belle_sip_header_contact_equals_with_uri_omitting(const belle_sip_header_contact_t *a, const belle_sip_header_contact_t *b) { if (!a | !b) return 0; return belle_sip_uri_equals_with_uri_omitting(belle_sip_header_address_get_uri(BELLE_SIP_HEADER_ADDRESS(a)), belle_sip_header_address_get_uri(BELLE_SIP_HEADER_ADDRESS(b))); } unsigned int belle_sip_header_contact_not_equals_with_uri_omitting(const belle_sip_header_contact_t *a, const belle_sip_header_contact_t *b) { return !belle_sip_header_contact_equals_with_uri_omitting(a, b); } void belle_sip_header_contact_set_automatic(belle_sip_header_contact_t *a, int enabled) { belle_sip_header_address_set_automatic((belle_sip_header_address_t *)a, enabled); } int belle_sip_header_contact_get_automatic(const belle_sip_header_contact_t *a) { return belle_sip_header_address_get_automatic((belle_sip_header_address_t *)a); } void belle_sip_header_contact_set_unknown(belle_sip_header_contact_t *a, int value) { a->unknown = value; } int belle_sip_header_contact_is_unknown(const belle_sip_header_contact_t *a) { return a->unknown; } /************************** * From header object inherits from header_address ***************************/ #define BELLE_SIP_FROM_LIKE_MARSHAL(header, force_angle_quote) \ belle_sip_error_code error = belle_sip_header_marshal(BELLE_SIP_HEADER(header), buff, buff_size, offset); \ if (error != BELLE_SIP_OK) return error; \ error = _belle_sip_header_address_marshal(&header->address, buff, buff_size, offset, force_angle_quote); \ if (error != BELLE_SIP_OK) return error; \ return error; struct _belle_sip_header_from { belle_sip_header_address_t address; }; static void belle_sip_header_from_destroy(belle_sip_header_from_t *from) { } static void belle_sip_header_from_clone(belle_sip_header_from_t *from, const belle_sip_header_from_t *cloned) { } belle_sip_error_code belle_sip_header_from_marshal(belle_sip_header_from_t *from, char *buff, size_t buff_size, size_t *offset) { BELLE_SIP_FROM_LIKE_MARSHAL(from, FALSE); } belle_sip_header_from_t *belle_sip_header_from_create2(const char *uri, const char *tag) { belle_sip_header_address_t *address = belle_sip_header_address_parse(uri); if (address) { belle_sip_header_from_t *from = belle_sip_header_from_create(address, tag); belle_sip_object_unref(address); return from; } else return NULL; } belle_sip_header_from_t *belle_sip_header_from_create(const belle_sip_header_address_t *address, const char *tag) { belle_sip_header_from_t *header = belle_sip_header_from_new(); belle_sip_uri_t *uri; _belle_sip_object_copy((belle_sip_object_t *)header, (belle_sip_object_t *)address); /*clear unwanted uri components*/ if ((uri = belle_sip_header_address_get_uri(BELLE_SIP_HEADER_ADDRESS(header)))) { belle_sip_parameters_t *params = BELLE_SIP_PARAMETERS(uri); belle_sip_parameters_remove_parameter(params, "lr"); belle_sip_parameters_remove_parameter(params, "ttl"); belle_sip_parameters_remove_parameter(params, "method"); belle_sip_parameters_remove_parameter(params, "maddr"); belle_sip_parameters_remove_parameter(params, "transport"); belle_sip_uri_set_port(uri, 0); belle_sip_uri_headers_clean(uri); } belle_sip_header_set_next(BELLE_SIP_HEADER(header), NULL); /*make sure only one header is kept*/ belle_sip_header_set_name(BELLE_SIP_HEADER(header), BELLE_SIP_FROM); /*restore header name*/ if (tag) belle_sip_header_from_set_tag(header, tag); return header; } BELLE_SIP_NEW_HEADER(header_from, header_address, BELLE_SIP_FROM); BELLE_SIP_PARSE_HEADER_WITH_URI_CHECK(header_from); GET_SET_STRING_PARAM2(belle_sip_header_from, tag, raw_tag); void belle_sip_header_from_set_random_tag(belle_sip_header_from_t *obj) { char tmp[BELLE_SIP_TAG_LENGTH]; belle_sip_header_from_set_raw_tag(obj, belle_sip_random_token(tmp, sizeof(tmp))); } void belle_sip_header_from_set_tag(belle_sip_header_from_t *obj, const char *tag) { if (tag == BELLE_SIP_RANDOM_TAG) belle_sip_header_from_set_random_tag(obj); else belle_sip_header_from_set_raw_tag(obj, tag); } const char *belle_sip_header_from_get_tag(const belle_sip_header_from_t *obj) { return belle_sip_header_from_get_raw_tag(obj); } /************************** * To header object inherits from header_address ***************************/ struct _belle_sip_header_to { belle_sip_header_address_t address; }; static void belle_sip_header_to_destroy(belle_sip_header_to_t *to) { } void belle_sip_header_to_clone(belle_sip_header_to_t *contact, const belle_sip_header_to_t *orig) { } belle_sip_error_code belle_sip_header_to_marshal(belle_sip_header_to_t *to, char *buff, size_t buff_size, size_t *offset){BELLE_SIP_FROM_LIKE_MARSHAL(to, FALSE)} BELLE_SIP_NEW_HEADER(header_to, header_address, BELLE_SIP_TO); BELLE_SIP_PARSE_HEADER_WITH_URI_CHECK(header_to); GET_SET_STRING_PARAM2(belle_sip_header_to, tag, raw_tag); belle_sip_header_to_t *belle_sip_header_to_create2(const char *uri, const char *tag) { belle_sip_header_address_t *address = belle_sip_header_address_parse(uri); if (address) { belle_sip_header_to_t *to = belle_sip_header_to_create(address, tag); belle_sip_object_unref(address); return to; } else return NULL; } belle_sip_header_to_t *belle_sip_header_to_create(const belle_sip_header_address_t *address, const char *tag) { belle_sip_header_to_t *header = belle_sip_header_to_new(); belle_sip_uri_t *uri; _belle_sip_object_copy((belle_sip_object_t *)header, (belle_sip_object_t *)address); /*clear unwanted uri components*/ if ((uri = belle_sip_header_address_get_uri(BELLE_SIP_HEADER_ADDRESS(header)))) { belle_sip_parameters_t *params = BELLE_SIP_PARAMETERS(uri); belle_sip_parameters_remove_parameter(params, "lr"); belle_sip_parameters_remove_parameter(params, "ttl"); belle_sip_parameters_remove_parameter(params, "method"); belle_sip_parameters_remove_parameter(params, "maddr"); belle_sip_parameters_remove_parameter(params, "transport"); belle_sip_uri_set_port(uri, 0); belle_sip_uri_headers_clean(uri); } belle_sip_header_set_next(BELLE_SIP_HEADER(header), NULL); /*make sure only one header is kept*/ belle_sip_header_set_name(BELLE_SIP_HEADER(header), BELLE_SIP_TO); /*restaure header name*/ if (tag) belle_sip_header_to_set_tag(header, tag); return header; } void belle_sip_header_to_set_random_tag(belle_sip_header_to_t *obj) { char tmp[8]; /*not less than 32bit */ belle_sip_header_to_set_tag(obj, belle_sip_random_token(tmp, sizeof(tmp))); } void belle_sip_header_to_set_tag(belle_sip_header_to_t *obj, const char *tag) { if (tag == BELLE_SIP_RANDOM_TAG) belle_sip_header_to_set_random_tag(obj); else belle_sip_header_to_set_raw_tag(obj, tag); } const char *belle_sip_header_to_get_tag(const belle_sip_header_to_t *obj) { return belle_sip_header_to_get_raw_tag(obj); } /************************** * Diversion header object inherits from header_address ***************************/ struct _belle_sip_header_diversion { belle_sip_header_address_t address; }; static void belle_sip_header_diversion_destroy(belle_sip_header_diversion_t *diversion) { } void belle_sip_header_diversion_clone(belle_sip_header_diversion_t *contact, const belle_sip_header_diversion_t *orig) { } belle_sip_error_code belle_sip_header_diversion_marshal(belle_sip_header_diversion_t *diversion, char *buff, size_t buff_size, size_t *offset){BELLE_SIP_FROM_LIKE_MARSHAL(diversion, FALSE)} BELLE_SIP_NEW_HEADER(header_diversion, header_address, BELLE_SIP_DIVERSION) BELLE_SIP_PARSE_FULL(header_diversion); GET_SET_STRING_PARAM2(belle_sip_header_diversion, tag, raw_tag); belle_sip_header_diversion_t *belle_sip_header_diversion_create2(const char *uri, const char *tag) { belle_sip_header_address_t *address = belle_sip_header_address_parse(uri); if (address) { belle_sip_header_diversion_t *diversion = belle_sip_header_diversion_create(address, tag); belle_sip_object_unref(address); return diversion; } else return NULL; } belle_sip_header_diversion_t *belle_sip_header_diversion_create(const belle_sip_header_address_t *address, const char *tag) { belle_sip_header_diversion_t *header = belle_sip_header_diversion_new(); belle_sip_uri_t *uri; _belle_sip_object_copy((belle_sip_object_t *)header, (belle_sip_object_t *)address); /*clear unwanted uri components*/ if ((uri = belle_sip_header_address_get_uri(BELLE_SIP_HEADER_ADDRESS(header)))) { belle_sip_parameters_t *params = BELLE_SIP_PARAMETERS(uri); belle_sip_parameters_remove_parameter(params, "lr"); belle_sip_parameters_remove_parameter(params, "ttl"); belle_sip_parameters_remove_parameter(params, "method"); belle_sip_parameters_remove_parameter(params, "maddr"); belle_sip_parameters_remove_parameter(params, "transport"); belle_sip_uri_set_port(uri, 0); belle_sip_uri_headers_clean(uri); } belle_sip_header_set_name(BELLE_SIP_HEADER(header), BELLE_SIP_DIVERSION); /*restaure header name*/ if (tag) belle_sip_header_diversion_set_tag(header, tag); return header; } void belle_sip_header_diversion_set_random_tag(belle_sip_header_diversion_t *obj) { char tmp[8]; /*not less than 32bit */ belle_sip_header_diversion_set_tag(obj, belle_sip_random_token(tmp, sizeof(tmp))); } void belle_sip_header_diversion_set_tag(belle_sip_header_diversion_t *obj, const char *tag) { if (tag == BELLE_SIP_RANDOM_TAG) belle_sip_header_diversion_set_random_tag(obj); else belle_sip_header_diversion_set_raw_tag(obj, tag); } const char *belle_sip_header_diversion_get_tag(const belle_sip_header_diversion_t *obj) { return belle_sip_header_diversion_get_raw_tag(obj); } /****************************** * Session-Expires inherits from parameters ******************************/ struct _belle_sip_header_session_expires { belle_sip_parameters_t params_list; int delta; }; static void belle_sip_header_session_expires_destroy(belle_sip_header_session_expires_t *session_expires) { } static void belle_sip_header_session_expires_clone(belle_sip_header_session_expires_t *session_expires, const belle_sip_header_session_expires_t *orig) { session_expires->delta = orig->delta; belle_sip_header_session_expires_set_refresher_value(session_expires, belle_sip_header_session_expires_get_refresher_value(orig)); } belle_sip_error_code belle_sip_header_session_expires_marshal(belle_sip_header_session_expires_t *session_expires, char *buff, size_t buff_size, size_t *offset) { belle_sip_error_code error = belle_sip_header_marshal(BELLE_SIP_HEADER(session_expires), buff, buff_size, offset); if (session_expires->delta) { error = belle_sip_snprintf(buff, buff_size, offset, "%i", session_expires->delta); if (error != BELLE_SIP_OK) return error; } error = belle_sip_parameters_marshal(&session_expires->params_list, buff, buff_size, offset); if (error != BELLE_SIP_OK) return error; return error; } belle_sip_header_session_expires_t * belle_sip_header_session_expires_create(int delta, belle_sip_header_session_expires_refresher_t refresher) { belle_sip_header_session_expires_t *session_expires = belle_sip_header_session_expires_new(); belle_sip_header_session_expires_set_delta(session_expires, delta); if (refresher != BELLE_SIP_HEADER_SESSION_EXPIRES_UNSPECIFIED) { belle_sip_header_session_expires_set_refresher_value(session_expires, refresher); } return session_expires; } BELLE_SIP_PARSE_FULL(header_session_expires); BELLE_SIP_NEW_HEADER(header_session_expires, parameters, BELLE_SIP_SESSION_EXPIRES); GET_SET_STRING_PARAM(belle_sip_header_session_expires, refresher); GET_SET_INT(belle_sip_header_session_expires, delta, int); belle_sip_header_session_expires_refresher_t belle_sip_header_session_expires_get_refresher_value(const belle_sip_header_session_expires_t *session_expires) { const char *refresher_value = belle_sip_header_session_expires_get_refresher(session_expires); if (refresher_value == NULL) { return BELLE_SIP_HEADER_SESSION_EXPIRES_UNSPECIFIED; } else if (strcmp("uac", refresher_value) == 0) { return BELLE_SIP_HEADER_SESSION_EXPIRES_UAC; } else if (strcmp("uas", refresher_value) == 0) { return BELLE_SIP_HEADER_SESSION_EXPIRES_UAS; } return BELLE_SIP_HEADER_SESSION_EXPIRES_UNSPECIFIED; } void belle_sip_header_session_expires_set_refresher_value( belle_sip_header_session_expires_t *session_expires, belle_sip_header_session_expires_refresher_t refresher_value) { switch (refresher_value) { case BELLE_SIP_HEADER_SESSION_EXPIRES_UNSPECIFIED: belle_sip_parameters_remove_parameter(BELLE_SIP_PARAMETERS(session_expires), "refresher"); break; case BELLE_SIP_HEADER_SESSION_EXPIRES_UAC: belle_sip_header_session_expires_set_refresher(session_expires, "uac"); break; case BELLE_SIP_HEADER_SESSION_EXPIRES_UAS: belle_sip_header_session_expires_set_refresher(session_expires, "uas"); break; } } /****************************** * User-Agent header inherits from header ******************************/ struct _belle_sip_header_user_agent { belle_sip_header_t header; belle_sip_list_t *products; }; static void belle_sip_header_user_agent_destroy(belle_sip_header_user_agent_t *user_agent) { belle_sip_header_user_agent_set_products(user_agent, NULL); } static void belle_sip_header_user_agent_clone(belle_sip_header_user_agent_t *user_agent, const belle_sip_header_user_agent_t *orig) { belle_sip_list_t *list = orig->products; for (; list != NULL; list = list->next) { belle_sip_header_user_agent_add_product(user_agent, (const char *)list->data); } } belle_sip_error_code belle_sip_header_user_agent_marshal(belle_sip_header_user_agent_t *user_agent, char *buff, size_t buff_size, size_t *offset) { belle_sip_error_code error = BELLE_SIP_OK; belle_sip_list_t *list = user_agent->products; error = belle_sip_header_marshal(BELLE_SIP_HEADER(user_agent), buff, buff_size, offset); if (error != BELLE_SIP_OK) return error; for (; list != NULL; list = list->next) { error = belle_sip_snprintf(buff, buff_size, offset, list == user_agent->products ? "%s" : " %s", (const char *)list->data); if (error != BELLE_SIP_OK) return error; } return error; } BELLE_SIP_NEW_HEADER(header_user_agent, header, "User-Agent"); BELLE_SIP_PARSE_FULL(header_user_agent); belle_sip_list_t *belle_sip_header_user_agent_get_products(const belle_sip_header_user_agent_t *user_agent) { return user_agent->products; } void belle_sip_header_user_agent_set_products(belle_sip_header_user_agent_t *user_agent, belle_sip_list_t *products) { belle_sip_list_t *list; if (user_agent->products) { for (list = user_agent->products; list != NULL; list = list->next) { belle_sip_free((void *)list->data); } belle_sip_list_free(user_agent->products); } user_agent->products = products; } void belle_sip_header_user_agent_add_product(belle_sip_header_user_agent_t *user_agent, const char *product) { user_agent->products = belle_sip_list_append(user_agent->products, belle_sip_strdup(product)); } int belle_sip_header_user_agent_get_products_as_string(const belle_sip_header_user_agent_t *user_agent, char *value, unsigned int value_size) { size_t result = 0; belle_sip_error_code error = BELLE_SIP_OK; belle_sip_list_t *list = user_agent->products; for (; list != NULL; list = list->next) { error = belle_sip_snprintf(value, value_size, &result, "%s ", (const char *)list->data); if (error != BELLE_SIP_OK) return -1; } if (result > 0) value[result - 1] = '\0'; /*remove last space */ return (int)result - 1; } /************************** * Via header object inherits from parameters ***************************/ struct _belle_sip_header_via { belle_sip_parameters_t params_list; char *protocol; char *transport; char *host; int port; char *received; }; static void belle_sip_header_via_destroy(belle_sip_header_via_t *via) { if (via->protocol) belle_sip_free(via->protocol); if (via->transport) belle_sip_free(via->transport); if (via->host) belle_sip_free(via->host); DESTROY_STRING(via, received) } static void belle_sip_header_via_clone(belle_sip_header_via_t *via, const belle_sip_header_via_t *orig) { CLONE_STRING(belle_sip_header_via, protocol, via, orig) CLONE_STRING(belle_sip_header_via, transport, via, orig) CLONE_STRING(belle_sip_header_via, host, via, orig) CLONE_STRING(belle_sip_header_via, received, via, orig) via->port = orig->port; } belle_sip_error_code belle_sip_header_via_marshal(belle_sip_header_via_t *via, char *buff, size_t buff_size, size_t *offset) { belle_sip_error_code error = belle_sip_header_marshal(BELLE_SIP_HEADER(via), buff, buff_size, offset); if (error != BELLE_SIP_OK) return error; error = belle_sip_snprintf(buff, buff_size, offset, "%s/%s", via->protocol, via->transport); if (error != BELLE_SIP_OK) return error; if (via->host) { if (strchr(via->host, ':')) { /*ipv6*/ error = belle_sip_snprintf(buff, buff_size, offset, " [%s]", via->host); } else { error = belle_sip_snprintf(buff, buff_size, offset, " %s", via->host); } if (error != BELLE_SIP_OK) return error; } else { belle_sip_warning("no host found in this via"); } if (via->port > 0) { error = belle_sip_snprintf(buff, buff_size, offset, ":%i", via->port); if (error != BELLE_SIP_OK) return error; } if (via->received) { error = belle_sip_snprintf(buff, buff_size, offset, ";received=%s", via->received); if (error != BELLE_SIP_OK) return error; } error = belle_sip_parameters_marshal(&via->params_list, buff, buff_size, offset); if (error != BELLE_SIP_OK) return error; return error; } belle_sip_header_via_t * belle_sip_header_via_create(const char *host, int port, const char *transport, const char *branch) { belle_sip_header_via_t *via = belle_sip_header_via_new(); via->host = belle_sip_strdup(host); via->port = port; via->transport = belle_sip_strdup(transport); via->protocol = belle_sip_strdup("SIP/2.0"); belle_sip_header_via_set_branch(via, branch); return via; } BELLE_SIP_NEW_HEADER(header_via, parameters, BELLE_SIP_VIA); BELLE_SIP_PARSE_WITH_CONTEXT(header_via); GET_SET_STRING(belle_sip_header_via, protocol); GET_SET_STRING(belle_sip_header_via, transport); GET_SET_STRING_NO_ENCLOSING_BRACKETS(belle_sip_header_via, host); GET_SET_STRING_NO_ENCLOSING_BRACKETS(belle_sip_header_via, received); GET_SET_INT_PRIVATE(belle_sip_header_via, port, int, _); GET_SET_STRING_PARAM(belle_sip_header_via, branch); GET_SET_STRING_PARAM(belle_sip_header_via, maddr); GET_SET_INT_PARAM_PRIVATE(belle_sip_header_via, rport, int, _); GET_SET_INT_PARAM_PRIVATE(belle_sip_header_via, ttl, int, _); int belle_sip_header_via_set_rport(belle_sip_header_via_t *obj, int value) { if (value == -1) { belle_sip_parameters_set_parameter(BELLE_SIP_PARAMETERS(obj), "rport", NULL); return 0; } if (value > 0 && value < 65536) { _belle_sip_header_via_set_rport(obj, value); return 0; } else { belle_sip_error("bad rport value [%i] for via", value); return -1; } } int belle_sip_header_via_set_ttl(belle_sip_header_via_t *obj, int value) { if (value == -1 || (value > 0 && value <= 255)) { _belle_sip_header_via_set_ttl(obj, value); return 0; } else { belle_sip_error("bad ttl value [%i] for via", value); return -1; } } int belle_sip_header_via_set_port(belle_sip_header_via_t *obj, int value) { if (value == -1 || (value > 0 && value < 65536)) { _belle_sip_header_via_set_port(obj, value); return 0; } else { belle_sip_error("bad port value [%i] for via", value); return -1; } } int belle_sip_header_via_get_listening_port(const belle_sip_header_via_t *via) { int ret = belle_sip_header_via_get_port(via); if (ret == 0) ret = belle_sip_listening_point_get_well_known_port(via->transport); return ret; } const char *belle_sip_header_via_get_transport_lowercase(const belle_sip_header_via_t *via) { if (strcasecmp("udp", via->transport) == 0) return "udp"; else if (strcasecmp("tcp", via->transport) == 0) return "tcp"; else if (strcasecmp("tls", via->transport) == 0) return "tls"; else if (strcasecmp("dtls", via->transport) == 0) return "dtls"; else { belle_sip_warning("Cannot convert [%s] to lower case", via->transport); return via->transport; } } /************************** * call_id header object inherits from object ***************************/ struct _belle_sip_header_call_id { belle_sip_header_t header; const char *call_id; }; static void belle_sip_header_call_id_destroy(belle_sip_header_call_id_t *call_id) { if (call_id->call_id) belle_sip_free((void *)call_id->call_id); } static void belle_sip_header_call_id_clone(belle_sip_header_call_id_t *call_id, const belle_sip_header_call_id_t *orig) { CLONE_STRING(belle_sip_header_call_id, call_id, call_id, orig); } belle_sip_error_code belle_sip_header_call_id_marshal(belle_sip_header_call_id_t *call_id, char *buff, size_t buff_size, size_t *offset) { belle_sip_error_code error = belle_sip_header_marshal(BELLE_SIP_HEADER(call_id), buff, buff_size, offset); if (error != BELLE_SIP_OK) return error; error = belle_sip_snprintf(buff, buff_size, offset, "%s", call_id->call_id); if (error != BELLE_SIP_OK) return error; return error; } unsigned int belle_sip_header_call_id_equals(const belle_sip_header_call_id_t *a, const belle_sip_header_call_id_t *b) { return strcasecmp(a->call_id, b->call_id) == 0; } BELLE_SIP_NEW_HEADER(header_call_id, header, BELLE_SIP_CALL_ID); BELLE_SIP_PARSE_FULL(header_call_id); GET_SET_STRING(belle_sip_header_call_id, call_id); /************************** * retry-after header object inherits from object ***************************/ struct _belle_sip_header_retry_after { belle_sip_header_t header; int retry_after; }; static void belle_sip_header_retry_after_destroy(belle_sip_header_retry_after_t *retry_after) { } static void belle_sip_header_retry_after_clone(belle_sip_header_retry_after_t *retry_after, const belle_sip_header_retry_after_t *orig) { retry_after->retry_after = orig->retry_after; } belle_sip_error_code belle_sip_header_retry_after_marshal(belle_sip_header_retry_after_t *retry_after, char *buff, size_t buff_size, size_t *offset) { belle_sip_error_code error = belle_sip_header_marshal(BELLE_SIP_HEADER(retry_after), buff, buff_size, offset); if (error != BELLE_SIP_OK) return error; if (retry_after->retry_after > 0) { error = belle_sip_snprintf(buff, buff_size, offset, "%i", retry_after->retry_after); if (error != BELLE_SIP_OK) return error; } return error; } BELLE_SIP_NEW_HEADER(header_retry_after, header, BELLE_SIP_RETRY_AFTER); BELLE_SIP_PARSE_FULL(header_retry_after); GET_SET_INT(belle_sip_header_retry_after, retry_after, int); belle_sip_header_retry_after_t *belle_sip_header_retry_after_create(int retry_after) { belle_sip_header_retry_after_t *obj; obj = belle_sip_header_retry_after_new(); belle_sip_header_retry_after_set_retry_after(obj, retry_after); return obj; } /************************** * cseq header object inherit from object ***************************/ struct _belle_sip_header_cseq { belle_sip_header_t header; char *method; unsigned int seq_number; }; static void belle_sip_header_cseq_destroy(belle_sip_header_cseq_t *cseq) { if (cseq->method) belle_sip_free(cseq->method); } static void belle_sip_header_cseq_clone(belle_sip_header_cseq_t *cseq, const belle_sip_header_cseq_t *orig) { CLONE_STRING(belle_sip_header_cseq, method, cseq, orig) cseq->seq_number = orig->seq_number; } belle_sip_error_code belle_sip_header_cseq_marshal(belle_sip_header_cseq_t *cseq, char *buff, size_t buff_size, size_t *offset) { belle_sip_error_code error = belle_sip_header_marshal(BELLE_SIP_HEADER(cseq), buff, buff_size, offset); if (error != BELLE_SIP_OK) return error; error = belle_sip_snprintf(buff, buff_size, offset, "%i %s", cseq->seq_number, cseq->method); if (error != BELLE_SIP_OK) return error; return error; } belle_sip_header_cseq_t *belle_sip_header_cseq_create(unsigned int number, const char *method) { belle_sip_header_cseq_t *cseq = belle_sip_header_cseq_new(); belle_sip_header_cseq_set_method(cseq, method); cseq->seq_number = number; return cseq; } BELLE_SIP_NEW_HEADER(header_cseq, header, BELLE_SIP_CSEQ); BELLE_SIP_PARSE_FULL(header_cseq); GET_SET_STRING(belle_sip_header_cseq, method); GET_SET_INT(belle_sip_header_cseq, seq_number, unsigned int); /************************** * content type header object inherit from parameters ***************************/ struct _belle_sip_header_content_type { belle_sip_parameters_t params_list; const char *type; const char *subtype; }; static void belle_sip_header_content_type_destroy(belle_sip_header_content_type_t *content_type) { if (content_type->type) belle_sip_free((void *)content_type->type); if (content_type->subtype) belle_sip_free((void *)content_type->subtype); } static void belle_sip_header_content_type_clone(belle_sip_header_content_type_t *content_type, const belle_sip_header_content_type_t *orig) { CLONE_STRING(belle_sip_header_content_type, type, content_type, orig); CLONE_STRING(belle_sip_header_content_type, subtype, content_type, orig); } belle_sip_error_code belle_sip_header_content_type_marshal(belle_sip_header_content_type_t *content_type, char *buff, size_t buff_size, size_t *offset) { belle_sip_error_code error = belle_sip_header_marshal(BELLE_SIP_HEADER(content_type), buff, buff_size, offset); if (error != BELLE_SIP_OK) return error; error = belle_sip_snprintf(buff, buff_size, offset, "%s/%s", content_type->type, content_type->subtype); if (error != BELLE_SIP_OK) return error; error = belle_sip_parameters_marshal(&content_type->params_list, buff, buff_size, offset); if (error != BELLE_SIP_OK) return error; return error; } BELLE_SIP_NEW_HEADER(header_content_type, parameters, BELLE_SIP_CONTENT_TYPE); BELLE_SIP_PARSE_FULL(header_content_type); belle_sip_header_content_type_t *belle_sip_header_content_type_create(const char *type, const char *sub_type) { belle_sip_header_content_type_t *header = belle_sip_header_content_type_new(); belle_sip_header_content_type_set_type(header, type); belle_sip_header_content_type_set_subtype(header, sub_type); return header; } GET_SET_STRING(belle_sip_header_content_type, type); GET_SET_STRING(belle_sip_header_content_type, subtype); /************************** * Route header object inherits from header_address ***************************/ struct _belle_sip_header_route { belle_sip_header_address_t address; }; static void belle_sip_header_route_destroy(belle_sip_header_route_t *route) { } static void belle_sip_header_route_clone(belle_sip_header_route_t *route, const belle_sip_header_route_t *orig) { } belle_sip_error_code belle_sip_header_route_marshal(belle_sip_header_route_t *route, char *buff, size_t buff_size, size_t *offset){BELLE_SIP_FROM_LIKE_MARSHAL(route, TRUE)} BELLE_SIP_NEW_HEADER(header_route, header_address, BELLE_SIP_ROUTE) BELLE_SIP_PARSE_WITH_CONTEXT(header_route); belle_sip_header_route_t *belle_sip_header_route_create(const belle_sip_header_address_t *route) { belle_sip_header_route_t *header = belle_sip_header_route_new(); _belle_sip_object_copy((belle_sip_object_t *)header, (belle_sip_object_t *)route); belle_sip_header_set_next(BELLE_SIP_HEADER(header), NULL); /*make sure only one header is kept*/ belle_sip_header_set_name(BELLE_SIP_HEADER(header), BELLE_SIP_ROUTE); /*restore header name*/ return header; } int belle_sip_header_route_equals(const belle_sip_header_route_t *route_a, const belle_sip_header_route_t *route_b) { // Routes are not identical if either is NULL if (!route_a | !route_b) { return -1; } return belle_sip_header_address_equals(&route_a->address, &route_b->address); } /************************** * Record route header object inherits from header_address ***************************/ struct _belle_sip_header_record_route { belle_sip_header_address_t address; unsigned char auto_outgoing; unsigned char pad[3]; }; static void belle_sip_header_record_route_destroy(belle_sip_header_record_route_t *record_route) { } static void belle_sip_header_record_route_clone(belle_sip_header_record_route_t *record_route, const belle_sip_header_record_route_t *orig) { } belle_sip_error_code belle_sip_header_record_route_marshal(belle_sip_header_record_route_t *record_route, char *buff, size_t buff_size, size_t *offset){ BELLE_SIP_FROM_LIKE_MARSHAL(record_route, TRUE)} belle_sip_header_record_route_t *belle_sip_header_record_route_new_auto_outgoing() { belle_sip_header_record_route_t *rr = belle_sip_header_record_route_new(); rr->auto_outgoing = TRUE; return rr; } unsigned char belle_sip_header_record_route_get_auto_outgoing(const belle_sip_header_record_route_t *a) { return a->auto_outgoing; } BELLE_SIP_NEW_HEADER(header_record_route, header_address, BELLE_SIP_RECORD_ROUTE); BELLE_SIP_PARSE_WITH_CONTEXT(header_record_route); /************************** * Service route header object inherits from header_address ***************************/ struct _belle_sip_header_service_route { belle_sip_header_address_t address; }; static void belle_sip_header_service_route_destroy(belle_sip_header_service_route_t *service_route) { } static void belle_sip_header_service_route_clone(belle_sip_header_service_route_t *service_route, const belle_sip_header_service_route_t *orig) { } belle_sip_error_code belle_sip_header_service_route_marshal(belle_sip_header_service_route_t *service_route, char *buff, size_t buff_size, size_t *offset){ BELLE_SIP_FROM_LIKE_MARSHAL(service_route, TRUE)} BELLE_SIP_NEW_HEADER(header_service_route, header_address, BELLE_SIP_SERVICE_ROUTE); BELLE_SIP_PARSE_WITH_CONTEXT(header_service_route); /************************** * content length header object inherit from object ***************************/ struct _belle_sip_header_content_length { belle_sip_header_t header; size_t content_length; }; static void belle_sip_header_content_length_destroy(belle_sip_header_content_length_t *content_length) { } static void belle_sip_header_content_length_clone(belle_sip_header_content_length_t *content_length, const belle_sip_header_content_length_t *orig) { content_length->content_length = orig->content_length; } belle_sip_error_code belle_sip_header_content_length_marshal(belle_sip_header_content_length_t *content_length, char *buff, size_t buff_size, size_t *offset) { belle_sip_error_code error = belle_sip_header_marshal(BELLE_SIP_HEADER(content_length), buff, buff_size, offset); if (error != BELLE_SIP_OK) return error; error = belle_sip_snprintf(buff, buff_size, offset, FORMAT_SIZE_T, content_length->content_length); if (error != BELLE_SIP_OK) return error; return error; } BELLE_SIP_NEW_HEADER(header_content_length, header, BELLE_SIP_CONTENT_LENGTH); BELLE_SIP_PARSE_FULL(header_content_length); GET_SET_INT(belle_sip_header_content_length, content_length, size_t); belle_sip_header_content_length_t *belle_sip_header_content_length_create(size_t content_length) { belle_sip_header_content_length_t *obj; obj = belle_sip_header_content_length_new(); belle_sip_header_content_length_set_content_length(obj, content_length); return obj; } /************************** * Expires header object inherit from header ***************************/ struct _belle_sip_header_expires { belle_sip_header_t header; int expires; }; static void belle_sip_header_expires_destroy(belle_sip_header_expires_t *expires) { } static void belle_sip_header_expires_clone(belle_sip_header_expires_t *expires, const belle_sip_header_expires_t *orig) { expires->expires = orig->expires; } belle_sip_error_code belle_sip_header_expires_marshal(belle_sip_header_expires_t *expires, char *buff, size_t buff_size, size_t *offset) { belle_sip_error_code error = belle_sip_header_marshal(BELLE_SIP_HEADER(expires), buff, buff_size, offset); if (error != BELLE_SIP_OK) return error; error = belle_sip_snprintf(buff, buff_size, offset, "%i", expires->expires); if (error != BELLE_SIP_OK) return error; return error; } BELLE_SIP_NEW_HEADER(header_expires, header, BELLE_SIP_EXPIRES); BELLE_SIP_PARSE_FULL(header_expires); GET_SET_INT(belle_sip_header_expires, expires, int); belle_sip_header_expires_t *belle_sip_header_expires_create(int expires) { belle_sip_header_expires_t *obj = belle_sip_header_expires_new(); belle_sip_header_expires_set_expires(obj, expires); return obj; } /****************************** * Extension header inherits from header ******************************/ struct _belle_sip_header_extension { belle_sip_header_t header; const char *value; }; static void belle_sip_header_extension_destroy(belle_sip_header_extension_t *extension) { if (extension->value) belle_sip_free((void *)extension->value); } static void belle_sip_header_extension_clone(belle_sip_header_extension_t *extension, const belle_sip_header_extension_t *orig){ CLONE_STRING(belle_sip_header_extension, value, extension, orig)} belle_sip_error_code belle_sip_header_extension_marshal(belle_sip_header_extension_t *extension, char *buff, size_t buff_size, size_t *offset) { belle_sip_error_code error = belle_sip_header_marshal(BELLE_SIP_HEADER(extension), buff, buff_size, offset); if (error != BELLE_SIP_OK) return error; if (extension->value) error = belle_sip_snprintf(buff, buff_size, offset, "%s", extension->value); if (error != BELLE_SIP_OK) return error; return error; } BELLE_SIP_NEW_HEADER(header_extension, header, NULL); belle_sip_header_extension_t *belle_sip_header_extension_create(const char *name, const char *value) { belle_sip_header_extension_t *ext = belle_sip_header_extension_new(); belle_sip_header_set_name(BELLE_SIP_HEADER(ext), name); belle_sip_header_extension_set_value(ext, value); return ext; } GET_SET_STRING(belle_sip_header_extension, value); /************************** *Authorization header object inherits from parameters ***************************/ #define AUTH_BASE \ belle_sip_parameters_t params_list; \ const char *scheme; \ const char *realm; \ const char *nonce; \ const char *algorithm; \ const char *opaque; #define AUTH_BASE_DESTROY(obj) \ if (obj->scheme) belle_sip_free((void *)obj->scheme); \ if (obj->realm) belle_sip_free((void *)obj->realm); \ if (obj->nonce) belle_sip_free((void *)obj->nonce); \ if (obj->algorithm) belle_sip_free((void *)obj->algorithm); \ if (obj->opaque) belle_sip_free((void *)obj->opaque); /*if (obj->params_list) FIXME free list*/ #define AUTH_BASE_CLONE(object_type, dest, src) \ CLONE_STRING(object_type, scheme, dest, src) \ CLONE_STRING(object_type, realm, dest, src) \ CLONE_STRING(object_type, nonce, dest, src) \ CLONE_STRING(object_type, algorithm, dest, src) \ CLONE_STRING(object_type, opaque, dest, src) #define AUTH_BASE_MARSHAL(header) \ std::string border = " "; \ const belle_sip_list_t *list; \ belle_sip_error_code error = belle_sip_header_marshal(BELLE_SIP_HEADER(header), buff, buff_size, offset); \ if (error != BELLE_SIP_OK) return error; \ list = belle_sip_parameters_get_parameters(&header->params_list); \ if (header->scheme) { \ error = belle_sip_snprintf(buff, buff_size, offset, "%s", header->scheme); \ if (error != BELLE_SIP_OK) return error; \ } else { \ belle_sip_error("missing mandatory scheme"); \ } \ for (; list != NULL; list = list->next) { \ belle_sip_param_pair_t *container = reinterpret_cast(list->data); \ error = belle_sip_snprintf(buff, buff_size, offset, "%s%s", border.c_str(), container->name); \ if (error != BELLE_SIP_OK) return error; \ if (container->value) error = belle_sip_snprintf(buff, buff_size, offset, "=%s", container->value); \ if (error != BELLE_SIP_OK) return error; \ border = ", "; \ } \ if (header->realm) { \ error = belle_sip_snprintf(buff, buff_size, offset, "%srealm=\"%s\"", border.c_str(), header->realm); \ if (error != BELLE_SIP_OK) return error; \ border = ", "; \ } \ if (header->nonce) { \ error = belle_sip_snprintf(buff, buff_size, offset, "%snonce=\"%s\"", border.c_str(), header->nonce); \ if (error != BELLE_SIP_OK) return error; \ border = ", "; \ } \ if (header->algorithm) { \ const char *format; \ if (BELLE_SIP_OBJECT_IS_INSTANCE_OF(header, belle_http_header_authorization_t)) { \ format = "%salgorithm=\"%s\""; \ } else { \ format = "%salgorithm=%s"; \ } \ error = belle_sip_snprintf(buff, buff_size, offset, format, border.c_str(), header->algorithm); \ if (error != BELLE_SIP_OK) return error; \ border = ", "; \ } \ if (header->opaque) { \ error = belle_sip_snprintf(buff, buff_size, offset, "%sopaque=\"%s\"", border.c_str(), header->opaque); \ if (error != BELLE_SIP_OK) return error; \ border = ", "; \ } struct _belle_sip_header_authorization { AUTH_BASE const char *username; belle_sip_uri_t *uri; const char *response; const char *cnonce; int nonce_count; const char *qop; belle_sip_bearer_token_t *token; }; static void belle_sip_header_authorization_destroy(belle_sip_header_authorization_t *authorization) { if (authorization->username) belle_sip_free((void *)authorization->username); if (authorization->uri) { belle_sip_object_unref(authorization->uri); } if (authorization->cnonce) belle_sip_free((void *)authorization->cnonce); AUTH_BASE_DESTROY(authorization) DESTROY_STRING(authorization, response); DESTROY_STRING(authorization, qop); } static void belle_sip_header_authorization_clone(belle_sip_header_authorization_t *authorization, const belle_sip_header_authorization_t *orig) { AUTH_BASE_CLONE(belle_sip_header_authorization, authorization, orig) CLONE_STRING(belle_sip_header_authorization, username, authorization, orig) if (belle_sip_header_authorization_get_uri(orig)) { belle_sip_header_authorization_set_uri( authorization, BELLE_SIP_URI(belle_sip_object_clone(BELLE_SIP_OBJECT(belle_sip_header_authorization_get_uri(orig))))); } CLONE_STRING(belle_sip_header_authorization, response, authorization, orig) CLONE_STRING(belle_sip_header_authorization, cnonce, authorization, orig) authorization->nonce_count = orig->nonce_count; CLONE_STRING(belle_sip_header_authorization, qop, authorization, orig) } static void belle_sip_header_authorization_init(belle_sip_header_authorization_t *authorization) { } belle_sip_uri_t *belle_sip_header_authorization_get_uri(const belle_sip_header_authorization_t *authorization) { return authorization->uri; } void belle_sip_header_authorization_set_uri(belle_sip_header_authorization_t *authorization, belle_sip_uri_t *uri) { if (uri) belle_sip_object_ref(uri); if (authorization->uri) { belle_sip_object_unref(BELLE_SIP_OBJECT(authorization->uri)); } authorization->uri = uri; } belle_sip_error_code belle_sip_header_authorization_marshal(belle_sip_header_authorization_t *authorization, char *buff, size_t buff_size, size_t *offset) { char nonce_count[10]; AUTH_BASE_MARSHAL(authorization) if (authorization->username) { error = belle_sip_snprintf(buff, buff_size, offset, "%susername=\"%s\"", border.c_str(), authorization->username); if (error != BELLE_SIP_OK) return error; border = ", "; } if (authorization->uri) { error = belle_sip_snprintf(buff, buff_size, offset, "%s uri=\"", border.c_str()); if (error != BELLE_SIP_OK) return error; border = ", "; error = belle_sip_uri_marshal(authorization->uri, buff, buff_size, offset); if (error != BELLE_SIP_OK) return error; error = belle_sip_snprintf(buff, buff_size, offset, "%s", "\""); if (error != BELLE_SIP_OK) return error; } if (authorization->response) { error = belle_sip_snprintf(buff, buff_size, offset, "%sresponse=\"%s\"", border.c_str(), authorization->response); if (error != BELLE_SIP_OK) return error; border = ", "; } if (authorization->cnonce) { error = belle_sip_snprintf(buff, buff_size, offset, "%scnonce=\"%s\"", border.c_str(), authorization->cnonce); if (error != BELLE_SIP_OK) return error; border = ", "; } if (authorization->nonce_count > 0) { belle_sip_header_authorization_get_nonce_count_as_string(authorization, nonce_count); error = belle_sip_snprintf(buff, buff_size, offset, "%snc=%s", border.c_str(), nonce_count); if (error != BELLE_SIP_OK) return error; border = ", "; } if (authorization->qop) { const char *format; format = "%sqop=%s"; error = belle_sip_snprintf(buff, buff_size, offset, format, border.c_str(), authorization->qop); if (error != BELLE_SIP_OK) return error; } return error; } BELLE_SIP_NEW_HEADER(header_authorization, parameters, BELLE_SIP_AUTHORIZATION); BELLE_SIP_PARSE_FULL(header_authorization); GET_SET_STRING(belle_sip_header_authorization, scheme); GET_SET_STRING(belle_sip_header_authorization, username); SET_QUOTED_STRING(belle_sip_header_authorization, username); GET_SET_STRING(belle_sip_header_authorization, realm); SET_QUOTED_STRING(belle_sip_header_authorization, realm); GET_SET_STRING(belle_sip_header_authorization, nonce); SET_QUOTED_STRING(belle_sip_header_authorization, nonce); GET_SET_STRING(belle_sip_header_authorization, response); SET_QUOTED_STRING(belle_sip_header_authorization, response); GET_SET_STRING(belle_sip_header_authorization, algorithm); GET_SET_STRING(belle_sip_header_authorization, cnonce); GET_SET_STRING(belle_sip_header_authorization, opaque); SET_QUOTED_STRING(belle_sip_header_authorization, opaque); GET_SET_STRING(belle_sip_header_authorization, qop); GET_SET_INT(belle_sip_header_authorization, nonce_count, int); int belle_sip_header_authorization_get_nonce_count_as_string(const belle_sip_header_authorization_t *authorization, char nounce_count[9]) { nounce_count[0] = '\0'; if (authorization->nonce_count > 0) { snprintf(nounce_count, 9, "%08x", authorization->nonce_count); return 0; } else { return -1; } } /************************** *Proxy-Authorization header object inherit from parameters ***************************/ struct _belle_sip_header_proxy_authorization { belle_sip_header_authorization_t authorization; }; static void belle_sip_header_proxy_authorization_destroy(belle_sip_header_proxy_authorization_t *proxy_authorization) { } static void belle_sip_header_proxy_authorization_clone(belle_sip_header_proxy_authorization_t *proxy_authorization, const belle_sip_header_proxy_authorization_t *orig) { } belle_sip_error_code belle_sip_header_proxy_authorization_marshal( belle_sip_header_proxy_authorization_t *proxy_authorization, char *buff, size_t buff_size, size_t *offset) { return belle_sip_header_authorization_marshal(&proxy_authorization->authorization, buff, buff_size, offset); } BELLE_SIP_NEW_HEADER(header_proxy_authorization, header_authorization, BELLE_SIP_PROXY_AUTHORIZATION); BELLE_SIP_PARSE_FULL(header_proxy_authorization); /************************** *HTTP Authorization header object inherit from Authorization ***************************/ struct _belle_http_header_authorization { belle_sip_header_authorization_t authorization; belle_generic_uri_t *uri; }; static void belle_http_header_authorization_init(belle_http_header_authorization_t *authorization) { belle_sip_header_set_name(BELLE_SIP_HEADER(authorization), BELLE_HTTP_AUTHORIZATION); } static void belle_http_header_authorization_destroy(belle_http_header_authorization_t *authorization) { if (authorization->uri) { belle_sip_object_unref(authorization->uri); } } static void belle_http_header_authorization_clone(belle_http_header_authorization_t *authorization, const belle_http_header_authorization_t *orig) { if (belle_http_header_authorization_get_uri(orig)) { belle_http_header_authorization_set_uri( authorization, BELLE_GENERIC_URI(belle_sip_object_clone(BELLE_SIP_OBJECT(belle_http_header_authorization_get_uri(orig))))); } } belle_sip_error_code belle_http_header_authorization_marshal(belle_http_header_authorization_t *authorization, char *buff, size_t buff_size, size_t *offset) { belle_sip_error_code error = BELLE_SIP_OK; /*first make sure there is no sip uri*/ if (belle_sip_header_authorization_get_uri(BELLE_SIP_HEADER_AUTHORIZATION(authorization))) { belle_sip_error("Cannot marshal http_header_authorization because a sip uri is set. Use " "belle_http_authorization_set uri instead of belle_sip_header_authorization_set_uri"); return BELLE_SIP_NOT_IMPLEMENTED; } belle_sip_header_authorization_marshal(BELLE_SIP_HEADER_AUTHORIZATION(authorization), buff, buff_size, offset); if (authorization->uri) { error = belle_sip_snprintf(buff, buff_size, offset, ", uri=\""); if (error != BELLE_SIP_OK) return error; error = belle_generic_uri_marshal(authorization->uri, buff, buff_size, offset); if (error != BELLE_SIP_OK) return error; error = belle_sip_snprintf(buff, buff_size, offset, "%s", "\""); if (error != BELLE_SIP_OK) return error; } return error; } BELLE_NEW(belle_http_header_authorization, belle_sip_header_authorization); belle_generic_uri_t *belle_http_header_authorization_get_uri(const belle_http_header_authorization_t *authorization) { return authorization->uri; } void belle_http_header_authorization_set_uri(belle_http_header_authorization_t *authorization, belle_generic_uri_t *uri) { if (authorization->uri) belle_sip_object_unref(authorization->uri); if (uri) belle_sip_object_ref(uri); authorization->uri = uri; } /************************** * WWW-Authenticate header object inherits from parameters **************************/ struct _belle_sip_header_www_authenticate { AUTH_BASE char *domain; char *authz_server; int stale; belle_sip_list_t *qop; }; static void belle_sip_header_www_authenticate_destroy(belle_sip_header_www_authenticate_t *www_authenticate) { AUTH_BASE_DESTROY(www_authenticate) if (www_authenticate->domain) belle_sip_free(www_authenticate->domain); if (www_authenticate->qop) belle_sip_list_free_with_data(www_authenticate->qop, belle_sip_free); if (www_authenticate->authz_server) belle_sip_free(www_authenticate->authz_server); } void belle_sip_header_www_authenticate_init(belle_sip_header_www_authenticate_t *www_authenticate) { www_authenticate->stale = -1; } static void belle_sip_header_www_authenticate_clone(belle_sip_header_www_authenticate_t *www_authenticate, const belle_sip_header_www_authenticate_t *orig) { AUTH_BASE_CLONE(belle_sip_header_www_authenticate, www_authenticate, orig) CLONE_STRING(belle_sip_header_www_authenticate, domain, www_authenticate, orig) www_authenticate->stale = orig->stale; www_authenticate->qop = belle_sip_list_copy_with_data(orig->qop, (void *(*)(void *))belle_sip_strdup); } belle_sip_error_code belle_sip_header_www_authenticate_marshal(belle_sip_header_www_authenticate_t *www_authenticate, char *buff, size_t buff_size, size_t *offset) { belle_sip_list_t *qops = www_authenticate->qop; AUTH_BASE_MARSHAL(www_authenticate) if (www_authenticate->domain) { error = belle_sip_snprintf(buff, buff_size, offset, "%sdomain=\"%s\"", border.c_str(), www_authenticate->domain); if (error != BELLE_SIP_OK) return error; border = ", "; } if (www_authenticate->stale >= 0) { error = belle_sip_snprintf(buff, buff_size, offset, "%sstale=%s", border.c_str(), www_authenticate->stale ? "true" : "false"); if (error != BELLE_SIP_OK) return error; } if (qops != NULL && qops->data != NULL) { error = belle_sip_snprintf(buff, buff_size, offset, "%sqop=\"", border.c_str()); if (error != BELLE_SIP_OK) return error; border = ""; for (; qops != NULL; qops = qops->next) { error = belle_sip_snprintf(buff, buff_size, offset, "%s%s", border.c_str(), (const char *)qops->data); if (error != BELLE_SIP_OK) return error; border = ","; } error = belle_sip_snprintf(buff, buff_size, offset, "\""); if (error != BELLE_SIP_OK) return error; border = ", "; } return error; } #define SET_ADD_STRING_LIST(header, name) \ void header##_set_##name(header##_t *obj, belle_sip_list_t *value) { \ if (obj->name) { \ belle_sip_list_free_with_data(obj->name, belle_sip_free); \ } \ obj->name = value; \ } \ void header##_add_##name(header##_t *obj, const char *value) { \ obj->name = belle_sip_list_append(obj->name, strdup(value)); \ } BELLE_SIP_NEW_HEADER_INIT(header_www_authenticate, parameters, BELLE_SIP_WWW_AUTHENTICATE, header_www_authenticate); BELLE_SIP_PARSE_WITH_CONTEXT(header_www_authenticate); GET_SET_STRING(belle_sip_header_www_authenticate, scheme); GET_SET_STRING(belle_sip_header_www_authenticate, realm); SET_QUOTED_STRING(belle_sip_header_www_authenticate, realm); GET_SET_STRING(belle_sip_header_www_authenticate, nonce); SET_QUOTED_STRING(belle_sip_header_www_authenticate, nonce); GET_SET_STRING(belle_sip_header_www_authenticate, algorithm); GET_SET_STRING(belle_sip_header_www_authenticate, opaque); SET_QUOTED_STRING(belle_sip_header_www_authenticate, opaque); SET_ADD_STRING_LIST(belle_sip_header_www_authenticate, qop); GET_SET_STRING(belle_sip_header_www_authenticate, domain); SET_QUOTED_STRING(belle_sip_header_www_authenticate, domain); GET_SET_BOOL(belle_sip_header_www_authenticate, stale, is); void belle_sip_header_www_authenticate_set_qop_options(belle_sip_header_www_authenticate_t *obj, belle_sip_qop_options_t *options) { belle_sip_header_www_authenticate_set_qop(obj, options->list); belle_sip_object_unref(options); } belle_sip_list_t * belle_sip_header_www_authenticate_get_qop(const belle_sip_header_www_authenticate_t *www_authetication) { return www_authetication->qop; } const char * belle_sip_header_www_authenticate_get_qop_first(const belle_sip_header_www_authenticate_t *www_authetication) { return www_authetication->qop ? (const char *)www_authetication->qop->data : NULL; } const char * belle_sip_header_www_authenticate_get_authz_server(const belle_sip_header_www_authenticate_t *www_authenticate) { belle_sip_header_www_authenticate_t *mutable_authenticate = (belle_sip_header_www_authenticate_t *)www_authenticate; const char *authz_server = belle_sip_parameters_get_parameter(BELLE_SIP_PARAMETERS(www_authenticate), "authz_server"); if (mutable_authenticate->authz_server) { belle_sip_free(mutable_authenticate->authz_server); mutable_authenticate->authz_server = NULL; } if (authz_server) mutable_authenticate->authz_server = _belle_sip_str_dup_and_unquote_string(authz_server); return mutable_authenticate->authz_server; } void belle_sip_header_www_authenticate_set_string_stale(belle_sip_header_www_authenticate_t *obj, const char *value) { if (strcmp("true", value) == 0) { belle_sip_header_www_authenticate_set_stale(obj, 1); } } belle_sip_qop_options_t *belle_sip_qop_options_new(void) { return belle_sip_object_new(belle_sip_qop_options_t); } void belle_sip_qop_options_append(belle_sip_qop_options_t *obj, const char *value) { obj->list = belle_sip_list_append(obj->list, belle_sip_strdup(value)); } BELLE_SIP_DECLARE_NO_IMPLEMENTED_INTERFACES(belle_sip_qop_options_t); BELLE_SIP_INSTANCIATE_VPTR(belle_sip_qop_options_t, belle_sip_object_t, nullptr, nullptr, nullptr, TRUE); /************************** *Proxy-authenticate header object inherits from www_authenticate ***************************/ struct _belle_sip_header_proxy_authenticate { belle_sip_header_www_authenticate_t www_authenticate; }; static void belle_sip_header_proxy_authenticate_destroy(belle_sip_header_proxy_authenticate_t *proxy_authenticate) { } static void belle_sip_header_proxy_authenticate_clone(belle_sip_header_proxy_authenticate_t *proxy_authenticate, const belle_sip_header_proxy_authenticate_t *orig) { } belle_sip_error_code belle_sip_header_proxy_authenticate_marshal( belle_sip_header_proxy_authenticate_t *proxy_authenticate, char *buff, size_t buff_size, size_t *offset) { return belle_sip_header_www_authenticate_marshal(&proxy_authenticate->www_authenticate, buff, buff_size, offset); } BELLE_SIP_NEW_HEADER(header_proxy_authenticate, header_www_authenticate, BELLE_SIP_PROXY_AUTHENTICATE); BELLE_SIP_PARSE_WITH_CONTEXT(header_proxy_authenticate); /************************** * max forwards header object inherit from header ***************************/ struct _belle_sip_header_max_forwards { belle_sip_header_t header; int max_forwards; }; static void belle_sip_header_max_forwards_destroy(belle_sip_header_max_forwards_t *max_forwards) { } static void belle_sip_header_max_forwards_clone(belle_sip_header_max_forwards_t *max_forwards, const belle_sip_header_max_forwards_t *orig) { max_forwards->max_forwards = orig->max_forwards; } belle_sip_error_code belle_sip_header_max_forwards_marshal(belle_sip_header_max_forwards_t *max_forwards, char *buff, size_t buff_size, size_t *offset) { belle_sip_error_code error = belle_sip_header_marshal(BELLE_SIP_HEADER(max_forwards), buff, buff_size, offset); if (error != BELLE_SIP_OK) return error; error = belle_sip_snprintf(buff, buff_size, offset, "%i", max_forwards->max_forwards); if (error != BELLE_SIP_OK) return error; return error; } BELLE_SIP_NEW_HEADER(header_max_forwards, header, "Max-Forwards"); BELLE_SIP_PARSE_FULL(header_max_forwards); GET_SET_INT(belle_sip_header_max_forwards, max_forwards, int); int belle_sip_header_max_forwards_decrement_max_forwards(belle_sip_header_max_forwards_t *max_forwards) { return max_forwards->max_forwards--; } belle_sip_header_max_forwards_t *belle_sip_header_max_forwards_create(int value) { belle_sip_header_max_forwards_t *max_forwards = belle_sip_header_max_forwards_new(); max_forwards->max_forwards = value; return max_forwards; } /************************** * Subscription state header object inherits from parameters ***************************/ struct _belle_sip_header_subscription_state { belle_sip_parameters_t parameters; const char *state; }; static void belle_sip_header_subscription_state_destroy(belle_sip_header_subscription_state_t *subscription_state) { DESTROY_STRING(subscription_state, state); } static void belle_sip_header_subscription_state_clone(belle_sip_header_subscription_state_t *subscription_state, const belle_sip_header_subscription_state_t *orig){ CLONE_STRING(belle_sip_header_subscription_state, state, subscription_state, orig)} belle_sip_error_code belle_sip_header_subscription_state_marshal( belle_sip_header_subscription_state_t *subscription_state, char *buff, size_t buff_size, size_t *offset) { belle_sip_error_code error = belle_sip_header_marshal(BELLE_SIP_HEADER(subscription_state), buff, buff_size, offset); if (error != BELLE_SIP_OK) return error; error = belle_sip_snprintf(buff, buff_size, offset, "%s", subscription_state->state); if (error != BELLE_SIP_OK) return error; error = belle_sip_parameters_marshal(BELLE_SIP_PARAMETERS(subscription_state), buff, buff_size, offset); if (error != BELLE_SIP_OK) return error; return error; } BELLE_SIP_NEW_HEADER(header_subscription_state, parameters, BELLE_SIP_SUBSCRIPTION_STATE); BELLE_SIP_PARSE_FULL(header_subscription_state); GET_SET_STRING(belle_sip_header_subscription_state, state); GET_SET_STRING_PARAM(belle_sip_header_subscription_state, reason); GET_SET_INT_PARAM2(belle_sip_header_subscription_state, retry - after, int, retry_after); GET_SET_INT_PARAM(belle_sip_header_subscription_state, expires, int); belle_sip_header_subscription_state_t *belle_sip_header_subscription_state_create(const char *subscription_state, int expires) { belle_sip_header_subscription_state_t *sub_state = belle_sip_header_subscription_state_new(); belle_sip_header_subscription_state_set_state(sub_state, subscription_state); belle_sip_header_subscription_state_set_expires(sub_state, expires); return sub_state; } #define HEADER_TO_LIKE_IMPL(name, header_name) \ struct _belle_sip_header_##name { \ belle_sip_header_address_t address; \ }; \ \ static void belle_sip_header_##name##_destroy(belle_sip_header_##name##_t *obj) { \ } \ void belle_sip_header_##name##_clone(belle_sip_header_##name##_t *contact, \ const belle_sip_header_##name##_t *orig) { \ } \ belle_sip_error_code belle_sip_header_##name##_marshal(belle_sip_header_##name##_t *name, char *buff, \ size_t buff_size, size_t *offset) { \ BELLE_SIP_FROM_LIKE_MARSHAL(name, FALSE); \ } \ BELLE_SIP_NEW_HEADER(header_##name, header_address, header_name) \ BELLE_SIP_PARSE_HEADER_WITH_URI_CHECK(header_##name) \ belle_sip_header_##name##_t *belle_sip_header_##name##_create(const belle_sip_header_address_t *address) { \ belle_sip_header_##name##_t *header = belle_sip_header_##name##_new(); \ _belle_sip_object_copy((belle_sip_object_t *)header, (belle_sip_object_t *)address); \ belle_sip_header_set_next(BELLE_SIP_HEADER(header), NULL); /*make sure only one header is kept*/ \ belle_sip_header_set_name(BELLE_SIP_HEADER(header), header_name); \ return header; \ } /************************** * Refer-To header object inherits from header_address ***************************/ HEADER_TO_LIKE_IMPL(refer_to, BELLE_SIP_REFER_TO); /************************** * Referred-By header object inherits from header_address ***************************/ HEADER_TO_LIKE_IMPL(referred_by, BELLE_SIP_REFERRED_BY); /************************** * Replaces header object inherits from parameters ***************************/ struct _belle_sip_header_replaces { belle_sip_parameters_t parameters; char *call_id; }; static void belle_sip_header_replaces_destroy(belle_sip_header_replaces_t *replaces) { DESTROY_STRING(replaces, call_id); } static void belle_sip_header_replaces_clone(belle_sip_header_replaces_t *replaces, const belle_sip_header_replaces_t *orig){ CLONE_STRING(belle_sip_header_replaces, call_id, replaces, orig)} belle_sip_error_code belle_sip_header_replaces_marshal(belle_sip_header_replaces_t *replaces, char *buff, size_t buff_size, size_t *offset) { belle_sip_error_code error = belle_sip_header_marshal(BELLE_SIP_HEADER(replaces), buff, buff_size, offset); if (error != BELLE_SIP_OK) return error; error = belle_sip_snprintf(buff, buff_size, offset, "%s", replaces->call_id); if (error != BELLE_SIP_OK) return error; error = belle_sip_parameters_marshal(BELLE_SIP_PARAMETERS(replaces), buff, buff_size, offset); if (error != BELLE_SIP_OK) return error; return error; } BELLE_SIP_NEW_HEADER(header_replaces, parameters, BELLE_SIP_REPLACES); BELLE_SIP_PARSE_FULL(header_replaces); GET_SET_STRING(belle_sip_header_replaces, call_id); // clang-format off GET_SET_STRING_PARAM2(belle_sip_header_replaces, to-tag, to_tag); GET_SET_STRING_PARAM2(belle_sip_header_replaces, from-tag, from_tag); // clang-format on static void escaped_to_ascii(const char *a, char *b, size_t n) { size_t index_a = 0, index_b = 0; while (a[index_a] != '\0' && index_a < n) index_a += belle_sip_get_char(a + index_a, b + index_b++); } #define REPLACES_PREF_OFFSET (strlen(BELLE_SIP_REPLACES) + 2) belle_sip_header_replaces_t *belle_sip_header_replaces_create2(const char *escaped_replace) { belle_sip_header_replaces_t *replaces; size_t len = strlen(escaped_replace); char *out = reinterpret_cast(belle_sip_malloc0(REPLACES_PREF_OFFSET + len + 1)); strcpy(out, BELLE_SIP_REPLACES ": "); escaped_to_ascii(escaped_replace, out + REPLACES_PREF_OFFSET, len); /*now we can parse*/ replaces = belle_sip_header_replaces_parse(out); belle_sip_free(out); return replaces; } char *belle_sip_header_replaces_value_to_escaped_string(const belle_sip_header_replaces_t *replaces) { char buff[BELLE_SIP_MAX_TO_STRING_SIZE]; size_t buff_size = sizeof(buff); size_t offset = 0; belle_sip_error_code error = BELLE_SIP_OK; /*first, marshall callid/from/to tags*/ error = belle_sip_snprintf(buff, buff_size, &offset, "%s", replaces->call_id); if (error != BELLE_SIP_OK) return NULL; error = belle_sip_parameters_marshal(BELLE_SIP_PARAMETERS(replaces), buff, buff_size, &offset); if (error != BELLE_SIP_OK) return NULL; buff[offset] = '\0'; return strdup(buff); } belle_sip_header_replaces_t * belle_sip_header_replaces_create(const char *call_id, const char *from_tag, const char *to_tag) { belle_sip_header_replaces_t *replaces = belle_sip_header_replaces_new(); belle_sip_header_replaces_set_call_id(replaces, call_id); belle_sip_header_replaces_set_from_tag(replaces, from_tag); belle_sip_header_replaces_set_to_tag(replaces, to_tag); return replaces; } /****************************** * Date header inherits from header ******************************/ struct belle_sip_header_date { belle_sip_header_t base; char *date; }; static void belle_sip_header_date_destroy(belle_sip_header_date_t *obj) { DESTROY_STRING(obj, date); } static void belle_sip_header_date_clone(belle_sip_header_date_t *obj, const belle_sip_header_date_t *orig) { CLONE_STRING(belle_sip_header_date, date, obj, orig); } belle_sip_error_code belle_sip_header_date_marshal(belle_sip_header_date_t *obj, char *buff, size_t buff_size, size_t *offset) { belle_sip_error_code error = belle_sip_header_marshal(BELLE_SIP_HEADER(obj), buff, buff_size, offset); if (error != BELLE_SIP_OK) return error; error = belle_sip_snprintf(buff, buff_size, offset, "%s", obj->date); if (error != BELLE_SIP_OK) return error; return error; } BELLE_SIP_NEW_HEADER(header_date, header, BELLE_SIP_DATE); BELLE_SIP_PARSE_FULL(header_date); BELLESIP_EXPORT belle_sip_header_date_t *belle_sip_header_date_create_from_time(const time_t *utc_time) { belle_sip_header_date_t *obj = belle_sip_header_date_new(); belle_sip_header_date_set_time(obj, utc_time); return obj; } static const char *days[] = {"Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat"}; static const char *months[] = {"Jan", "Feb", "Mar", "Apr", "May", "Jun", "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"}; BELLESIP_EXPORT time_t belle_sip_header_date_get_time(belle_sip_header_date_t *obj) { struct tm ret = {0}; char tmp1[4] = {0}; char tmp2[17] = {0}; int i, j; time_t seconds; /* time headers are in GMT as spec says */ sscanf(obj->date, "%3c,%d %16s %d %d:%d:%d", tmp1, &ret.tm_mday, tmp2, &ret.tm_year, &ret.tm_hour, &ret.tm_min, &ret.tm_sec); ret.tm_year -= 1900; for (i = 0; i < 7; i++) { if (strcmp(tmp1, days[i]) == 0) { ret.tm_wday = i; for (j = 0; j < 12; j++) { if (strcmp(tmp2, months[j]) == 0) { ret.tm_mon = j; goto success; } } } } belle_sip_warning("Failed to parse date %s", obj->date); return (time_t)-1; success: ret.tm_isdst = 0; if (seconds == (time_t)-1) { belle_sip_error("timegm() failed: %s", strerror(errno)); return (time_t)-1; } return seconds; } BELLESIP_EXPORT void belle_sip_header_date_set_time(belle_sip_header_date_t *obj, const time_t *utc_time) { struct tm *ret; #ifndef _WIN32 struct tm gmt; ret = gmtime_r(utc_time, &gmt); #else ret = gmtime(utc_time); #endif /*cannot use strftime because it is locale dependant*/ if (obj->date) { belle_sip_free(obj->date); } /*SIP-date = rfc1123-date rfc1123-date = wkday "," SP date1 SP time SP "GMT" date1 = 2DIGIT SP month SP 4DIGIT ; day month year (e.g., 02 Jun 1982) time = 2DIGIT ":" 2DIGIT ":" 2DIGIT ; 00:00:00 - 23:59:59 wkday = "Mon" / "Tue" / "Wed" / "Thu" / "Fri" / "Sat" / "Sun" month = "Jan" / "Feb" / "Mar" / "Apr" / "May" / "Jun" / "Jul" / "Aug" / "Sep" / "Oct" / "Nov" / "Dec" */ obj->date = belle_sip_strdup_printf("%s, %02i %s %04i %02i:%02i:%02i GMT", days[ret->tm_wday], ret->tm_mday, months[ret->tm_mon], 1900 + ret->tm_year, ret->tm_hour, ret->tm_min, ret->tm_sec); } GET_SET_STRING(belle_sip_header_date, date); /************************ * header_p_prefered_identity ***********************/ struct _belle_sip_header_p_preferred_identity { belle_sip_header_address_t address; }; void belle_sip_header_p_preferred_identity_destroy(belle_sip_header_p_preferred_identity_t *p_preferred_identity) { } void belle_sip_header_p_preferred_identity_clone(belle_sip_header_p_preferred_identity_t *p_preferred_identity, const belle_sip_header_p_preferred_identity_t *orig) { } belle_sip_error_code belle_sip_header_p_preferred_identity_marshal( belle_sip_header_p_preferred_identity_t *p_preferred_identity, char *buff, size_t buff_size, size_t *offset) { belle_sip_error_code error = belle_sip_header_marshal(BELLE_SIP_HEADER(p_preferred_identity), buff, buff_size, offset); if (error != BELLE_SIP_OK) return error; error = belle_sip_header_address_marshal(&p_preferred_identity->address, buff, buff_size, offset); if (error != BELLE_SIP_OK) return error; return error; } BELLE_SIP_NEW_HEADER(header_p_preferred_identity, header_address, BELLE_SIP_P_PREFERRED_IDENTITY); BELLE_SIP_PARSE_HEADER_WITH_URI_CHECK(header_p_preferred_identity); belle_sip_header_p_preferred_identity_t * belle_sip_header_p_preferred_identity_create(const belle_sip_header_address_t *p_preferred_identity) { belle_sip_header_p_preferred_identity_t *header = belle_sip_header_p_preferred_identity_new(); _belle_sip_object_copy(BELLE_SIP_OBJECT(header), BELLE_SIP_OBJECT(p_preferred_identity)); belle_sip_header_set_next(BELLE_SIP_HEADER(header), NULL); /*make sure only one header is kept*/ belle_sip_header_set_name(BELLE_SIP_HEADER(header), BELLE_SIP_P_PREFERRED_IDENTITY); /*restaure header name*/ return header; } #define PRIVACY_LIKE_HEADER(header_name, string_name, separator) \ struct _belle_sip_header_##header_name { \ belle_sip_header_t header; \ belle_sip_list_t *header_name; \ }; \ \ static void belle_sip_header_##header_name##_destroy(belle_sip_header_##header_name##_t *p) { \ belle_sip_header_##header_name##_set_##header_name(p, NULL); \ } \ \ static void belle_sip_header_##header_name##_clone(belle_sip_header_##header_name##_t *p, \ const belle_sip_header_##header_name##_t *orig) { \ belle_sip_list_t *list = orig->header_name; \ for (; list != NULL; list = list->next) { \ belle_sip_header_##header_name##_add_##header_name(p, (const char *)list->data); \ } \ } \ \ belle_sip_error_code belle_sip_header_##header_name##_marshal(belle_sip_header_##header_name##_t *p, char *buff, \ size_t buff_size, size_t *offset) { \ belle_sip_error_code error = BELLE_SIP_OK; \ belle_sip_list_t *list = p->header_name; \ error = belle_sip_header_marshal(BELLE_SIP_HEADER(p), buff, buff_size, offset); \ if (error != BELLE_SIP_OK) return error; \ for (; list != NULL; list = list->next) { \ error = belle_sip_snprintf(buff, buff_size, offset, list == p->header_name ? "%s" : separator " %s", \ (const char *)list->data); \ if (error != BELLE_SIP_OK) return error; \ } \ return error; \ } \ \ BELLE_SIP_NEW_HEADER(header_##header_name, header, string_name); \ BELLE_SIP_PARSE_FULL(header_##header_name); \ belle_sip_list_t *belle_sip_header_##header_name##_get_##header_name( \ const belle_sip_header_##header_name##_t *p) { \ return p->header_name; \ } \ SET_ADD_STRING_LIST(belle_sip_header_##header_name, header_name); \ \ belle_sip_header_##header_name##_t *belle_sip_header_##header_name##_create(const char *header_name) { \ belle_sip_header_##header_name##_t *header = belle_sip_header_##header_name##_new(); \ belle_sip_header_##header_name##_add_##header_name(header, header_name); \ return header; \ } /****************************** * Privacy header inherits from header ******************************/ PRIVACY_LIKE_HEADER(privacy, BELLE_SIP_PRIVACY, ";"); /************************** * Event header object inherits from parameters ***************************/ struct _belle_sip_header_event { belle_sip_parameters_t parameters; const char *package_name; }; static void belle_sip_header_event_destroy(belle_sip_header_event_t *event) { DESTROY_STRING(event, package_name); } static void belle_sip_header_event_clone(belle_sip_header_event_t *event, const belle_sip_header_event_t *orig){ CLONE_STRING(belle_sip_header_event, package_name, event, orig)} belle_sip_error_code belle_sip_header_event_marshal(belle_sip_header_event_t *event, char *buff, size_t buff_size, size_t *offset) { belle_sip_error_code error = belle_sip_header_marshal(BELLE_SIP_HEADER(event), buff, buff_size, offset); if (error != BELLE_SIP_OK) return error; error = belle_sip_snprintf(buff, buff_size, offset, "%s", event->package_name); if (error != BELLE_SIP_OK) return error; error = belle_sip_parameters_marshal(BELLE_SIP_PARAMETERS(event), buff, buff_size, offset); if (error != BELLE_SIP_OK) return error; return error; } BELLE_SIP_NEW_HEADER(header_event, parameters, BELLE_SIP_EVENT); BELLE_SIP_PARSE_FULL(header_event); GET_SET_STRING(belle_sip_header_event, package_name); GET_SET_STRING_PARAM(belle_sip_header_event, id); belle_sip_header_event_t *belle_sip_header_event_create(const char *package_name) { belle_sip_header_event_t *event = belle_sip_header_event_new(); belle_sip_header_event_set_package_name(event, package_name); return event; } /****************************** * Supported header inherits from header ******************************/ PRIVACY_LIKE_HEADER(supported, BELLE_SIP_SUPPORTED, ","); int belle_sip_header_supported_contains_tag(const belle_sip_header_supported_t *supported, const char *tag) { belle_sip_list_t *list = belle_sip_header_supported_get_supported(supported); while (list) { if (strcmp(reinterpret_cast(list->data), tag) == 0) return TRUE; list = list->next; } return FALSE; } /****************************** * Require header inherits from header ******************************/ PRIVACY_LIKE_HEADER(require, BELLE_SIP_REQUIRE, ","); int belle_sip_header_require_contains_tag(const belle_sip_header_require_t *require, const char *tag) { const bctbx_list_t *elem; for (elem = require->require; elem != NULL; elem = elem->next) { if (strcmp((const char *)elem->data, tag) == 0) return TRUE; } return FALSE; } /****************************** * Content-Disposition header inherits from parameters ******************************/ struct _belle_sip_header_content_disposition { belle_sip_parameters_t parameters; const char *content_disposition; }; static void belle_sip_header_content_disposition_destroy(belle_sip_header_content_disposition_t *content_disposition) { DESTROY_STRING(content_disposition, content_disposition); } static void belle_sip_header_content_disposition_clone(belle_sip_header_content_disposition_t *content_disposition, const belle_sip_header_content_disposition_t *orig){ CLONE_STRING(belle_sip_header_content_disposition, content_disposition, content_disposition, orig)} belle_sip_error_code belle_sip_header_content_disposition_marshal( belle_sip_header_content_disposition_t *content_disposition, char *buff, size_t buff_size, size_t *offset) { belle_sip_error_code error = belle_sip_header_marshal(BELLE_SIP_HEADER(content_disposition), buff, buff_size, offset); if (error != BELLE_SIP_OK) return error; error = belle_sip_snprintf(buff, buff_size, offset, "%s", content_disposition->content_disposition); if (error != BELLE_SIP_OK) return error; error = belle_sip_parameters_marshal(BELLE_SIP_PARAMETERS(content_disposition), buff, buff_size, offset); if (error != BELLE_SIP_OK) return error; return error; } BELLE_SIP_NEW_HEADER(header_content_disposition, parameters, BELLE_SIP_CONTENT_DISPOSITION); BELLE_SIP_PARSE_FULL(header_content_disposition); GET_SET_STRING(belle_sip_header_content_disposition, content_disposition); belle_sip_header_content_disposition_t *belle_sip_header_content_disposition_create(const char *value) { belle_sip_header_content_disposition_t *header = belle_sip_header_content_disposition_new(); belle_sip_header_content_disposition_set_content_disposition(header, value); return header; } /****************************** * Accept header inherits from parameters ******************************/ struct _belle_sip_header_accept { belle_sip_parameters_t params_list; const char *type; const char *subtype; }; static void belle_sip_header_accept_destroy(belle_sip_header_accept_t *accept) { if (accept->type) belle_sip_free((void *)accept->type); if (accept->subtype) belle_sip_free((void *)accept->subtype); } static void belle_sip_header_accept_clone(belle_sip_header_accept_t *accept, const belle_sip_header_accept_t *orig) { CLONE_STRING(belle_sip_header_accept, type, accept, orig); CLONE_STRING(belle_sip_header_accept, subtype, accept, orig); } belle_sip_error_code belle_sip_header_accept_marshal(belle_sip_header_accept_t *accept, char *buff, size_t buff_size, size_t *offset) { belle_sip_error_code error = belle_sip_header_marshal(BELLE_SIP_HEADER(accept), buff, buff_size, offset); if (error != BELLE_SIP_OK) return error; error = belle_sip_snprintf(buff, buff_size, offset, "%s/%s", accept->type, accept->subtype); if (error != BELLE_SIP_OK) return error; error = belle_sip_parameters_marshal(&accept->params_list, buff, buff_size, offset); if (error != BELLE_SIP_OK) return error; return error; } BELLE_SIP_NEW_HEADER(header_accept, parameters, BELLE_SIP_ACCEPT); BELLE_SIP_PARSE_WITH_CONTEXT(header_accept); belle_sip_header_accept_t *belle_sip_header_accept_create(const char *type, const char *sub_type) { belle_sip_header_accept_t *header = belle_sip_header_accept_new(); belle_sip_header_accept_set_type(header, type); belle_sip_header_accept_set_subtype(header, sub_type); return header; } GET_SET_STRING(belle_sip_header_accept, type); GET_SET_STRING(belle_sip_header_accept, subtype); /****************************** * Reason header object inherits from parameters ******************************/ struct _belle_sip_header_reason { belle_sip_parameters_t params_list; const char *protocol; const char *unquoted_text; }; static void belle_sip_header_reason_destroy(belle_sip_header_reason_t *reason) { DESTROY_STRING(reason, protocol); DESTROY_STRING(reason, unquoted_text); } static void belle_sip_header_reason_clone(belle_sip_header_reason_t *reason, const belle_sip_header_reason_t *orig){ CLONE_STRING(belle_sip_header_reason, protocol, reason, orig)} belle_sip_error_code belle_sip_header_reason_marshal(belle_sip_header_reason_t *reason, char *buff, size_t buff_size, size_t *offset) { belle_sip_error_code error = belle_sip_header_marshal(BELLE_SIP_HEADER(reason), buff, buff_size, offset); if (error != BELLE_SIP_OK) return error; error = belle_sip_snprintf(buff, buff_size, offset, "%s ", reason->protocol); if (error != BELLE_SIP_OK) return error; error = belle_sip_parameters_marshal(BELLE_SIP_PARAMETERS(reason), buff, buff_size, offset); if (error != BELLE_SIP_OK) return error; if (reason->unquoted_text) error = belle_sip_snprintf(buff, buff_size, offset, "; text=\"%s\"", reason->unquoted_text); return error; } GET_SET_STRING(belle_sip_header_reason, unquoted_text); void belle_sip_header_reason_set_text(belle_sip_header_reason_t *reason, const char *text) { belle_sip_parameters_remove_parameter(BELLE_SIP_PARAMETERS(reason), "text"); /*just in case*/ belle_sip_header_reason_set_unquoted_text((belle_sip_header_reason_t *)reason, text); } BELLESIP_EXPORT const char *belle_sip_header_reason_get_text(const belle_sip_header_reason_t *reason) { if (!reason->unquoted_text) { /*try from params*/ const char *quoted = belle_sip_parameters_get_parameter(BELLE_SIP_PARAMETERS(reason), "text"); if (quoted) { char *unquoted = _belle_sip_str_dup_and_unquote_string(quoted); belle_sip_header_reason_set_unquoted_text((belle_sip_header_reason_t *)reason, unquoted); /*change internal param, ,even if reason is const*/ belle_sip_parameters_remove_parameter(BELLE_SIP_PARAMETERS(reason), "text"); belle_sip_free(unquoted); } } return reason->unquoted_text; } GET_SET_STRING(belle_sip_header_reason, protocol); GET_SET_INT_PARAM(belle_sip_header_reason, cause, int); BELLE_SIP_PARSE_WITH_CONTEXT(header_reason); BELLE_SIP_NEW_HEADER(header_reason, parameters, BELLE_SIP_REASON); /****************************** * AuthenticationInfo header inherits from header ******************************/ struct _belle_sip_header_authentication_info { belle_sip_header_t header; const char *rsp_auth; const char *cnonce; int nonce_count; const char *qop; const char *next_nonce; }; static void belle_sip_header_authentication_info_destroy(belle_sip_header_authentication_info_t *authentication_info) { DESTROY_STRING(authentication_info, rsp_auth); DESTROY_STRING(authentication_info, cnonce); DESTROY_STRING(authentication_info, qop); DESTROY_STRING(authentication_info, next_nonce); } static void belle_sip_header_authentication_info_clone(belle_sip_header_authentication_info_t *authentication_info, const belle_sip_header_authentication_info_t *orig){ CLONE_STRING(belle_sip_header_authentication_info, rsp_auth, authentication_info, orig) CLONE_STRING(belle_sip_header_authentication_info, cnonce, authentication_info, orig) CLONE_STRING(belle_sip_header_authentication_info, qop, authentication_info, orig) CLONE_STRING(belle_sip_header_authentication_info, next_nonce, authentication_info, orig)} belle_sip_error_code belle_sip_header_authentication_info_marshal( belle_sip_header_authentication_info_t *authentication_info, char *buff, size_t buff_size, size_t *offset) { std::string border = ""; belle_sip_error_code error = belle_sip_header_marshal(BELLE_SIP_HEADER(authentication_info), buff, buff_size, offset); if (error != BELLE_SIP_OK) return error; if (authentication_info->rsp_auth) { error = belle_sip_snprintf(buff, buff_size, offset, "%srspauth=\"%s\"", border.c_str(), authentication_info->rsp_auth); border = ", "; } if (error != BELLE_SIP_OK) return error; if (authentication_info->cnonce) { error = belle_sip_snprintf(buff, buff_size, offset, "%scnonce=\"%s\"", border.c_str(), authentication_info->cnonce); border = ", "; } if (error != BELLE_SIP_OK) return error; if (authentication_info->nonce_count >= 0) { error = belle_sip_snprintf(buff, buff_size, offset, "%snc=%08x", border.c_str(), authentication_info->nonce_count); border = ", "; } if (error != BELLE_SIP_OK) return error; if (authentication_info->qop) { error = belle_sip_snprintf(buff, buff_size, offset, "%sqop=%s", border.c_str(), authentication_info->qop); border = ", "; } if (error != BELLE_SIP_OK) return error; if (authentication_info->next_nonce) { error = belle_sip_snprintf(buff, buff_size, offset, "%snextnonce=\"%s\"", border.c_str(), authentication_info->next_nonce); } return error; } void belle_sip_header_authentication_info_init(belle_sip_header_authentication_info_t *header_authentication) { header_authentication->nonce_count = -1; } BELLE_SIP_NEW_HEADER_INIT(header_authentication_info, header, BELLE_SIP_AUTHENTICATION_INFO, header_authentication_info); BELLE_SIP_PARSE_FULL(header_authentication_info); GET_SET_STRING(belle_sip_header_authentication_info, rsp_auth); GET_SET_STRING(belle_sip_header_authentication_info, qop); GET_SET_STRING(belle_sip_header_authentication_info, next_nonce); GET_SET_STRING(belle_sip_header_authentication_info, cnonce); GET_SET_INT(belle_sip_header_authentication_info, nonce_count, int);