mirror of
				https://github.com/asterisk/asterisk.git
				synced 2025-11-03 20:38:59 +00:00 
			
		
		
		
	* app_minivm: Use built-in completion facilities to complete optional arguments. * app_voicemail: Use built-in completion facilities to complete optional arguments. * app_confbridge: Add missing colons after 'Usage' text. * chan_alsa: Use built-in completion facilities to complete optional arguments. * chan_sip: Use built-in completion facilities to complete optional arguments. Add completions for 'load' for 'sip show user', 'sip show peer', and 'sip qualify peer.' * chan_skinny: Correct and extend completions for 'skinny reset' and 'skinny show line.' * func_odbc: Correct completions for 'odbc read' and 'odbc write' * main/astmm: Use built-in completion facilities to complete arguments for 'memory' commands. * main/bridge: Correct completions for 'bridge kick.' * main/ccss: Use built-in completion facilities to complete arguments for 'cc cancel' command. * main/cli: Add 'all' completion for 'channel request hangup.' Correct completions for 'core set debug channel.' Correct completions for 'core show calls.' * main/pbx_app: Remove redundant completions for 'core show applications.' * main/pbx_hangup_handler: Remove unused completions for 'core show hanguphandlers all.' * res_sorcery_memory_cache: Add completion for 'reload' argument of 'sorcery memory cache stale' and properly implement. Change-Id: Iee58c7392f6fec34ad9d596109117af87697bbca
		
			
				
	
	
		
			299 lines
		
	
	
		
			7.0 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			299 lines
		
	
	
		
			7.0 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/*
 | 
						|
 * Asterisk -- An open source telephony toolkit.
 | 
						|
 *
 | 
						|
 * Copyright (C) 2016, CFWare, LLC
 | 
						|
 *
 | 
						|
 * Corey Farrell <git@cfware.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 PBX Hangup Handler management routines.
 | 
						|
 *
 | 
						|
 * \author Corey Farrell <git@cfware.com>
 | 
						|
 */
 | 
						|
 | 
						|
/*** MODULEINFO
 | 
						|
	<support_level>core</support_level>
 | 
						|
 ***/
 | 
						|
 | 
						|
#include "asterisk.h"
 | 
						|
 | 
						|
#include "asterisk/_private.h"
 | 
						|
#include "asterisk/app.h"
 | 
						|
#include "asterisk/cli.h"
 | 
						|
#include "asterisk/linkedlists.h"
 | 
						|
#include "asterisk/pbx.h"
 | 
						|
#include "asterisk/stasis_channels.h"
 | 
						|
#include "asterisk/utils.h"
 | 
						|
 | 
						|
/*!
 | 
						|
 * \internal
 | 
						|
 * \brief Publish a hangup handler related message to \ref stasis
 | 
						|
 */
 | 
						|
static void publish_hangup_handler_message(const char *action, struct ast_channel *chan, const char *handler)
 | 
						|
{
 | 
						|
	RAII_VAR(struct ast_json *, blob, NULL, ast_json_unref);
 | 
						|
 | 
						|
	blob = ast_json_pack("{s: s, s: s}",
 | 
						|
			"type", action,
 | 
						|
			"handler", S_OR(handler, ""));
 | 
						|
	if (!blob) {
 | 
						|
		return;
 | 
						|
	}
 | 
						|
 | 
						|
	ast_channel_publish_blob(chan, ast_channel_hangup_handler_type(), blob);
 | 
						|
}
 | 
						|
 | 
						|
int ast_pbx_hangup_handler_run(struct ast_channel *chan)
 | 
						|
{
 | 
						|
	struct ast_hangup_handler_list *handlers;
 | 
						|
	struct ast_hangup_handler *h_handler;
 | 
						|
 | 
						|
	ast_channel_lock(chan);
 | 
						|
	handlers = ast_channel_hangup_handlers(chan);
 | 
						|
	if (AST_LIST_EMPTY(handlers)) {
 | 
						|
		ast_channel_unlock(chan);
 | 
						|
		return 0;
 | 
						|
	}
 | 
						|
 | 
						|
	/*
 | 
						|
	 * Make sure that the channel is marked as hungup since we are
 | 
						|
	 * going to run the hangup handlers on it.
 | 
						|
	 */
 | 
						|
	ast_softhangup_nolock(chan, AST_SOFTHANGUP_HANGUP_EXEC);
 | 
						|
 | 
						|
	for (;;) {
 | 
						|
		handlers = ast_channel_hangup_handlers(chan);
 | 
						|
		h_handler = AST_LIST_REMOVE_HEAD(handlers, node);
 | 
						|
		if (!h_handler) {
 | 
						|
			break;
 | 
						|
		}
 | 
						|
 | 
						|
		publish_hangup_handler_message("run", chan, h_handler->args);
 | 
						|
		ast_channel_unlock(chan);
 | 
						|
 | 
						|
		ast_app_exec_sub(NULL, chan, h_handler->args, 1);
 | 
						|
		ast_free(h_handler);
 | 
						|
 | 
						|
		ast_channel_lock(chan);
 | 
						|
	}
 | 
						|
	ast_channel_unlock(chan);
 | 
						|
	return 1;
 | 
						|
}
 | 
						|
 | 
						|
