mirror of
				https://github.com/asterisk/asterisk.git
				synced 2025-11-04 05:15:22 +00:00 
			
		
		
		
	A subscriber can now indicate that it only wants messages that have formatters of a specific type. For instance, manager can indicate that it only wants messages that have a "to_ami" formatter. You can combine this with the existing filter for message type to get only messages with specific formatters or messages of specific types. ASTERISK-28186 Change-Id: Ifdb7a222a73b6b56c6bb9e4ee93dc8a394a5494c
		
			
				
	
	
		
			255 lines
		
	
	
		
			5.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			255 lines
		
	
	
		
			5.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/*
 | 
						|
 * Asterisk -- An open source telephony toolkit.
 | 
						|
 *
 | 
						|
 * Copyright (C) 2013, Digium, Inc.
 | 
						|
 *
 | 
						|
 * David M. Lee, II <dlee@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.
 | 
						|
 */
 | 
						|
 | 
						|
/*! \file
 | 
						|
 *
 | 
						|
 * \brief Stasis Message API.
 | 
						|
 *
 | 
						|
 * \author David M. Lee, II <dlee@digium.com>
 | 
						|
 */
 | 
						|
 | 
						|
/*** MODULEINFO
 | 
						|
	<support_level>core</support_level>
 | 
						|
 ***/
 | 
						|
 | 
						|
#include "asterisk.h"
 | 
						|
 | 
						|
#include "asterisk/astobj2.h"
 | 
						|
#include "asterisk/stasis.h"
 | 
						|
#include "asterisk/utils.h"
 | 
						|
#include "asterisk/hashtab.h"
 | 
						|
 | 
						|
/*! \internal */
 | 
						|
struct stasis_message_type {
 | 
						|
	struct stasis_message_vtable *vtable;
 | 
						|
	char *name;
 | 
						|
	unsigned int hash;
 | 
						|
	int id;
 | 
						|
	enum stasis_subscription_message_formatters available_formatters;
 | 
						|
};
 | 
						|
 | 
						|
static struct stasis_message_vtable null_vtable = {};
 | 
						|
static int message_type_id;
 | 
						|
 | 
						|
static void message_type_dtor(void *obj)
 | 
						|
{
 | 
						|
	struct stasis_message_type *type = obj;
 | 
						|
	ast_free(type->name);
 | 
						|
	type->name = NULL;
 | 
						|
}
 | 
						|
 | 
						|
int stasis_message_type_create(const char *name,
 | 
						|
	struct stasis_message_vtable *vtable,
 | 
						|
	struct stasis_message_type **result)
 | 
						|
{
 | 
						|
	struct stasis_message_type *type;
 | 
						|
 | 
						|
	/* Check for declination */
 | 
						|
	if (name && stasis_message_type_declined(name)) {
 | 
						|
		return STASIS_MESSAGE_TYPE_DECLINED;
 | 
						|
	}
 | 
						|
 | 
						|
	type = ao2_t_alloc_options(sizeof(*type), message_type_dtor,
 | 
						|
		AO2_ALLOC_OPT_LOCK_NOLOCK, name ?: "");
 | 
						|
	if (!type) {
 | 
						|
		return STASIS_MESSAGE_TYPE_ERROR;
 | 
						|
	}
 | 
						|
	if (!vtable) {
 | 
						|
		/* Null object pattern, FTW! */
 | 
						|
		vtable = &null_vtable;
 | 
						|
	}
 | 
						|
 | 
						|
	type->name = ast_strdup(name);
 | 
						|
	if (!type->name) {
 | 
						|
		ao2_cleanup(type);
 | 
						|
		return STASIS_MESSAGE_TYPE_ERROR;
 | 
						|
	}
 | 
						|
	type->hash = ast_hashtab_hash_string(name);
 | 
						|
	type->vtable = vtable;
 | 
						|
	if (vtable->to_json) {
 | 
						|
		type->available_formatters |= STASIS_SUBSCRIPTION_FORMATTER_JSON;
 | 
						|
	}
 | 
						|
	if (vtable->to_ami) {
 | 
						|
		type->available_formatters |= STASIS_SUBSCRIPTION_FORMATTER_AMI;
 | 
						|
	}
 | 
						|
	if (vtable->to_event) {
 | 
						|
		type->available_formatters |= STASIS_SUBSCRIPTION_FORMATTER_EVENT;
 | 
						|
	}
 | 
						|
	type->id = ast_atomic_fetchadd_int(&message_type_id, +1);
 | 
						|
	*result = type;
 | 
						|
 | 
						|
	return STASIS_MESSAGE_TYPE_SUCCESS;
 | 
						|
}
 | 
						|
 | 
						|
const char *stasis_message_type_name(const struct stasis_message_type *type)
 | 
						|
{
 | 
						|
	return type->name;
 | 
						|
}
 | 
						|
 | 
						|
unsigned int stasis_message_type_hash(const struct stasis_message_type *type)
 | 
						|
{
 | 
						|
	return type->hash;
 | 
						|
}
 | 
						|
 | 
						|
int stasis_message_type_id(const struct stasis_message_type *type)
 | 
						|
{
 | 
						|
	return type->id;
 | 
						|
}
 | 
						|
 | 
						|
enum stasis_subscription_message_formatters stasis_message_type_available_formatters(
 | 
						|
	const struct stasis_message_type *type)
 | 
						|
{
 | 
						|
	return type->available_formatters;
 | 
						|
}
 | 
						|
 | 
						|
/*! \internal */
 | 
						|
