mirror of
				https://github.com/asterisk/asterisk.git
				synced 2025-10-25 06:00:36 +00:00 
			
		
		
		
	This patch started out simply as fixing the bouncing tests introduced in r382685, but required some other changes to give it a decent implementation. To fix the bouncing tests, the UserEvent and Newexten AMI events needed to be refactored to dispatch via Stasis. Dispatching directly to AMI resulted in those events sometimes getting ahead of the associated Newchannel events, which would understandably confuse anyone. I found that instead of creating a zillion different message types and structures associated with them, it would be preferable to define a message type that has a channel snapshot and a blob of structured data with a small bit of additional information. The JSON object model provides a very nice way of representing structured data, so I went with that. * Move JSON support from res_json.c to main/json.c * Made libjansson-dev a required dependency * Added an ast_channel_blob message type, which has a channel snapshot and JSON blob of data. * Changed UserEvent and Newexten events so that they are dispatched via ast_channel_blob messages on the channel's topic. * Got rid of the ast_channel_varset message; used ast_channel_blob instead. * Extracted the manager functions converting Stasis channel events to AMI events into manager_channel.c. (issue ASTERISK-21096) Review: https://reviewboard.asterisk.org/r/2381/ git-svn-id: https://origsvn.digium.com/svn/asterisk/trunk@383579 65c4cc65-6c06-0410-ace0-fbb531ad65f3
		
			
				
	
	
		
			124 lines
		
	
	
		
			3.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			124 lines
		
	
	
		
			3.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * Asterisk -- An open source telephony toolkit.
 | |
|  *
 | |
|  * Copyright (C) 1999 - 2005, Digium, Inc.
 | |
|  *
 | |
|  * 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 UserEvent application -- send manager event
 | |
|  * 
 | |
|  * \ingroup applications
 | |
|  */
 | |
| 
 | |
| /*** MODULEINFO
 | |
| 	<support_level>core</support_level>
 | |
|  ***/
 | |
| 
 | |
| #include "asterisk.h"
 | |
| 
 | |
| ASTERISK_FILE_VERSION(__FILE__, "$Revision$")
 | |
| 
 | |
| #include "asterisk/pbx.h"
 | |
| #include "asterisk/module.h"
 | |
| #include "asterisk/manager.h"
 | |
| #include "asterisk/app.h"
 | |
| #include "asterisk/json.h"
 | |
| 
 | |
| /*** DOCUMENTATION
 | |
| 	<application name="UserEvent" language="en_US">
 | |
| 		<synopsis>
 | |
| 			Send an arbitrary event to the manager interface.
 | |
| 		</synopsis>
 | |
| 		<syntax>
 | |
| 			<parameter name="eventname" required="true" />
 | |
| 			<parameter name="body" />
 | |
| 		</syntax>
 | |
| 		<description>
 | |
| 			<para>Sends an arbitrary event to the manager interface, with an optional
 | |
| 			<replaceable>body</replaceable> representing additional arguments. The
 | |
| 			<replaceable>body</replaceable> may be specified as
 | |
| 			a <literal>,</literal> delimited list of headers. Each additional
 | |
| 			argument will be placed on a new line in the event. The format of the
 | |
| 			event will be:</para>
 | |
| 			<para>    Event: UserEvent</para>
 | |
| 			<para>    UserEvent: <specified event name></para>
 | |
| 			<para>    [body]</para>
 | |
| 			<para>If no <replaceable>body</replaceable> is specified, only Event and UserEvent headers will be present.</para>
 | |
| 		</description>
 | |
| 	</application>
 | |
|  ***/
 | |
| 
 | |
| static char *app = "UserEvent";
 | |
| 
 | |
| static int userevent_exec(struct ast_channel *chan, const char *data)
 | |
| {
 | |
| 	char *parse;
 | |
| 	int x;
 | |
| 	AST_DECLARE_APP_ARGS(args,
 | |
| 		AST_APP_ARG(eventname);
 | |
| 		AST_APP_ARG(extra)[100];
 | |
| 	);
 | |
| 	RAII_VAR(struct ast_str *, body, ast_str_create(16), ast_free);
 | |
| 	RAII_VAR(struct ast_json *, blob, NULL, ast_json_unref);
 | |
| 	RAII_VAR(struct stasis_message *, msg, NULL, ao2_cleanup);
 | |
| 
 | |
| 	if (ast_strlen_zero(data)) {
 | |
| 		ast_log(LOG_WARNING, "UserEvent requires an argument (eventname,optional event body)\n");
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	if (!body) {
 | |
| 		ast_log(LOG_WARNING, "Unable to allocate buffer\n");
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	parse = ast_strdupa(data);
 | |
| 
 | |
| 	AST_STANDARD_APP_ARGS(args, parse);
 | |
| 
 | |
| 	for (x = 0; x < args.argc - 1; x++) {
 | |
| 		ast_str_append(&body, 0, "%s\r\n", args.extra[x]);
 | |
| 	}
 | |
| 
 | |
| 	blob = ast_json_pack("{s: s, s: s, s: s}",
 | |
| 			     "type", "userevent",
 | |
| 			     "eventname", args.eventname,
 | |
| 			     "body", ast_str_buffer(body));
 | |
| 	if (!blob) {
 | |
| 		ast_log(LOG_WARNING, "Unable to create message buffer\n");
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	msg = ast_channel_blob_create(chan, blob);
 | |
| 	if (!msg) {
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	stasis_publish(ast_channel_topic(chan), msg);
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int unload_module(void)
 | |
| {
 | |
| 	return ast_unregister_application(app);
 | |
| }
 | |
| 
 | |
| static int load_module(void)
 | |
| {
 | |
| 	return ast_register_application_xml(app, userevent_exec);
 | |
| }
 | |
| 
 | |
| AST_MODULE_INFO_STANDARD(ASTERISK_GPL_KEY, "Custom User Event Application");
 |