| 
									
										
										
										
											2008-11-12 06:46:04 +00:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Asterisk -- An open source telephony toolkit. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Copyright (C) 2008, Digium, Inc. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Joshua Colp <jcolp@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 CLI Aliases | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \author\verbatim Joshua Colp <jcolp@digium.com> \endverbatim | 
					
						
							|  |  |  |  *  | 
					
						
							|  |  |  |  * This module provides the capability to create aliases to other | 
					
						
							|  |  |  |  * CLI commands. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "asterisk.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ASTERISK_FILE_VERSION(__FILE__, "$Revision$") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "asterisk/module.h"
 | 
					
						
							|  |  |  | #include "asterisk/config.h"
 | 
					
						
							|  |  |  | #include "asterisk/cli.h"
 | 
					
						
							|  |  |  | #include "asterisk/astobj2.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*! Maximum number of buckets for CLI aliases */ | 
					
						
							|  |  |  | #define MAX_ALIAS_BUCKETS 53
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*! Configuration file used for this application */ | 
					
						
							|  |  |  | static const char config_file[] = "cli_aliases.conf"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct cli_alias { | 
					
						
							|  |  |  | 	struct ast_cli_entry cli_entry; /*!< Actual CLI structure used for this alias */ | 
					
						
							|  |  |  | 	char *alias;                    /*!< CLI Alias */ | 
					
						
							|  |  |  | 	char *real_cmd;                 /*!< Actual CLI command it is aliased to */ | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct ao2_container *cli_aliases; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*! \brief Hashing function used for aliases */ | 
					
						
							|  |  |  | static int alias_hash_cb(const void *obj, const int flags) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	const struct cli_alias *alias = obj; | 
					
						
							|  |  |  | 	return ast_str_hash(alias->cli_entry.command); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*! \brief Comparison function used for aliases */ | 
					
						
							| 
									
										
										
										
											2008-11-25 01:01:49 +00:00
										 |  |  | static int alias_cmp_cb(void *obj, void *arg, int flags) | 
					
						
							| 
									
										
										
										
											2008-11-12 06:46:04 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	const struct cli_alias *alias0 = obj, *alias1 = arg; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return (alias0->cli_entry.command == alias1->cli_entry.command ? CMP_MATCH | CMP_STOP : 0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*! \brief Destruction function used for aliases */ | 
					
						
							|  |  |  | static void alias_destroy(void *obj) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct cli_alias *alias = obj; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Unregister the CLI entry from the core */ | 
					
						
							|  |  |  | 	ast_cli_unregister(&alias->cli_entry); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*! \brief Function which passes through an aliased CLI command to the real one */ | 
					
						
							|  |  |  | static char *cli_alias_passthrough(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct cli_alias *alias; | 
					
						
							|  |  |  | 	struct cli_alias tmp = { | 
					
						
							|  |  |  | 		.cli_entry.command = e->command, | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | 	char *generator; | 
					
						
							|  |  |  | 	const char *line; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Try to find the alias based on the CLI entry */ | 
					
						
							| 
									
										
										
										
											2008-11-25 01:01:49 +00:00
										 |  |  | 	if (!(alias = ao2_find(cli_aliases, &tmp, OBJ_POINTER))) { | 
					
						
							| 
									
										
										
										
											2008-11-12 06:46:04 +00:00
										 |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	switch (cmd) { | 
					
						
							|  |  |  | 	case CLI_INIT: | 
					
						
							|  |  |  | 		ao2_ref(alias, -1); | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	case CLI_GENERATE: | 
					
						
							|  |  |  | 		line = a->line; | 
					
						
							|  |  |  | 		line += (strlen(alias->alias)); | 
					
						
							| 
									
										
										
										
											2009-05-06 16:09:27 +00:00
										 |  |  | 		if (!strncasecmp(alias->alias, alias->real_cmd, strlen(alias->alias))) { | 
					
						
							| 
									
										
										
										
											2009-05-06 14:35:47 +00:00
										 |  |  | 			generator = NULL; | 
					
						
							|  |  |  | 		} else if (!ast_strlen_zero(a->word)) { | 
					
						
							| 
									
										
										
										
											2008-11-12 06:46:04 +00:00
										 |  |  | 			struct ast_str *real_cmd = ast_str_alloca(strlen(alias->real_cmd) + strlen(line) + 1); | 
					
						
							|  |  |  | 			ast_str_append(&real_cmd, 0, "%s%s", alias->real_cmd, line); | 
					
						
							| 
									
										
										
										
											2008-12-13 08:36:35 +00:00
										 |  |  | 			generator = ast_cli_generator(ast_str_buffer(real_cmd), a->word, a->n); | 
					
						
							| 
									
										
										
										
											2008-11-12 06:46:04 +00:00
										 |  |  | 		} else { | 
					
						
							|  |  |  | 			generator = ast_cli_generator(alias->real_cmd, a->word, a->n); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		ao2_ref(alias, -1); | 
					
						
							|  |  |  | 		return generator; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* If they gave us extra arguments we need to construct a string to pass in */ | 
					
						
							|  |  |  | 	if (a->argc != e->args) { | 
					
						
							|  |  |  | 		struct ast_str *real_cmd = ast_str_alloca(2048); | 
					
						
							|  |  |  | 		int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		ast_str_append(&real_cmd, 0, "%s", alias->real_cmd); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* Add the additional arguments that have been passed in */ | 
					
						
							|  |  |  | 		for (i = e->args + 1; i <= a->argc; i++) { | 
					
						
							|  |  |  | 			ast_str_append(&real_cmd, 0, " %s", a->argv[i - 1]); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-13 08:36:35 +00:00
										 |  |  | 		ast_cli_command(a->fd, ast_str_buffer(real_cmd)); | 
					
						
							| 
									
										
										
										
											2008-11-12 06:46:04 +00:00
										 |  |  | 	} else { | 
					
						
							|  |  |  | 		ast_cli_command(a->fd, alias->real_cmd); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ao2_ref(alias, -1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return CLI_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*! \brief CLI Command to display CLI Aliases */ | 
					
						
							|  |  |  | static char *alias_show(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | #define FORMAT "%-50.50s %-50.50s\n"
 | 
					
						
							|  |  |  | 	struct cli_alias *alias; | 
					
						
							|  |  |  | 	struct ao2_iterator i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	switch (cmd) { | 
					
						
							|  |  |  | 	case CLI_INIT: | 
					
						
							|  |  |  | 		e->command = "cli show aliases"; | 
					
						
							|  |  |  | 		e->usage = | 
					
						
							|  |  |  | 			"Usage: cli show aliases\n" | 
					
						
							|  |  |  | 			"       Displays a list of aliased CLI commands.\n"; | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	case CLI_GENERATE: | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ast_cli(a->fd, FORMAT, "Alias Command", "Real Command"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	i = ao2_iterator_init(cli_aliases, 0); | 
					
						
							|  |  |  | 	for (; (alias = ao2_iterator_next(&i)); ao2_ref(alias, -1)) { | 
					
						
							|  |  |  | 		ast_cli(a->fd, FORMAT, alias->alias, alias->real_cmd); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-10-06 01:24:24 +00:00
										 |  |  | 	ao2_iterator_destroy(&i); | 
					
						
							| 
									
										
										
										
											2008-11-12 06:46:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return CLI_SUCCESS; | 
					
						
							|  |  |  | #undef FORMAT
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*! \brief CLI commands to interact with things */ | 
					
						
							|  |  |  | static struct ast_cli_entry cli_alias[] = { | 
					
						
							|  |  |  | 	AST_CLI_DEFINE(alias_show, "Show CLI command aliases"), | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*! \brief Function called to load or reload the configuration file */ | 
					
						
							|  |  |  | static void load_config(int reload) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct ast_config *cfg = NULL; | 
					
						
							|  |  |  | 	struct ast_flags config_flags = { reload ? CONFIG_FLAG_FILEUNCHANGED : 0 }; | 
					
						
							|  |  |  | 	struct cli_alias *alias; | 
					
						
							|  |  |  | 	struct ast_variable *v, *v1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-11-19 19:25:14 +00:00
										 |  |  | 	if (!(cfg = ast_config_load(config_file, config_flags)) || cfg == CONFIG_STATUS_FILEINVALID) { | 
					
						
							| 
									
										
										
										
											2008-11-12 06:46:04 +00:00
										 |  |  | 		ast_log(LOG_ERROR, "res_clialiases configuration file '%s' not found\n", config_file); | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} else if (cfg == CONFIG_STATUS_FILEUNCHANGED) { | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-10 22:48:58 +00:00
										 |  |  | 	/* Destroy any existing CLI aliases */ | 
					
						
							| 
									
										
										
										
											2008-11-12 06:46:04 +00:00
										 |  |  | 	if (reload) { | 
					
						
							| 
									
										
										
										
											2010-01-11 16:41:44 +00:00
										 |  |  | 		ao2_callback(cli_aliases, OBJ_UNLINK | OBJ_NODATA | OBJ_MULTIPLE, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2008-11-12 06:46:04 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (v = ast_variable_browse(cfg, "general"); v; v = v->next) { | 
					
						
							|  |  |  | 		if (strcmp(v->name, "template")) { | 
					
						
							|  |  |  | 			ast_log(LOG_WARNING, "%s is not a correct option in [%s]\n", v->name, "general"); | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		/* Read in those there CLI aliases */ | 
					
						
							|  |  |  | 		for (v1 = ast_variable_browse(cfg, v->value); v1; v1 = v1->next) { | 
					
						
							|  |  |  | 			if (!(alias = ao2_alloc((sizeof(*alias) + strlen(v1->name) + strlen(v1->value) + 2), alias_destroy))) { | 
					
						
							|  |  |  | 				continue; | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2008-11-12 19:38:51 +00:00
										 |  |  | 			alias->alias = ((char *) alias) + sizeof(*alias); | 
					
						
							| 
									
										
										
										
											2008-11-12 20:27:40 +00:00
										 |  |  | 			alias->real_cmd = ((char *) alias->alias) + strlen(v1->name) + 1; | 
					
						
							| 
									
										
										
										
											2008-11-12 06:46:04 +00:00
										 |  |  | 			strcpy(alias->alias, v1->name); | 
					
						
							|  |  |  | 			strcpy(alias->real_cmd, v1->value); | 
					
						
							|  |  |  | 			alias->cli_entry.handler = cli_alias_passthrough; | 
					
						
							|  |  |  | 			alias->cli_entry.command = alias->alias; | 
					
						
							| 
									
										
										
										
											2010-03-16 19:36:24 +00:00
										 |  |  | 			alias->cli_entry.usage = "Aliased CLI Command\n"; | 
					
						
							| 
									
										
										
										
											2008-11-12 06:46:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-11-12 20:27:40 +00:00
										 |  |  | 			ast_cli_register(&alias->cli_entry); | 
					
						
							| 
									
										
										
										
											2008-11-12 06:46:04 +00:00
										 |  |  | 			ao2_link(cli_aliases, alias); | 
					
						
							|  |  |  | 			ast_verbose(VERBOSE_PREFIX_2 "Aliased CLI command '%s' to '%s'\n", v1->name, v1->value); | 
					
						
							|  |  |  | 			ao2_ref(alias, -1); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ast_config_destroy(cfg); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*! \brief Function called to reload the module */ | 
					
						
							|  |  |  | static int reload_module(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	load_config(1); | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*! \brief Function called to unload the module */ | 
					
						
							|  |  |  | static int unload_module(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	ao2_ref(cli_aliases, -1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ast_cli_unregister_multiple(cli_alias, ARRAY_LEN(cli_alias)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*! \brief Function called to load the module */ | 
					
						
							|  |  |  | static int load_module(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (!(cli_aliases = ao2_container_alloc(MAX_ALIAS_BUCKETS, alias_hash_cb, alias_cmp_cb))) { | 
					
						
							|  |  |  | 		return AST_MODULE_LOAD_DECLINE; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	load_config(0); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-11-12 20:27:40 +00:00
										 |  |  | 	ast_cli_register_multiple(cli_alias, ARRAY_LEN(cli_alias)); | 
					
						
							| 
									
										
										
										
											2008-11-12 06:46:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return AST_MODULE_LOAD_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | AST_MODULE_INFO(ASTERISK_GPL_KEY, AST_MODFLAG_DEFAULT, "CLI Aliases", | 
					
						
							|  |  |  | 		.load = load_module, | 
					
						
							|  |  |  | 		.unload = unload_module, | 
					
						
							|  |  |  | 		.reload = reload_module, | 
					
						
							|  |  |  | 		); |