| 
									
										
										
										
											2013-03-08 15:15:13 +00:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * 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"
 | 
					
						
							| 
									
										
										
										
											2018-08-06 08:36:22 -03:00
										 |  |  | #include "asterisk/hashtab.h"
 | 
					
						
							| 
									
										
										
										
											2013-03-08 15:15:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-15 17:35:16 +00:00
										 |  |  | /*! \internal */ | 
					
						
							| 
									
										
										
										
											2013-03-08 15:15:13 +00:00
										 |  |  | struct stasis_message_type { | 
					
						
							| 
									
										
										
										
											2013-06-11 15:46:35 +00:00
										 |  |  | 	struct stasis_message_vtable *vtable; | 
					
						
							| 
									
										
										
										
											2013-03-08 15:15:13 +00:00
										 |  |  | 	char *name; | 
					
						
							| 
									
										
										
										
											2018-08-06 08:36:22 -03:00
										 |  |  | 	unsigned int hash; | 
					
						
							| 
									
										
										
										
											2018-09-23 17:50:01 -03:00
										 |  |  | 	int id; | 
					
						
							| 
									
										
										
										
											2018-11-29 08:53:51 -07:00
										 |  |  | 	enum stasis_subscription_message_formatters available_formatters; | 
					
						
							| 
									
										
										
										
											2013-03-08 15:15:13 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-11 15:46:35 +00:00
										 |  |  | static struct stasis_message_vtable null_vtable = {}; | 
					
						
							| 
									
										
										
										
											2018-09-23 17:50:01 -03:00
										 |  |  | static int message_type_id; | 
					
						
							| 
									
										
										
										
											2013-06-11 15:46:35 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-08 15:15:13 +00:00
										 |  |  | static void message_type_dtor(void *obj) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct stasis_message_type *type = obj; | 
					
						
							|  |  |  | 	ast_free(type->name); | 
					
						
							|  |  |  | 	type->name = NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-09 08:41:32 -06:00
										 |  |  | enum stasis_message_type_result stasis_message_type_create(const char *name, | 
					
						
							| 
									
										
										
										
											2014-08-06 12:55:28 +00:00
										 |  |  | 	struct stasis_message_vtable *vtable, | 
					
						
							|  |  |  | 	struct stasis_message_type **result) | 
					
						
							| 
									
										
										
										
											2013-03-08 15:15:13 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2014-03-07 20:41:13 +00:00
										 |  |  | 	struct stasis_message_type *type; | 
					
						
							| 
									
										
										
										
											2013-03-08 15:15:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-08-06 12:55:28 +00:00
										 |  |  | 	/* Check for declination */ | 
					
						
							|  |  |  | 	if (name && stasis_message_type_declined(name)) { | 
					
						
							|  |  |  | 		return STASIS_MESSAGE_TYPE_DECLINED; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-14 15:48:24 -05:00
										 |  |  | 	type = ao2_t_alloc_options(sizeof(*type), message_type_dtor, | 
					
						
							|  |  |  | 		AO2_ALLOC_OPT_LOCK_NOLOCK, name ?: ""); | 
					
						
							| 
									
										
										
										
											2013-03-08 15:15:13 +00:00
										 |  |  | 	if (!type) { | 
					
						
							| 
									
										
										
										
											2014-08-06 12:55:28 +00:00
										 |  |  | 		return STASIS_MESSAGE_TYPE_ERROR; | 
					
						
							| 
									
										
										
										
											2013-03-08 15:15:13 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2013-06-11 15:46:35 +00:00
										 |  |  | 	if (!vtable) { | 
					
						
							|  |  |  | 		/* Null object pattern, FTW! */ | 
					
						
							|  |  |  | 		vtable = &null_vtable; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2013-03-08 15:15:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	type->name = ast_strdup(name); | 
					
						
							|  |  |  | 	if (!type->name) { | 
					
						
							| 
									
										
										
										
											2014-03-07 20:41:13 +00:00
										 |  |  | 		ao2_cleanup(type); | 
					
						
							| 
									
										
										
										
											2014-08-06 12:55:28 +00:00
										 |  |  | 		return STASIS_MESSAGE_TYPE_ERROR; | 
					
						
							| 
									
										
										
										
											2013-03-08 15:15:13 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-08-06 08:36:22 -03:00
										 |  |  | 	type->hash = ast_hashtab_hash_string(name); | 
					
						
							| 
									
										
										
										
											2013-06-11 15:46:35 +00:00
										 |  |  | 	type->vtable = vtable; | 
					
						
							| 
									
										
										
										
											2018-11-29 08:53:51 -07:00
										 |  |  | 	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; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-09-23 17:50:01 -03:00
										 |  |  | 	type->id = ast_atomic_fetchadd_int(&message_type_id, +1); | 
					
						
							| 
									
										
										
										
											2014-08-06 12:55:28 +00:00
										 |  |  | 	*result = type; | 
					
						
							| 
									
										
										
										
											2013-03-08 15:15:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-08-06 12:55:28 +00:00
										 |  |  | 	return STASIS_MESSAGE_TYPE_SUCCESS; | 
					
						
							| 
									
										
										
										
											2013-03-08 15:15:13 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const char *stasis_message_type_name(const struct stasis_message_type *type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return type->name; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-06 08:36:22 -03:00
										 |  |  | unsigned int stasis_message_type_hash(const struct stasis_message_type *type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return type->hash; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-23 17:50:01 -03:00
										 |  |  | int stasis_message_type_id(const struct stasis_message_type *type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return type->id; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-29 08:53:51 -07:00
										 |  |  | enum stasis_subscription_message_formatters stasis_message_type_available_formatters( | 
					
						
							|  |  |  | 	const struct stasis_message_type *type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return type->available_formatters; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-15 17:35:16 +00:00
										 |  |  | /*! \internal */ | 
					
						
							| 
									
										
										
										
											2013-03-08 15:15:13 +00:00
										 |  |  | struct stasis_message { | 
					
						
							|  |  |  | 	/*! Time the message was created */ | 
					
						
							|  |  |  | 	struct timeval timestamp; | 
					
						
							|  |  |  | 	/*! Type of the message */ | 
					
						
							|  |  |  | 	struct stasis_message_type *type; | 
					
						
							| 
									
										
										
										
											2014-03-07 20:41:13 +00:00
										 |  |  | 	/*! Where this message originated.  NULL if aggregate message. */ | 
					
						
							|  |  |  | 	const struct ast_eid *eid_ptr; | 
					
						
							| 
									
										
										
										
											2013-03-08 15:15:13 +00:00
										 |  |  | 	/*! Message content */ | 
					
						
							|  |  |  | 	void *data; | 
					
						
							| 
									
										
										
										
											2014-03-07 20:41:13 +00:00
										 |  |  | 	/*! Where this message originated. */ | 
					
						
							|  |  |  | 	struct ast_eid eid; | 
					
						
							| 
									
										
										
										
											2013-03-08 15:15:13 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void stasis_message_dtor(void *obj) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct stasis_message *message = obj; | 
					
						
							|  |  |  | 	ao2_cleanup(message->data); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-07 20:41:13 +00:00
										 |  |  | struct stasis_message *stasis_message_create_full(struct stasis_message_type *type, void *data, const struct ast_eid *eid) | 
					
						
							| 
									
										
										
										
											2013-03-08 15:15:13 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2014-03-07 20:41:13 +00:00
										 |  |  | 	struct stasis_message *message; | 
					
						
							| 
									
										
										
										
											2013-03-08 15:15:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (type == NULL || data == NULL) { | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-14 15:48:24 -05:00
										 |  |  | 	message = ao2_t_alloc_options(sizeof(*message), stasis_message_dtor, | 
					
						
							|  |  |  | 		AO2_ALLOC_OPT_LOCK_NOLOCK, type->name); | 
					
						
							| 
									
										
										
										
											2013-03-08 15:15:13 +00:00
										 |  |  | 	if (message == NULL) { | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	message->timestamp = ast_tvnow(); | 
					
						
							| 
									
										
										
										
											2018-09-14 15:51:41 -05:00
										 |  |  | 	/*
 | 
					
						
							|  |  |  | 	 * 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. | 
					
						
							|  |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2013-03-08 15:15:13 +00:00
										 |  |  | 	message->type = type; | 
					
						
							|  |  |  | 	ao2_ref(data, +1); | 
					
						
							|  |  |  | 	message->data = data; | 
					
						
							| 
									
										
										
										
											2014-03-07 20:41:13 +00:00
										 |  |  | 	if (eid) { | 
					
						
							|  |  |  | 		message->eid_ptr = &message->eid; | 
					
						
							|  |  |  | 		message->eid = *eid; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2013-03-08 15:15:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return message; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-07 20:41:13 +00:00
										 |  |  | 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; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-08 15:15:13 +00:00
										 |  |  | 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; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2013-06-11 15:46:35 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-12 15:29:52 -05:00
										 |  |  | #define INVOKE_VIRTUAL(fn, ...)					\
 | 
					
						
							|  |  |  | 	({											\ | 
					
						
							|  |  |  | 		if (!msg) {								\ | 
					
						
							|  |  |  | 			return NULL;						\ | 
					
						
							|  |  |  | 		}										\ | 
					
						
							|  |  |  | 		ast_assert(msg->type != NULL);			\ | 
					
						
							| 
									
										
										
										
											2013-06-11 15:46:35 +00:00
										 |  |  | 		ast_assert(msg->type->vtable != NULL);	\ | 
					
						
							| 
									
										
										
										
											2016-04-12 15:29:52 -05:00
										 |  |  | 		if (!msg->type->vtable->fn) {			\ | 
					
						
							|  |  |  | 			return NULL;						\ | 
					
						
							|  |  |  | 		}										\ | 
					
						
							|  |  |  | 		msg->type->vtable->fn(__VA_ARGS__);		\ | 
					
						
							| 
									
										
										
										
											2013-06-11 15:46:35 +00:00
										 |  |  | 	}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct ast_manager_event_blob *stasis_message_to_ami(struct stasis_message *msg) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return INVOKE_VIRTUAL(to_ami, msg); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-22 20:10:46 +00:00
										 |  |  | struct ast_json *stasis_message_to_json( | 
					
						
							|  |  |  | 	struct stasis_message *msg, | 
					
						
							|  |  |  | 	struct stasis_message_sanitizer *sanitize) | 
					
						
							| 
									
										
										
										
											2013-06-11 15:46:35 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-11-22 20:10:46 +00:00
										 |  |  | 	return INVOKE_VIRTUAL(to_json, msg, sanitize); | 
					
						
							| 
									
										
										
										
											2013-06-11 15:46:35 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2014-05-22 12:01:37 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | struct ast_event *stasis_message_to_event(struct stasis_message *msg) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return INVOKE_VIRTUAL(to_event, msg); | 
					
						
							| 
									
										
										
										
											2014-07-18 19:55:24 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-04-12 15:29:52 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | #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); | 
					
						
							|  |  |  | } |