void ast_pbx_hangup_handler_init(struct ast_channel *chan)
 | 
						|
{
 | 
						|
	struct ast_hangup_handler_list *handlers;
 | 
						|
 | 
						|
	handlers = ast_channel_hangup_handlers(chan);
 | 
						|
	AST_LIST_HEAD_INIT_NOLOCK(handlers);
 | 
						|
}
 | 
						|
 | 
						|
void ast_pbx_hangup_handler_destroy(struct ast_channel *chan)
 | 
						|
{
 | 
						|
	struct ast_hangup_handler_list *handlers;
 | 
						|
	struct ast_hangup_handler *h_handler;
 | 
						|
 | 
						|
	ast_channel_lock(chan);
 | 
						|
 | 
						|
	/* Get rid of each of the hangup handlers on the channel */
 | 
						|
	handlers = ast_channel_hangup_handlers(chan);
 | 
						|
	while ((h_handler = AST_LIST_REMOVE_HEAD(handlers, node))) {
 | 
						|
		ast_free(h_handler);
 | 
						|
	}
 | 
						|
 | 
						|
	ast_channel_unlock(chan);
 | 
						|
}
 | 
						|
 | 
						|
int ast_pbx_hangup_handler_pop(struct ast_channel *chan)
 | 
						|
{
 | 
						|
	struct ast_hangup_handler_list *handlers;
 | 
						|
	struct ast_hangup_handler *h_handler;
 | 
						|
 | 
						|
	ast_channel_lock(chan);
 | 
						|
	handlers = ast_channel_hangup_handlers(chan);
 | 
						|
	h_handler = AST_LIST_REMOVE_HEAD(handlers, node);
 | 
						|
	if (h_handler) {
 | 
						|
		publish_hangup_handler_message("pop", chan, h_handler->args);
 | 
						|
	}
 | 
						|
	ast_channel_unlock(chan);
 | 
						|
	if (h_handler) {
 | 
						|
		ast_free(h_handler);
 | 
						|
		return 1;
 | 
						|
	}
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
void ast_pbx_hangup_handler_push(struct ast_channel *chan, const char *handler)
 | 
						|
{
 | 
						|
	struct ast_hangup_handler_list *handlers;
 | 
						|
	struct ast_hangup_handler *h_handler;
 | 
						|
	const char *expanded_handler;
 | 
						|
 | 
						|
	if (ast_strlen_zero(handler)) {
 | 
						|
		return;
 | 
						|
	}
 | 
						|
 | 
						|
	expanded_handler = ast_app_expand_sub_args(chan, handler);
 | 
						|
	if (!expanded_handler) {
 | 
						|
		return;
 | 
						|
	}
 | 
						|
	h_handler = ast_malloc(sizeof(*h_handler) + 1 + strlen(expanded_handler));
 | 
						|
	if (!h_handler) {
 | 
						|
		ast_free((char *) expanded_handler);
 | 
						|
		return;
 | 
						|
	}
 | 
						|
	strcpy(h_handler->args, expanded_handler);/* Safe */
 | 
						|
	ast_free((char *) expanded_handler);
 | 
						|
 | 
						|
	ast_channel_lock(chan);
 | 
						|
 | 
						|
	handlers = ast_channel_hangup_handlers(chan);
 | 
						|
	AST_LIST_INSERT_HEAD(handlers, h_handler, node);
 | 
						|
	publish_hangup_handler_message("push", chan, h_handler->args);
 | 
						|
	ast_channel_unlock(chan);
 | 
						|
}
 | 
						|
 | 
						|
#define HANDLER_FORMAT	"%-30s %s\n"
 | 
						|
 | 
						|
/*!
 | 
						|
 * \internal
 | 
						|
 * \brief CLI output the hangup handler headers.
 | 
						|
 * \since 11.0
 | 
						|
 *
 | 
						|
 * \param fd CLI file descriptor to use.
 | 
						|
 *
 | 
						|
 * \return Nothing
 | 
						|
 */
 | 
						|
static void ast_pbx_hangup_handler_headers(int fd)
 | 
						|
{
 | 
						|
	ast_cli(fd, HANDLER_FORMAT, "Channel", "Handler");
 | 
						|
}
 | 
						|
 | 
						|
/*!
 | 
						|
 * \internal
 | 
						|
 * \brief CLI output the channel hangup handlers.
 | 
						|
 * \since 11.0
 | 
						|
 *
 | 
						|
 * \param fd CLI file descriptor to use.
 | 
						|
 * \param chan Channel to show hangup handlers.
 | 
						|
 *
 | 
						|
 * \return Nothing
 | 
						|
 */
 | 
						|
static void ast_pbx_hangup_handler_show(int fd, struct ast_channel *chan)
 | 
						|
{
 | 
						|
	struct ast_hangup_handler_list *handlers;
 | 
						|
	struct ast_hangup_handler *h_handler;
 | 
						|
	int first = 1;
 | 
						|
 | 
						|
	ast_channel_lock(chan);
 | 
						|
	handlers = ast_channel_hangup_handlers(chan);
 | 
						|
	AST_LIST_TRAVERSE(handlers, h_handler, node) {
 | 
						|
		ast_cli(fd, HANDLER_FORMAT, first ? ast_channel_name(chan) : "", h_handler->args);
 | 
						|
		first = 0;
 | 
						|
	}
 | 
						|
	ast_channel_unlock(chan);
 | 
						|
}
 | 
						|
 | 
						|
/*
 | 
						|
 * \brief 'show hanguphandlers <channel>' CLI command implementation function...
 | 
						|
 */
 | 
						|
static char *handle_show_hangup_channel(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a)
 | 
						|
{
 | 
						|
	struct ast_channel *chan;
 | 
						|
 | 
						|
	switch (cmd) {
 | 
						|
	case CLI_INIT:
 | 
						|
		e->command = "core show hanguphandlers";
 | 
						|
		e->usage =
 | 
						|
			"Usage: core show hanguphandlers <channel>\n"
 | 
						|
			"       Show hangup handlers of a specified channel.\n";
 | 
						|
		return NULL;
 | 
						|
	case CLI_GENERATE:
 | 
						|
		return ast_complete_channels(a->line, a->word, a->pos, a->n, e->args);
 | 
						|
	}
 | 
						|
 | 
						|
	if (a->argc < 4) {
 | 
						|
		return CLI_SHOWUSAGE;
 | 
						|
	}
 | 
						|
 | 
						|
	chan = ast_channel_get_by_name(a->argv[3]);
 | 
						|
	if (!chan) {
 | 
						|
		ast_cli(a->fd, "Channel does not exist.\n");
 | 
						|
		return CLI_FAILURE;
 | 
						|
	}
 | 
						|
 | 
						|
	ast_pbx_hangup_handler_headers(a->fd);
 | 
						|
	ast_pbx_hangup_handler_show(a->fd, chan);
 | 
						|
 | 
						|
	ast_channel_unref(chan);
 | 
						|
 | 
						|
	return CLI_SUCCESS;
 | 
						|
}
 | 
						|
 | 
						|
/*
 | 
						|
 * \brief 'show hanguphandlers all' CLI command implementation function...
 | 
						|
 */
 | 
						|
static char *handle_show_hangup_all(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a)
 | 
						|
{
 | 
						|
	struct ast_channel_iterator *iter;
 | 
						|
	struct ast_channel *chan;
 | 
						|
 | 
						|
	switch (cmd) {
 | 
						|
	case CLI_INIT:
 | 
						|
		e->command = "core show hanguphandlers all";
 | 
						|
		e->usage =
 | 
						|
			"Usage: core show hanguphandlers all\n"
 | 
						|
			"       Show hangup handlers for all channels.\n";
 | 
						|
		return NULL;
 | 
						|
	case CLI_GENERATE:
 | 
						|
		return NULL;
 | 
						|
	}
 | 
						|
 | 
						|
	if (a->argc < 4) {
 | 
						|
		return CLI_SHOWUSAGE;
 | 
						|
	}
 | 
						|
 | 
						|
	iter = ast_channel_iterator_all_new();
 | 
						|
	if (!iter) {
 | 
						|
		return CLI_FAILURE;
 | 
						|
	}
 | 
						|
 | 
						|
	ast_pbx_hangup_handler_headers(a->fd);
 | 
						|
	for (; (chan = ast_channel_iterator_next(iter)); ast_channel_unref(chan)) {
 | 
						|
		ast_pbx_hangup_handler_show(a->fd, chan);
 | 
						|
	}
 | 
						|
	ast_channel_iterator_destroy(iter);
 | 
						|
 | 
						|
	return CLI_SUCCESS;
 | 
						|
}
 | 
						|
 | 
						|
static struct ast_cli_entry cli[] = {
 | 
						|
	AST_CLI_DEFINE(handle_show_hangup_all, "Show hangup handlers of all channels"),
 | 
						|
	AST_CLI_DEFINE(handle_show_hangup_channel, "Show hangup handlers of a specified channel"),
 | 
						|
};
 | 
						|
 | 
						|
static void unload_pbx_hangup_handler(void)
 | 
						|
{
 | 
						|
	ast_cli_unregister_multiple(cli, ARRAY_LEN(cli));
 | 
						|
}
 | 
						|
 | 
						|
int load_pbx_hangup_handler(void)
 | 
						|
{
 | 
						|
	ast_cli_register_multiple(cli, ARRAY_LEN(cli));
 | 
						|
	ast_register_cleanup(unload_pbx_hangup_handler);
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 |