struct stasis_message {
 | 
						|
	/*! Time the message was created */
 | 
						|
	struct timeval timestamp;
 | 
						|
	/*! Type of the message */
 | 
						|
	struct stasis_message_type *type;
 | 
						|
	/*! Where this message originated.  NULL if aggregate message. */
 | 
						|
	const struct ast_eid *eid_ptr;
 | 
						|
	/*! Message content */
 | 
						|
	void *data;
 | 
						|
	/*! Where this message originated. */
 | 
						|
	struct ast_eid eid;
 | 
						|
};
 | 
						|
 | 
						|
static void stasis_message_dtor(void *obj)
 | 
						|
{
 | 
						|
	struct stasis_message *message = obj;
 | 
						|
	ao2_cleanup(message->data);
 | 
						|
}
 | 
						|
 | 
						|
struct stasis_message *stasis_message_create_full(struct stasis_message_type *type, void *data, const struct ast_eid *eid)
 | 
						|
{
 | 
						|
	struct stasis_message *message;
 | 
						|
 | 
						|
	if (type == NULL || data == NULL) {
 | 
						|
		return NULL;
 | 
						|
	}
 | 
						|
 | 
						|
	message = ao2_t_alloc_options(sizeof(*message), stasis_message_dtor,
 | 
						|
		AO2_ALLOC_OPT_LOCK_NOLOCK, type->name);
 | 
						|
	if (message == NULL) {
 | 
						|
		return NULL;
 | 
						|
	}
 | 
						|
 | 
						|
	message->timestamp = ast_tvnow();
 | 
						|
	/*
 | 
						|
	 * XXX Normal ao2 ref counting rules says we should increment the message
 | 
						|
	 * type ref here and decrement it in stasis_message_dtor().  However, the
 | 
						|
	 * stasis message could be cached and legitimately cause the type ref count
 | 
						|
	 * to hit the excessive ref count assertion.  Since the message type
 | 
						|
	 * practically has to be a global object anyway, we can get away with not
 | 
						|
	 * holding a ref in the stasis message.
 | 
						|
	 */
 | 
						|
	message->type = type;
 | 
						|
	ao2_ref(data, +1);
 | 
						|
	message->data = data;
 | 
						|
	if (eid) {
 | 
						|
		message->eid_ptr = &message->eid;
 | 
						|
		message->eid = *eid;
 | 
						|
	}
 | 
						|
 | 
						|
	return message;
 | 
						|
}
 | 
						|
 | 
						|
struct stasis_message *stasis_message_create(struct stasis_message_type *type, void *data)
 | 
						|
{
 | 
						|
	return stasis_message_create_full(type, data, &ast_eid_default);
 | 
						|
}
 | 
						|
 | 
						|
const struct ast_eid *stasis_message_eid(const struct stasis_message *msg)
 | 
						|
{
 | 
						|
	if (msg == NULL) {
 | 
						|
		return NULL;
 | 
						|
	}
 | 
						|
	return msg->eid_ptr;
 | 
						|
}
 | 
						|
 | 
						|
struct stasis_message_type *stasis_message_type(const struct stasis_message *msg)
 | 
						|
{
 | 
						|
	if (msg == NULL) {
 | 
						|
		return NULL;
 | 
						|
	}
 | 
						|
	return msg->type;
 | 
						|
}
 | 
						|
 | 
						|
void *stasis_message_data(const struct stasis_message *msg)
 | 
						|
{
 | 
						|
	if (msg == NULL) {
 | 
						|
		return NULL;
 | 
						|
	}
 | 
						|
	return msg->data;
 | 
						|
}
 | 
						|
 | 
						|
const struct timeval *stasis_message_timestamp(const struct stasis_message *msg)
 | 
						|
{
 | 
						|
	if (msg == NULL) {
 | 
						|
		return NULL;
 | 
						|
	}
 | 
						|
	return &msg->timestamp;
 | 
						|
}
 | 
						|
 | 
						|
#define INVOKE_VIRTUAL(fn, ...)					\
 | 
						|
	({											\
 | 
						|
		if (!msg) {								\
 | 
						|
			return NULL;						\
 | 
						|
		}										\
 | 
						|
		ast_assert(msg->type != NULL);			\
 | 
						|
		ast_assert(msg->type->vtable != NULL);	\
 | 
						|
		if (!msg->type->vtable->fn) {			\
 | 
						|
			return NULL;						\
 | 
						|
		}										\
 | 
						|
		msg->type->vtable->fn(__VA_ARGS__);		\
 | 
						|
	})
 | 
						|
 | 
						|
struct ast_manager_event_blob *stasis_message_to_ami(struct stasis_message *msg)
 | 
						|
{
 | 
						|
	return INVOKE_VIRTUAL(to_ami, msg);
 | 
						|
}
 | 
						|
 | 
						|
struct ast_json *stasis_message_to_json(
 | 
						|
	struct stasis_message *msg,
 | 
						|
	struct stasis_message_sanitizer *sanitize)
 | 
						|
{
 | 
						|
	return INVOKE_VIRTUAL(to_json, msg, sanitize);
 | 
						|
}
 | 
						|
 | 
						|
struct ast_event *stasis_message_to_event(struct stasis_message *msg)
 | 
						|
{
 | 
						|
	return INVOKE_VIRTUAL(to_event, msg);
 | 
						|
}
 | 
						|
 | 
						|
#define HAS_VIRTUAL(fn, msg)					\
 | 
						|
	({											\
 | 
						|
		if (!msg) {								\
 | 
						|
			return 0;							\
 | 
						|
		}										\
 | 
						|
		ast_assert(msg->type != NULL);			\
 | 
						|
		ast_assert(msg->type->vtable != NULL);	\
 | 
						|
		!!msg->type->vtable->fn;				\
 | 
						|
	})
 | 
						|
 | 
						|
int stasis_message_can_be_ami(struct stasis_message *msg)
 | 
						|
{
 | 
						|
	return HAS_VIRTUAL(to_ami, msg);
 | 
						|
}
 |