| 
									
										
										
										
											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"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ASTERISK_FILE_VERSION(__FILE__, "$Revision$") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "asterisk/astobj2.h"
 | 
					
						
							|  |  |  | #include "asterisk/stasis.h"
 | 
					
						
							|  |  |  | #include "asterisk/utils.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-11 15:46:35 +00:00
										 |  |  | static struct stasis_message_vtable null_vtable = {}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-08-06 12:55:28 +00:00
										 |  |  | int stasis_message_type_create(const char *name, | 
					
						
							|  |  |  | 	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; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-18 19:55:24 +00:00
										 |  |  | 	type = ao2_t_alloc(sizeof(*type), message_type_dtor, 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
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2013-06-11 15:46:35 +00:00
										 |  |  | 	type->vtable = vtable; | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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->type); | 
					
						
							|  |  |  | 	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; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-18 19:55:24 +00:00
										 |  |  | 	message = ao2_t_alloc(sizeof(*message), stasis_message_dtor, type->name); | 
					
						
							| 
									
										
										
										
											2013-03-08 15:15:13 +00:00
										 |  |  | 	if (message == NULL) { | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	message->timestamp = ast_tvnow(); | 
					
						
							|  |  |  | 	ao2_ref(type, +1); | 
					
						
							|  |  |  | 	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); | 
					
						
							|  |  |  | } |