asterisk/res/res_pjsip/pjsip_global_headers.c

188 lines
4.4 KiB
C

/*
* Asterisk -- An open source telephony toolkit.
*
* Copyright (C) 2013, Digium, Inc.
*
* Mark Michelson <mmichelson@digium.com>
*
* See http://www.asterisk.org for more information about
* the Asterisk project. Please do not directly contact
* any of the maintainers of this project for assistance;
* the project provides a web site, mailing lists and IRC
* channels for your use.
*
* This program is free software, distributed under the terms of
* the GNU General Public License Version 2. See the LICENSE file
* at the top of the source tree.
*/
#include "asterisk.h"
#include <pjsip.h>
#include <pjlib.h>
#include "asterisk/res_pjsip.h"
#include "asterisk/linkedlists.h"
#include "include/res_pjsip_private.h"
static pj_status_t add_request_headers(pjsip_tx_data *tdata);
static pj_status_t add_response_headers(pjsip_tx_data *tdata);
/*!
* \brief Indicator we've already handled a specific request/response
*
* PJSIP tends to reuse requests and responses. If we already have added
* headers to a request or response, we mark the message with this value
* so that we know not to re-add the headers again.
*/
static unsigned int handled_id = 0xCA115785;
static pjsip_module global_header_mod = {
.name = {"Global headers", 13},
.priority = PJSIP_MOD_PRIORITY_APPLICATION,
.on_tx_request = add_request_headers,
.on_tx_response = add_response_headers,
};
struct header {
AST_DECLARE_STRING_FIELDS(
AST_STRING_FIELD(name);
AST_STRING_FIELD(value);
);
AST_LIST_ENTRY(header) next;
};
static struct header *alloc_header(const char *name, const char *value)
{
struct header *alloc;
alloc = ast_calloc_with_stringfields(1, struct header, 32);
if (!alloc) {
return NULL;
}
ast_string_field_set(alloc, name, name);
ast_string_field_set(alloc, value, value);
return alloc;
}
static void destroy_header(struct header *to_destroy)
{
ast_string_field_free_memory(to_destroy);
ast_free(to_destroy);
}
AST_RWLIST_HEAD(header_list, header);
static struct header_list request_headers;
static struct header_list response_headers;
static void add_headers_to_message(struct header_list *headers, pjsip_tx_data *tdata)
{
struct header *iter;
SCOPED_LOCK(lock, headers, AST_RWLIST_RDLOCK, AST_RWLIST_UNLOCK);
if (tdata->mod_data[global_header_mod.id] == &handled_id) {
return;
}
AST_LIST_TRAVERSE(headers, iter, next) {
pj_str_t name;
pjsip_generic_string_hdr *hdr;
hdr = pjsip_msg_find_hdr_by_name(tdata->msg, pj_cstr(&name, iter->name), NULL);
if (hdr) {
continue;
}
ast_sip_add_header(tdata, iter->name, iter->value);
};
tdata->mod_data[global_header_mod.id] = &handled_id;
}
static pj_status_t add_request_headers(pjsip_tx_data *tdata)
{
add_headers_to_message(&request_headers, tdata);
return PJ_SUCCESS;
}
static pj_status_t add_response_headers(pjsip_tx_data *tdata)
{
add_headers_to_message(&response_headers, tdata);
return PJ_SUCCESS;
}
static void remove_header(struct header_list *headers, const char *to_remove)
{
struct header *iter;
AST_LIST_TRAVERSE_SAFE_BEGIN(headers, iter, next) {
if (!strcasecmp(iter->name, to_remove)) {
AST_LIST_REMOVE_CURRENT(next);
destroy_header(iter);
break;
}
}
AST_LIST_TRAVERSE_SAFE_END;
}
static int add_header(struct header_list *headers, const char *name, const char *value, int replace)
{
struct header *to_add = NULL;
if (!ast_strlen_zero(value)) {
to_add = alloc_header(name, value);
if (!to_add) {
return -1;
}
}
AST_RWLIST_WRLOCK(headers);
if (replace) {
remove_header(headers, name);
}
if (to_add) {
AST_LIST_INSERT_TAIL(headers, to_add, next);
}
AST_RWLIST_UNLOCK(headers);
return 0;
}
int ast_sip_add_global_request_header(const char *name, const char *value, int replace)
{
return add_header(&request_headers, name, value, replace);
}
int ast_sip_add_global_response_header(const char *name, const char *value, int replace)
{
return add_header(&response_headers, name, value, replace);
}
void ast_sip_initialize_global_headers(void)
{
AST_RWLIST_HEAD_INIT(&request_headers);
AST_RWLIST_HEAD_INIT(&response_headers);
ast_sip_register_service(&global_header_mod);
}
static void destroy_headers(struct header_list *headers)
{
struct header *iter;
while ((iter = AST_RWLIST_REMOVE_HEAD(headers, next))) {
destroy_header(iter);
}
AST_RWLIST_HEAD_DESTROY(headers);
}
void ast_sip_destroy_global_headers(void)
{
destroy_headers(&request_headers);
destroy_headers(&response_headers);
ast_sip_unregister_service(&global_header_mod);
}