| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Asterisk -- A telephony toolkit for Linux. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Standard Command Line Interface | 
					
						
							|  |  |  |  *  | 
					
						
							| 
									
										
										
										
											2004-09-03 03:44:35 +00:00
										 |  |  |  * Copyright (C) 1999-2004, Digium, Inc. | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2004-09-03 03:44:35 +00:00
										 |  |  |  * Mark Spencer <markster@digium.com> | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * This program is free software, distributed under the terms of | 
					
						
							|  |  |  |  * the GNU General Public License | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <unistd.h>
 | 
					
						
							|  |  |  | #include <stdlib.h>
 | 
					
						
							|  |  |  | #include <asterisk/logger.h>
 | 
					
						
							|  |  |  | #include <asterisk/options.h>
 | 
					
						
							|  |  |  | #include <asterisk/cli.h>
 | 
					
						
							|  |  |  | #include <asterisk/module.h>
 | 
					
						
							|  |  |  | #include <asterisk/channel.h>
 | 
					
						
							| 
									
										
										
										
											2002-07-29 15:38:22 +00:00
										 |  |  | #include <asterisk/channel_pvt.h>
 | 
					
						
							| 
									
										
										
										
											2004-07-08 19:41:34 +00:00
										 |  |  | #include <asterisk/manager.h>
 | 
					
						
							| 
									
										
										
										
											2004-05-06 20:49:24 +00:00
										 |  |  | #include <asterisk/utils.h>
 | 
					
						
							| 
									
										
										
										
											2004-06-22 18:49:00 +00:00
										 |  |  | #include <asterisk/lock.h>
 | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | #include <sys/signal.h>
 | 
					
						
							|  |  |  | #include <stdio.h>
 | 
					
						
							|  |  |  | #include <signal.h>
 | 
					
						
							|  |  |  | #include <string.h>
 | 
					
						
							|  |  |  | /* For rl_filename_completion */ | 
					
						
							| 
									
										
										
										
											2003-01-30 15:03:20 +00:00
										 |  |  | #include "editline/readline/readline.h"
 | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | /* For module directory */ | 
					
						
							|  |  |  | #include "asterisk.h"
 | 
					
						
							| 
									
										
										
										
											2001-05-09 03:11:22 +00:00
										 |  |  | #include "build.h"
 | 
					
						
							| 
									
										
										
										
											2003-01-30 15:03:20 +00:00
										 |  |  | #include "astconf.h"
 | 
					
						
							| 
									
										
										
										
											2001-05-09 03:11:22 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #define VERSION_INFO "Asterisk " ASTERISK_VERSION " built by " BUILD_USER "@" BUILD_HOSTNAME \
 | 
					
						
							|  |  |  | 	" on a " BUILD_MACHINE " running " BUILD_OS | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | void ast_cli(int fd, char *fmt, ...) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2003-10-22 03:52:56 +00:00
										 |  |  | 	char *stuff; | 
					
						
							| 
									
										
										
										
											2004-07-14 07:44:19 +00:00
										 |  |  | 	int res = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 	va_list ap; | 
					
						
							|  |  |  | 	va_start(ap, fmt); | 
					
						
							| 
									
										
										
										
											2004-07-14 07:44:19 +00:00
										 |  |  | 	res = vasprintf(&stuff, fmt, ap); | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 	va_end(ap); | 
					
						
							| 
									
										
										
										
											2004-07-14 07:44:19 +00:00
										 |  |  | 	if (res == -1) { | 
					
						
							|  |  |  | 		ast_log(LOG_ERROR, "Out of memory\n"); | 
					
						
							| 
									
										
										
										
											2004-07-14 14:53:24 +00:00
										 |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2004-07-14 07:44:19 +00:00
										 |  |  | 		ast_carefulwrite(fd, stuff, strlen(stuff), 100); | 
					
						
							|  |  |  | 		free(stuff); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-06-09 01:45:08 +00:00
										 |  |  | AST_MUTEX_DEFINE_STATIC(clilock); | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | struct ast_cli_entry *helpers = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static char load_help[] =  | 
					
						
							|  |  |  | "Usage: load <module name>\n" | 
					
						
							|  |  |  | "       Loads the specified module into Asterisk.\n"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static char unload_help[] =  | 
					
						
							|  |  |  | "Usage: unload [-f|-h] <module name>\n" | 
					
						
							|  |  |  | "       Unloads the specified module from Asterisk.  The -f\n" | 
					
						
							|  |  |  | "       option causes the module to be unloaded even if it is\n" | 
					
						
							|  |  |  | "       in use (may cause a crash) and the -h module causes the\n" | 
					
						
							|  |  |  | "       module to be unloaded even if the module says it cannot, \n" | 
					
						
							|  |  |  | "       which almost always will cause a crash.\n"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static char help_help[] = | 
					
						
							|  |  |  | "Usage: help [topic]\n" | 
					
						
							|  |  |  | "       When called with a topic as an argument, displays usage\n" | 
					
						
							|  |  |  | "       information on the given command.  If called without a\n" | 
					
						
							|  |  |  | "       topic, it provides a list of commands.\n"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static char chanlist_help[] =  | 
					
						
							| 
									
										
										
										
											2004-06-11 00:18:30 +00:00
										 |  |  | "Usage: show channels [concise]\n" | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | "       Lists currently defined channels and some information about\n" | 
					
						
							| 
									
										
										
										
											2004-06-11 00:18:30 +00:00
										 |  |  | "       them.  If 'concise' is specified, format is abridged and in\n" | 
					
						
							|  |  |  | "       a more easily machine parsable format\n"; | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-05-09 03:11:22 +00:00
										 |  |  | static char reload_help[] =  | 
					
						
							| 
									
										
										
										
											2004-09-03 14:02:12 +00:00
										 |  |  | "Usage: reload [module ...]\n" | 
					
						
							|  |  |  | "       Reloads configuration files for all listed modules which support\n" | 
					
						
							|  |  |  | "       reloading, or for all supported modules if none are listed.\n"; | 
					
						
							| 
									
										
										
										
											2001-05-09 03:11:22 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static char set_verbose_help[] =  | 
					
						
							|  |  |  | "Usage: set verbose <level>\n" | 
					
						
							|  |  |  | "       Sets level of verbose messages to be displayed.  0 means\n" | 
					
						
							| 
									
										
										
										
											2004-09-13 18:19:15 +00:00
										 |  |  | "       no messages should be displayed. Equivalent to -v[v[v...]]\n" | 
					
						
							|  |  |  | "       on startup\n"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static char set_debug_help[] =  | 
					
						
							|  |  |  | "Usage: set debug <level>\n" | 
					
						
							|  |  |  | "       Sets level of core debug messages to be displayed.  0 means\n" | 
					
						
							|  |  |  | "       no messages should be displayed. Equivalent to -d[d[d...]]\n" | 
					
						
							|  |  |  | "       on startup.\n"; | 
					
						
							| 
									
										
										
										
											2001-05-09 03:11:22 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static char softhangup_help[] = | 
					
						
							|  |  |  | "Usage: soft hangup <channel>\n" | 
					
						
							|  |  |  | "       Request that a channel be hung up.  The hangup takes effect\n" | 
					
						
							|  |  |  | "       the next time the driver reads or writes from the channel\n"; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | static int handle_load(int fd, int argc, char *argv[]) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (argc != 2) | 
					
						
							|  |  |  | 		return RESULT_SHOWUSAGE; | 
					
						
							|  |  |  | 	if (ast_load_resource(argv[1])) { | 
					
						
							|  |  |  | 		ast_cli(fd, "Unable to load module %s\n", argv[1]); | 
					
						
							|  |  |  | 		return RESULT_FAILURE; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return RESULT_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-05-09 03:11:22 +00:00
										 |  |  | static int handle_reload(int fd, int argc, char *argv[]) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2004-09-02 20:45:24 +00:00
										 |  |  | 	int x; | 
					
						
							| 
									
										
										
										
											2004-10-16 21:14:05 +00:00
										 |  |  | 	int res; | 
					
						
							| 
									
										
										
										
											2004-09-02 20:45:24 +00:00
										 |  |  | 	if (argc < 1) | 
					
						
							| 
									
										
										
										
											2001-05-09 03:11:22 +00:00
										 |  |  | 		return RESULT_SHOWUSAGE; | 
					
						
							| 
									
										
										
										
											2004-09-02 20:45:24 +00:00
										 |  |  | 	if (argc > 1) {  | 
					
						
							| 
									
										
										
										
											2004-10-16 21:14:05 +00:00
										 |  |  | 		for (x=1;x<argc;x++) { | 
					
						
							|  |  |  | 			res = ast_module_reload(argv[x]); | 
					
						
							|  |  |  | 			switch(res) { | 
					
						
							|  |  |  | 			case 0: | 
					
						
							|  |  |  | 				ast_cli(fd, "No such module '%s'\n", argv[x]); | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			case 1: | 
					
						
							|  |  |  | 				ast_cli(fd, "Module '%s' does not support reload\n", argv[x]); | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2004-09-02 20:45:24 +00:00
										 |  |  | 	} else | 
					
						
							|  |  |  | 		ast_module_reload(NULL); | 
					
						
							| 
									
										
										
										
											2001-05-09 03:11:22 +00:00
										 |  |  | 	return RESULT_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int handle_set_verbose(int fd, int argc, char *argv[]) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2004-09-13 18:19:15 +00:00
										 |  |  | 	int val = 0; | 
					
						
							|  |  |  | 	int oldval = 0; | 
					
						
							| 
									
										
										
										
											2001-05-09 03:11:22 +00:00
										 |  |  | 	/* Has a hidden 'at least' argument */ | 
					
						
							|  |  |  | 	if ((argc != 3) && (argc != 4)) | 
					
						
							|  |  |  | 		return RESULT_SHOWUSAGE; | 
					
						
							|  |  |  | 	if ((argc == 4) && strcasecmp(argv[2], "atleast")) | 
					
						
							|  |  |  | 		return RESULT_SHOWUSAGE; | 
					
						
							| 
									
										
										
										
											2004-09-13 18:19:15 +00:00
										 |  |  | 	oldval = option_verbose; | 
					
						
							| 
									
										
										
										
											2001-05-09 03:11:22 +00:00
										 |  |  | 	if (argc == 3) | 
					
						
							|  |  |  | 		option_verbose = atoi(argv[2]); | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		val = atoi(argv[3]); | 
					
						
							|  |  |  | 		if (val > option_verbose) | 
					
						
							|  |  |  | 			option_verbose = val; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2004-09-13 18:19:15 +00:00
										 |  |  | 	if (oldval != option_verbose && option_verbose > 0) | 
					
						
							|  |  |  | 		ast_cli(fd, "Verbosity was %d and is now %d\n", oldval, option_verbose); | 
					
						
							|  |  |  | 	else if (oldval > 0 && option_verbose > 0) | 
					
						
							|  |  |  | 		ast_cli(fd, "Verbosity is atleast %d\n", option_verbose); | 
					
						
							| 
									
										
										
										
											2004-10-08 04:23:22 +00:00
										 |  |  | 	else if (oldval > 0 && option_verbose == 0) | 
					
						
							| 
									
										
										
										
											2004-09-13 18:19:15 +00:00
										 |  |  | 		ast_cli(fd, "Verbosity is now OFF\n"); | 
					
						
							|  |  |  | 	return RESULT_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int handle_set_debug(int fd, int argc, char *argv[]) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int val = 0; | 
					
						
							|  |  |  | 	int oldval = 0; | 
					
						
							|  |  |  | 	/* Has a hidden 'at least' argument */ | 
					
						
							|  |  |  | 	if ((argc != 3) && (argc != 4)) | 
					
						
							|  |  |  | 		return RESULT_SHOWUSAGE; | 
					
						
							|  |  |  | 	if ((argc == 4) && strcasecmp(argv[2], "atleast")) | 
					
						
							|  |  |  | 		return RESULT_SHOWUSAGE; | 
					
						
							|  |  |  | 	oldval = option_debug; | 
					
						
							|  |  |  | 	if (argc == 3) | 
					
						
							|  |  |  | 		option_debug = atoi(argv[2]); | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		val = atoi(argv[3]); | 
					
						
							|  |  |  | 		if (val > option_debug) | 
					
						
							|  |  |  | 			option_debug = val; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (oldval != option_debug && option_debug > 0) | 
					
						
							|  |  |  | 		ast_cli(fd, "Core debug was %d and is now %d\n", oldval, option_debug); | 
					
						
							|  |  |  | 	else if (oldval > 0 && option_debug > 0) | 
					
						
							|  |  |  | 		ast_cli(fd, "Core debug is atleast %d\n", option_debug); | 
					
						
							|  |  |  | 	else if (oldval > 0 && option_debug == 0) | 
					
						
							|  |  |  | 		ast_cli(fd, "Core debug is now OFF\n"); | 
					
						
							| 
									
										
										
										
											2001-05-09 03:11:22 +00:00
										 |  |  | 	return RESULT_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | static int handle_unload(int fd, int argc, char *argv[]) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int x; | 
					
						
							|  |  |  | 	int force=AST_FORCE_SOFT; | 
					
						
							|  |  |  | 	if (argc < 2) | 
					
						
							|  |  |  | 		return RESULT_SHOWUSAGE; | 
					
						
							|  |  |  | 	for (x=1;x<argc;x++) { | 
					
						
							|  |  |  | 		if (argv[x][0] == '-') { | 
					
						
							|  |  |  | 			switch(argv[x][1]) { | 
					
						
							|  |  |  | 			case 'f': | 
					
						
							|  |  |  | 				force = AST_FORCE_FIRM; | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			case 'h': | 
					
						
							|  |  |  | 				force = AST_FORCE_HARD; | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			default: | 
					
						
							|  |  |  | 				return RESULT_SHOWUSAGE; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} else if (x !=  argc - 1)  | 
					
						
							|  |  |  | 			return RESULT_SHOWUSAGE; | 
					
						
							|  |  |  | 		else if (ast_unload_resource(argv[x], force)) { | 
					
						
							|  |  |  | 			ast_cli(fd, "Unable to unload resource %s\n", argv[x]); | 
					
						
							|  |  |  | 			return RESULT_FAILURE; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return RESULT_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-01-16 01:24:00 +00:00
										 |  |  | #define MODLIST_FORMAT  "%-25s %-40.40s %-10d\n"
 | 
					
						
							| 
									
										
										
										
											2004-01-15 22:56:53 +00:00
										 |  |  | #define MODLIST_FORMAT2 "%-25s %-40.40s %-10s\n"
 | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-06-09 01:45:08 +00:00
										 |  |  | AST_MUTEX_DEFINE_STATIC(climodentrylock); | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | static int climodentryfd = -1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-10-14 04:38:29 +00:00
										 |  |  | static int modlist_modentry(char *module, char *description, int usecnt, char *like) | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2004-10-14 04:38:29 +00:00
										 |  |  | 	/* Comparing the like with the module */ | 
					
						
							|  |  |  | 	if ( strstr(module,like) != NULL) { | 
					
						
							|  |  |  | 		ast_cli(climodentryfd, MODLIST_FORMAT, module, description, usecnt); | 
					
						
							|  |  |  | 		return 1; | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 	}  | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static char modlist_help[] = | 
					
						
							| 
									
										
										
										
											2004-10-14 04:38:29 +00:00
										 |  |  | "Usage: show modules [like keyword]\n" | 
					
						
							|  |  |  | "       Shows Asterisk modules currently in use, and usage statistics.\n"; | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-05-09 03:11:22 +00:00
										 |  |  | static char version_help[] = | 
					
						
							|  |  |  | "Usage: show version\n" | 
					
						
							|  |  |  | "       Shows Asterisk version information.\n "; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-05-02 15:37:34 +00:00
										 |  |  | static char *format_uptimestr(time_t timeval) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int years = 0, weeks = 0, days = 0, hours = 0, mins = 0, secs = 0; | 
					
						
							| 
									
										
										
										
											2004-07-14 07:44:19 +00:00
										 |  |  | 	char timestr[256]=""; | 
					
						
							|  |  |  | 	int bytes = 0; | 
					
						
							|  |  |  | 	int maxbytes = 0; | 
					
						
							|  |  |  | 	int offset = 0; | 
					
						
							| 
									
										
										
										
											2003-05-02 15:37:34 +00:00
										 |  |  | #define SECOND (1)
 | 
					
						
							| 
									
										
										
										
											2004-06-22 17:42:14 +00:00
										 |  |  | #define MINUTE (SECOND*60)
 | 
					
						
							|  |  |  | #define HOUR (MINUTE*60)
 | 
					
						
							| 
									
										
										
										
											2003-05-02 15:37:34 +00:00
										 |  |  | #define DAY (HOUR*24)
 | 
					
						
							|  |  |  | #define WEEK (DAY*7)
 | 
					
						
							|  |  |  | #define YEAR (DAY*365)
 | 
					
						
							| 
									
										
										
										
											2004-07-14 07:44:19 +00:00
										 |  |  | #define ESS(x) ((x == 1) ? "" : "s")
 | 
					
						
							| 
									
										
										
										
											2003-05-02 15:37:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-14 07:44:19 +00:00
										 |  |  | 	maxbytes = sizeof(timestr); | 
					
						
							| 
									
										
										
										
											2004-03-14 09:11:32 +00:00
										 |  |  | 	if (timeval < 0) | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							| 
									
										
										
										
											2003-05-02 15:37:34 +00:00
										 |  |  | 	if (timeval > YEAR) { | 
					
						
							|  |  |  | 		years = (timeval / YEAR); | 
					
						
							|  |  |  | 		timeval -= (years * YEAR); | 
					
						
							| 
									
										
										
										
											2004-07-14 07:44:19 +00:00
										 |  |  | 		if (years > 0) { | 
					
						
							|  |  |  | 			snprintf(timestr + offset, maxbytes, "%d year%s, ", years, ESS(years)); | 
					
						
							|  |  |  | 			bytes = strlen(timestr + offset); | 
					
						
							|  |  |  | 			offset += bytes; | 
					
						
							|  |  |  | 			maxbytes -= bytes; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2003-05-02 15:37:34 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	if (timeval > WEEK) { | 
					
						
							|  |  |  | 		weeks = (timeval / WEEK); | 
					
						
							|  |  |  | 		timeval -= (weeks * WEEK); | 
					
						
							| 
									
										
										
										
											2004-07-14 07:44:19 +00:00
										 |  |  | 		if (weeks > 0) { | 
					
						
							|  |  |  | 			snprintf(timestr + offset, maxbytes, "%d week%s, ", weeks, ESS(weeks)); | 
					
						
							|  |  |  | 			bytes = strlen(timestr + offset); | 
					
						
							|  |  |  | 			offset += bytes; | 
					
						
							|  |  |  | 			maxbytes -= bytes; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2003-05-02 15:37:34 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	if (timeval > DAY) { | 
					
						
							|  |  |  | 		days = (timeval / DAY); | 
					
						
							|  |  |  | 		timeval -= (days * DAY); | 
					
						
							| 
									
										
										
										
											2004-07-14 07:44:19 +00:00
										 |  |  | 		if (days > 0) { | 
					
						
							|  |  |  | 			snprintf(timestr + offset, maxbytes, "%d day%s, ", days, ESS(days)); | 
					
						
							|  |  |  | 			bytes = strlen(timestr + offset); | 
					
						
							|  |  |  | 			offset += bytes; | 
					
						
							|  |  |  | 			maxbytes -= bytes; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2003-05-02 15:37:34 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	if (timeval > HOUR) { | 
					
						
							|  |  |  | 		hours = (timeval / HOUR); | 
					
						
							|  |  |  | 		timeval -= (hours * HOUR); | 
					
						
							| 
									
										
										
										
											2004-07-14 07:44:19 +00:00
										 |  |  | 		if (hours > 0) { | 
					
						
							|  |  |  | 			snprintf(timestr + offset, maxbytes, "%d hour%s, ", hours, ESS(hours)); | 
					
						
							|  |  |  | 			bytes = strlen(timestr + offset); | 
					
						
							|  |  |  | 			offset += bytes; | 
					
						
							|  |  |  | 			maxbytes -= bytes; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2003-05-02 15:37:34 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2004-06-22 17:42:14 +00:00
										 |  |  | 	if (timeval > MINUTE) { | 
					
						
							|  |  |  | 		mins = (timeval / MINUTE); | 
					
						
							|  |  |  | 		timeval -= (mins * MINUTE); | 
					
						
							| 
									
										
										
										
											2004-07-14 07:44:19 +00:00
										 |  |  | 		if (mins > 0) { | 
					
						
							|  |  |  | 			snprintf(timestr + offset, maxbytes, "%d minute%s, ", mins, ESS(mins)); | 
					
						
							|  |  |  | 			bytes = strlen(timestr + offset); | 
					
						
							|  |  |  | 			offset += bytes; | 
					
						
							|  |  |  | 			maxbytes -= bytes; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2003-05-02 15:37:34 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	secs = timeval; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-14 07:44:19 +00:00
										 |  |  | 	if (secs > 0) { | 
					
						
							|  |  |  | 		snprintf(timestr + offset, maxbytes, "%d second%s", secs, ESS(secs)); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2003-05-02 15:37:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return timestr ? strdup(timestr) : NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int handle_showuptime(int fd, int argc, char *argv[]) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	time_t curtime, tmptime; | 
					
						
							|  |  |  | 	char *timestr; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	time(&curtime); | 
					
						
							|  |  |  | 	if (ast_startuptime) { | 
					
						
							|  |  |  | 		tmptime = curtime - ast_startuptime; | 
					
						
							|  |  |  | 		timestr = format_uptimestr(tmptime); | 
					
						
							|  |  |  | 		if (timestr) { | 
					
						
							|  |  |  | 			ast_cli(fd, "System uptime: %s\n", timestr); | 
					
						
							|  |  |  | 			free(timestr); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	}		 | 
					
						
							|  |  |  | 	if (ast_lastreloadtime) { | 
					
						
							|  |  |  | 		tmptime = curtime - ast_lastreloadtime; | 
					
						
							|  |  |  | 		timestr = format_uptimestr(tmptime); | 
					
						
							|  |  |  | 		if (timestr) { | 
					
						
							|  |  |  | 			ast_cli(fd, "Last reload: %s\n", timestr); | 
					
						
							|  |  |  | 			free(timestr); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return RESULT_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | static int handle_modlist(int fd, int argc, char *argv[]) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2004-10-14 04:38:29 +00:00
										 |  |  | 	char *like = ""; | 
					
						
							|  |  |  | 	if (argc == 3) | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 		return RESULT_SHOWUSAGE; | 
					
						
							| 
									
										
										
										
											2004-10-14 04:38:29 +00:00
										 |  |  | 	else if (argc >= 4) { | 
					
						
							|  |  |  | 		if ( strcmp(argv[2],"like") )  | 
					
						
							|  |  |  | 			return RESULT_SHOWUSAGE; | 
					
						
							|  |  |  | 		like = argv[3]; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 		 | 
					
						
							| 
									
										
										
										
											2003-08-13 15:25:16 +00:00
										 |  |  | 	ast_mutex_lock(&climodentrylock); | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 	climodentryfd = fd; | 
					
						
							|  |  |  | 	ast_cli(fd, MODLIST_FORMAT2, "Module", "Description", "Use Count"); | 
					
						
							| 
									
										
										
										
											2004-10-14 04:38:29 +00:00
										 |  |  | 	ast_cli(fd,"%d modules loaded\n",ast_update_module_list(modlist_modentry,like)); | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 	climodentryfd = -1; | 
					
						
							| 
									
										
										
										
											2003-08-13 15:25:16 +00:00
										 |  |  | 	ast_mutex_unlock(&climodentrylock); | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 	return RESULT_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-05-09 03:11:22 +00:00
										 |  |  | static int handle_version(int fd, int argc, char *argv[]) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (argc != 2) | 
					
						
							|  |  |  | 		return RESULT_SHOWUSAGE; | 
					
						
							| 
									
										
										
										
											2001-10-11 18:51:39 +00:00
										 |  |  | 	ast_cli(fd, "%s\n", VERSION_INFO); | 
					
						
							| 
									
										
										
										
											2001-05-09 03:11:22 +00:00
										 |  |  | 	return RESULT_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | static int handle_chanlist(int fd, int argc, char *argv[]) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2001-05-09 03:11:22 +00:00
										 |  |  | #define FORMAT_STRING  "%15s  (%-10s %-12s %-4d) %7s %-12s  %-15s\n"
 | 
					
						
							|  |  |  | #define FORMAT_STRING2 "%15s  (%-10s %-12s %-4s) %7s %-12s  %-15s\n"
 | 
					
						
							| 
									
										
										
										
											2004-06-11 00:18:30 +00:00
										 |  |  | #define CONCISE_FORMAT_STRING  "%s:%s:%s:%d:%s:%s:%s:%s:%s:%d\n"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 	struct ast_channel *c=NULL; | 
					
						
							| 
									
										
										
										
											2003-02-12 13:59:15 +00:00
										 |  |  | 	int numchans = 0; | 
					
						
							| 
									
										
										
										
											2004-06-11 00:18:30 +00:00
										 |  |  | 	int concise = 0; | 
					
						
							|  |  |  | 	if (argc < 2 || argc > 3) | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 		return RESULT_SHOWUSAGE; | 
					
						
							| 
									
										
										
										
											2004-06-11 00:18:30 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	concise = (argc == 3 && (!strcasecmp(argv[2],"concise"))); | 
					
						
							| 
									
										
										
										
											2004-05-20 16:30:10 +00:00
										 |  |  | 	c = ast_channel_walk_locked(NULL); | 
					
						
							| 
									
										
										
										
											2004-06-11 00:18:30 +00:00
										 |  |  | 	if(!concise) | 
					
						
							|  |  |  | 		ast_cli(fd, FORMAT_STRING2, "Channel", "Context", "Extension", "Pri", "State", "Appl.", "Data"); | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 	while(c) { | 
					
						
							| 
									
										
										
										
											2004-06-11 00:18:30 +00:00
										 |  |  | 		if(concise) | 
					
						
							|  |  |  | 			ast_cli(fd, CONCISE_FORMAT_STRING, c->name, c->context, c->exten, c->priority, ast_state2str(c->_state), | 
					
						
							|  |  |  | 					c->appl ? c->appl : "(None)", c->data ? ( !ast_strlen_zero(c->data) ? c->data : "" ): "", | 
					
						
							| 
									
										
										
										
											2004-10-02 00:58:31 +00:00
										 |  |  | 					(c->cid.cid_num && !ast_strlen_zero(c->cid.cid_num)) ? c->cid.cid_num : "", | 
					
						
							| 
									
										
										
										
											2004-06-11 00:18:30 +00:00
										 |  |  | 					(c->accountcode && !ast_strlen_zero(c->accountcode)) ? c->accountcode : "",c->amaflags); | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			ast_cli(fd, FORMAT_STRING, c->name, c->context, c->exten, c->priority, ast_state2str(c->_state), | 
					
						
							|  |  |  | 					c->appl ? c->appl : "(None)", c->data ? ( !ast_strlen_zero(c->data) ? c->data : "(Empty)" ): "(None)"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-02-12 13:59:15 +00:00
										 |  |  | 		numchans++; | 
					
						
							| 
									
										
										
										
											2004-05-20 16:30:10 +00:00
										 |  |  | 		ast_mutex_unlock(&c->lock); | 
					
						
							|  |  |  | 		c = ast_channel_walk_locked(c); | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2004-06-11 00:18:30 +00:00
										 |  |  | 	if(!concise) | 
					
						
							|  |  |  | 		ast_cli(fd, "%d active channel(s)\n", numchans); | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 	return RESULT_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static char showchan_help[] =  | 
					
						
							|  |  |  | "Usage: show channel <channel>\n" | 
					
						
							|  |  |  | "       Shows lots of information about the specified channel.\n"; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-03-12 06:00:18 +00:00
										 |  |  | static char debugchan_help[] =  | 
					
						
							|  |  |  | "Usage: debug channel <channel>\n" | 
					
						
							|  |  |  | "       Enables debugging on a specific channel.\n"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static char nodebugchan_help[] =  | 
					
						
							|  |  |  | "Usage: no debug channel <channel>\n" | 
					
						
							|  |  |  | "       Disables debugging on a specific channel.\n"; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-05-09 03:11:22 +00:00
										 |  |  | static char commandcomplete_help[] =  | 
					
						
							|  |  |  | "Usage: _command complete \"<line>\" text state\n" | 
					
						
							|  |  |  | "       This function is used internally to help with command completion and should.\n" | 
					
						
							|  |  |  | "       never be called by the user directly.\n"; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-01-30 15:03:20 +00:00
										 |  |  | static char commandnummatches_help[] =  | 
					
						
							|  |  |  | "Usage: _command nummatches \"<line>\" text \n" | 
					
						
							|  |  |  | "       This function is used internally to help with command completion and should.\n" | 
					
						
							|  |  |  | "       never be called by the user directly.\n"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static char commandmatchesarray_help[] =  | 
					
						
							|  |  |  | "Usage: _command matchesarray \"<line>\" text \n" | 
					
						
							|  |  |  | "       This function is used internally to help with command completion and should.\n" | 
					
						
							|  |  |  | "       never be called by the user directly.\n"; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-05-09 03:11:22 +00:00
										 |  |  | static int handle_softhangup(int fd, int argc, char *argv[]) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct ast_channel *c=NULL; | 
					
						
							|  |  |  | 	if (argc != 3) | 
					
						
							|  |  |  | 		return RESULT_SHOWUSAGE; | 
					
						
							| 
									
										
										
										
											2004-05-20 16:30:10 +00:00
										 |  |  | 	c = ast_channel_walk_locked(NULL); | 
					
						
							| 
									
										
										
										
											2001-05-09 03:11:22 +00:00
										 |  |  | 	while(c) { | 
					
						
							|  |  |  | 		if (!strcasecmp(c->name, argv[2])) { | 
					
						
							|  |  |  | 			ast_cli(fd, "Requested Hangup on channel '%s'\n", c->name); | 
					
						
							| 
									
										
										
										
											2002-07-29 15:38:22 +00:00
										 |  |  | 			ast_softhangup(c, AST_SOFTHANGUP_EXPLICIT); | 
					
						
							| 
									
										
										
										
											2004-05-20 16:30:10 +00:00
										 |  |  | 			ast_mutex_unlock(&c->lock); | 
					
						
							| 
									
										
										
										
											2001-05-09 03:11:22 +00:00
										 |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2004-05-20 16:30:10 +00:00
										 |  |  | 		ast_mutex_unlock(&c->lock); | 
					
						
							|  |  |  | 		c = ast_channel_walk_locked(c); | 
					
						
							| 
									
										
										
										
											2001-05-09 03:11:22 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	if (!c)  | 
					
						
							|  |  |  | 		ast_cli(fd, "%s is not a known channel\n", argv[2]); | 
					
						
							|  |  |  | 	return RESULT_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static char *__ast_cli_generator(char *text, char *word, int state, int lock); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-01-30 15:03:20 +00:00
										 |  |  | static int handle_commandmatchesarray(int fd, int argc, char *argv[]) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2004-10-14 04:12:05 +00:00
										 |  |  | 	char *buf, *obuf; | 
					
						
							| 
									
										
										
										
											2004-04-06 07:42:01 +00:00
										 |  |  | 	int buflen = 2048; | 
					
						
							| 
									
										
										
										
											2003-01-30 15:03:20 +00:00
										 |  |  | 	int len = 0; | 
					
						
							|  |  |  | 	char **matches; | 
					
						
							| 
									
										
										
										
											2004-10-14 04:12:05 +00:00
										 |  |  | 	int x, matchlen; | 
					
						
							| 
									
										
										
										
											2003-01-30 15:03:20 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (argc != 4) | 
					
						
							|  |  |  | 		return RESULT_SHOWUSAGE; | 
					
						
							| 
									
										
										
										
											2004-04-06 07:42:01 +00:00
										 |  |  | 	buf = malloc(buflen); | 
					
						
							|  |  |  | 	if (!buf) | 
					
						
							|  |  |  | 		return RESULT_FAILURE; | 
					
						
							| 
									
										
										
										
											2003-01-30 15:03:20 +00:00
										 |  |  | 	buf[len] = '\0'; | 
					
						
							|  |  |  | 	matches = ast_cli_completion_matches(argv[2], argv[3]); | 
					
						
							|  |  |  | 	if (matches) { | 
					
						
							|  |  |  | 		for (x=0; matches[x]; x++) { | 
					
						
							|  |  |  | #if 0
 | 
					
						
							|  |  |  | 			printf("command matchesarray for '%s' %s got '%s'\n", argv[2], argv[3], matches[x]); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2004-10-14 04:12:05 +00:00
										 |  |  | 			matchlen = strlen(matches[x]) + 1; | 
					
						
							|  |  |  | 			if (len + matchlen >= buflen) { | 
					
						
							|  |  |  | 				buflen += matchlen * 3; | 
					
						
							|  |  |  | 				obuf = buf; | 
					
						
							|  |  |  | 				buf = realloc(obuf, buflen); | 
					
						
							|  |  |  | 				if (!buf)  | 
					
						
							|  |  |  | 					/* Out of memory...  Just free old buffer and be done */ | 
					
						
							|  |  |  | 					free(obuf); | 
					
						
							| 
									
										
										
										
											2004-04-06 07:42:01 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2004-10-14 04:12:05 +00:00
										 |  |  | 			if (buf) | 
					
						
							|  |  |  | 				len += sprintf( buf + len, "%s ", matches[x]); | 
					
						
							| 
									
										
										
										
											2003-04-08 13:45:36 +00:00
										 |  |  | 			free(matches[x]); | 
					
						
							|  |  |  | 			matches[x] = NULL; | 
					
						
							| 
									
										
										
										
											2003-01-30 15:03:20 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2003-04-08 13:45:36 +00:00
										 |  |  | 		free(matches); | 
					
						
							| 
									
										
										
										
											2003-01-30 15:03:20 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | #if 0
 | 
					
						
							|  |  |  | 	printf("array for '%s' %s got '%s'\n", argv[2], argv[3], buf); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	if (buf) { | 
					
						
							| 
									
										
										
										
											2004-04-06 07:42:01 +00:00
										 |  |  | 		ast_cli(fd, "%s%s",buf, AST_CLI_COMPLETE_EOF); | 
					
						
							|  |  |  | 		free(buf); | 
					
						
							| 
									
										
										
										
											2003-01-30 15:03:20 +00:00
										 |  |  | 	} else | 
					
						
							|  |  |  | 		ast_cli(fd, "NULL\n"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return RESULT_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int handle_commandnummatches(int fd, int argc, char *argv[]) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int matches = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (argc != 4) | 
					
						
							|  |  |  | 		return RESULT_SHOWUSAGE; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	matches = ast_cli_generatornummatches(argv[2], argv[3]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if 0
 | 
					
						
							|  |  |  | 	printf("Search for '%s' %s got '%d'\n", argv[2], argv[3], matches); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 	ast_cli(fd, "%d", matches); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return RESULT_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-05-09 03:11:22 +00:00
										 |  |  | static int handle_commandcomplete(int fd, int argc, char *argv[]) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	char *buf; | 
					
						
							|  |  |  | #if 0
 | 
					
						
							|  |  |  | 	printf("Search for %d args: '%s', '%s', '%s', '%s'\n", argc, argv[0], argv[1], argv[2], argv[3]); | 
					
						
							|  |  |  | #endif	
 | 
					
						
							|  |  |  | 	if (argc != 5) | 
					
						
							|  |  |  | 		return RESULT_SHOWUSAGE; | 
					
						
							|  |  |  | 	buf = __ast_cli_generator(argv[2], argv[3], atoi(argv[4]), 0); | 
					
						
							|  |  |  | #if 0
 | 
					
						
							|  |  |  | 	printf("Search for '%s' %s %d got '%s'\n", argv[2], argv[3], atoi(argv[4]), buf); | 
					
						
							|  |  |  | #endif	
 | 
					
						
							|  |  |  | 	if (buf) { | 
					
						
							|  |  |  | 		ast_cli(fd, buf); | 
					
						
							|  |  |  | 		free(buf); | 
					
						
							|  |  |  | 	} else | 
					
						
							|  |  |  | 		ast_cli(fd, "NULL\n"); | 
					
						
							|  |  |  | 	return RESULT_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-03-12 06:00:18 +00:00
										 |  |  | static int handle_debugchan(int fd, int argc, char *argv[]) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct ast_channel *c=NULL; | 
					
						
							|  |  |  | 	if (argc != 3) | 
					
						
							|  |  |  | 		return RESULT_SHOWUSAGE; | 
					
						
							| 
									
										
										
										
											2004-05-20 16:30:10 +00:00
										 |  |  | 	c = ast_channel_walk_locked(NULL); | 
					
						
							| 
									
										
										
										
											2003-03-12 06:00:18 +00:00
										 |  |  | 	while(c) { | 
					
						
							|  |  |  | 		if (!strcasecmp(c->name, argv[2])) { | 
					
						
							|  |  |  | 			c->fin |= 0x80000000; | 
					
						
							|  |  |  | 			c->fout |= 0x80000000; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2004-05-20 16:30:10 +00:00
										 |  |  | 		ast_mutex_unlock(&c->lock); | 
					
						
							|  |  |  | 		c = ast_channel_walk_locked(c); | 
					
						
							| 
									
										
										
										
											2003-03-12 06:00:18 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2004-05-20 16:30:10 +00:00
										 |  |  | 	if (c) { | 
					
						
							| 
									
										
										
										
											2003-03-12 06:00:18 +00:00
										 |  |  | 		ast_cli(fd, "Debugging enabled on channel %s\n", c->name); | 
					
						
							| 
									
										
										
										
											2004-05-20 16:30:10 +00:00
										 |  |  | 		ast_mutex_unlock(&c->lock); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2003-03-12 06:00:18 +00:00
										 |  |  | 	else | 
					
						
							|  |  |  | 		ast_cli(fd, "No such channel %s\n", argv[2]); | 
					
						
							|  |  |  | 	return RESULT_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int handle_nodebugchan(int fd, int argc, char *argv[]) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct ast_channel *c=NULL; | 
					
						
							|  |  |  | 	if (argc != 4) | 
					
						
							|  |  |  | 		return RESULT_SHOWUSAGE; | 
					
						
							| 
									
										
										
										
											2004-05-20 16:30:10 +00:00
										 |  |  | 	c = ast_channel_walk_locked(NULL); | 
					
						
							| 
									
										
										
										
											2003-03-12 06:00:18 +00:00
										 |  |  | 	while(c) { | 
					
						
							|  |  |  | 		if (!strcasecmp(c->name, argv[3])) { | 
					
						
							|  |  |  | 			c->fin &= 0x7fffffff; | 
					
						
							|  |  |  | 			c->fout &= 0x7fffffff; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2004-05-20 16:30:10 +00:00
										 |  |  | 		ast_mutex_unlock(&c->lock); | 
					
						
							|  |  |  | 		c = ast_channel_walk_locked(c); | 
					
						
							| 
									
										
										
										
											2003-03-12 06:00:18 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2004-05-20 16:30:10 +00:00
										 |  |  | 	if (c) { | 
					
						
							| 
									
										
										
										
											2003-03-12 06:00:18 +00:00
										 |  |  | 		ast_cli(fd, "Debugging disabled on channel %s\n", c->name); | 
					
						
							| 
									
										
										
										
											2004-05-20 16:30:10 +00:00
										 |  |  | 		ast_mutex_unlock(&c->lock); | 
					
						
							|  |  |  | 	} else | 
					
						
							| 
									
										
										
										
											2003-03-12 06:00:18 +00:00
										 |  |  | 		ast_cli(fd, "No such channel %s\n", argv[2]); | 
					
						
							|  |  |  | 	return RESULT_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | static int handle_showchan(int fd, int argc, char *argv[]) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct ast_channel *c=NULL; | 
					
						
							| 
									
										
										
										
											2004-08-03 17:48:18 +00:00
										 |  |  | 	struct timeval now; | 
					
						
							|  |  |  | 	long elapsed_seconds=0; | 
					
						
							| 
									
										
										
										
											2004-09-03 03:44:35 +00:00
										 |  |  | 	int hour=0, min=0, sec=0; | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 	if (argc != 3) | 
					
						
							|  |  |  | 		return RESULT_SHOWUSAGE; | 
					
						
							| 
									
										
										
										
											2004-08-03 17:48:18 +00:00
										 |  |  | 	gettimeofday(&now, NULL); | 
					
						
							| 
									
										
										
										
											2004-05-20 16:30:10 +00:00
										 |  |  | 	c = ast_channel_walk_locked(NULL); | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 	while(c) { | 
					
						
							|  |  |  | 		if (!strcasecmp(c->name, argv[2])) { | 
					
						
							| 
									
										
										
										
											2004-08-03 17:48:18 +00:00
										 |  |  | 			if(c->cdr) { | 
					
						
							|  |  |  | 				elapsed_seconds = now.tv_sec - c->cdr->start.tv_sec; | 
					
						
							| 
									
										
										
										
											2004-09-03 03:44:35 +00:00
										 |  |  | 				hour = elapsed_seconds / 3600; | 
					
						
							|  |  |  | 				min = (elapsed_seconds % 3600) / 60; | 
					
						
							|  |  |  | 				sec = elapsed_seconds % 60; | 
					
						
							| 
									
										
										
										
											2004-08-03 17:48:18 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 			ast_cli(fd,  | 
					
						
							|  |  |  | 	" -- General --\n" | 
					
						
							|  |  |  | 	"           Name: %s\n" | 
					
						
							|  |  |  | 	"           Type: %s\n" | 
					
						
							| 
									
										
										
										
											2003-05-30 04:41:18 +00:00
										 |  |  | 	"       UniqueID: %s\n" | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 	"      Caller ID: %s\n" | 
					
						
							| 
									
										
										
										
											2004-10-02 00:58:31 +00:00
										 |  |  | 	" Caller ID Name: %s\n" | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 	"    DNID Digits: %s\n" | 
					
						
							| 
									
										
										
										
											2001-05-09 03:11:22 +00:00
										 |  |  | 	"          State: %s (%d)\n" | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 	"          Rings: %d\n" | 
					
						
							| 
									
										
										
										
											2003-01-30 15:03:20 +00:00
										 |  |  | 	"   NativeFormat: %d\n" | 
					
						
							| 
									
										
										
										
											2001-03-10 19:12:11 +00:00
										 |  |  | 	"    WriteFormat: %d\n" | 
					
						
							|  |  |  | 	"     ReadFormat: %d\n" | 
					
						
							| 
									
										
										
										
											2001-05-09 03:11:22 +00:00
										 |  |  | 	"1st File Descriptor: %d\n" | 
					
						
							| 
									
										
										
										
											2003-03-12 06:00:18 +00:00
										 |  |  | 	"      Frames in: %d%s\n" | 
					
						
							|  |  |  | 	"     Frames out: %d%s\n" | 
					
						
							| 
									
										
										
										
											2003-09-27 02:45:37 +00:00
										 |  |  | 	" Time to Hangup: %ld\n" | 
					
						
							| 
									
										
										
										
											2004-09-03 03:44:35 +00:00
										 |  |  | 	"   Elapsed Time: %dh%dm%ds\n" | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 	" --   PBX   --\n" | 
					
						
							|  |  |  | 	"        Context: %s\n" | 
					
						
							|  |  |  | 	"      Extension: %s\n" | 
					
						
							|  |  |  | 	"       Priority: %d\n" | 
					
						
							| 
									
										
										
										
											2003-04-09 04:00:43 +00:00
										 |  |  | 	"     Call Group: %d\n" | 
					
						
							|  |  |  | 	"   Pickup Group: %d\n" | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 	"    Application: %s\n" | 
					
						
							|  |  |  | 	"           Data: %s\n" | 
					
						
							|  |  |  | 	"          Stack: %d\n" | 
					
						
							|  |  |  | 	"    Blocking in: %s\n", | 
					
						
							| 
									
										
										
										
											2003-05-30 04:41:18 +00:00
										 |  |  | 	c->name, c->type, c->uniqueid, | 
					
						
							| 
									
										
										
										
											2004-10-02 00:58:31 +00:00
										 |  |  | 	(c->cid.cid_num ? c->cid.cid_num : "(N/A)"), | 
					
						
							|  |  |  | 	(c->cid.cid_name ? c->cid.cid_name : "(N/A)"), | 
					
						
							|  |  |  | 	(c->cid.cid_dnid ? c->cid.cid_dnid : "(N/A)" ), ast_state2str(c->_state), c->_state, c->rings, c->nativeformats, c->writeformat, c->readformat, | 
					
						
							| 
									
										
										
										
											2003-03-12 06:00:18 +00:00
										 |  |  | 	c->fds[0], c->fin & 0x7fffffff, (c->fin & 0x80000000) ? " (DEBUGGED)" : "", | 
					
						
							| 
									
										
										
										
											2003-09-27 02:45:37 +00:00
										 |  |  | 	c->fout & 0x7fffffff, (c->fout & 0x80000000) ? " (DEBUGGED)" : "", (long)c->whentohangup, | 
					
						
							| 
									
										
										
										
											2004-09-03 03:44:35 +00:00
										 |  |  | 	hour, min, sec,  | 
					
						
							| 
									
										
										
										
											2003-04-09 04:00:43 +00:00
										 |  |  | 	c->context, c->exten, c->priority, c->callgroup, c->pickupgroup, ( c->appl ? c->appl : "(N/A)" ), | 
					
						
							| 
									
										
										
										
											2004-05-06 20:49:24 +00:00
										 |  |  | 	( c-> data ? (!ast_strlen_zero(c->data) ? c->data : "(Empty)") : "(None)"), | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 	c->stack, (c->blocking ? c->blockproc : "(Not Blocking)")); | 
					
						
							| 
									
										
										
										
											2004-05-20 16:30:10 +00:00
										 |  |  | 		ast_mutex_unlock(&c->lock); | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 		break; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2004-05-20 16:30:10 +00:00
										 |  |  | 		ast_mutex_unlock(&c->lock); | 
					
						
							|  |  |  | 		c = ast_channel_walk_locked(c); | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	if (!c)  | 
					
						
							|  |  |  | 		ast_cli(fd, "%s is not a known channel\n", argv[2]); | 
					
						
							|  |  |  | 	return RESULT_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-08-07 19:27:54 +00:00
										 |  |  | static char *complete_ch_helper(char *line, char *word, int pos, int state, int rpos) | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct ast_channel *c; | 
					
						
							|  |  |  | 	int which=0; | 
					
						
							| 
									
										
										
										
											2004-05-20 16:30:10 +00:00
										 |  |  | 	char *ret; | 
					
						
							| 
									
										
										
										
											2004-08-07 19:27:54 +00:00
										 |  |  | 	if (pos != rpos) | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							| 
									
										
										
										
											2004-05-20 16:30:10 +00:00
										 |  |  | 	c = ast_channel_walk_locked(NULL); | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 	while(c) { | 
					
						
							| 
									
										
										
										
											2001-05-09 03:11:22 +00:00
										 |  |  | 		if (!strncasecmp(word, c->name, strlen(word))) { | 
					
						
							|  |  |  | 			if (++which > state) | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2004-05-20 16:30:10 +00:00
										 |  |  | 		ast_mutex_unlock(&c->lock); | 
					
						
							|  |  |  | 		c = ast_channel_walk_locked(c); | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2004-05-20 16:30:10 +00:00
										 |  |  | 	if (c) { | 
					
						
							|  |  |  | 		ret = strdup(c->name); | 
					
						
							|  |  |  | 		ast_mutex_unlock(&c->lock); | 
					
						
							|  |  |  | 	} else | 
					
						
							|  |  |  | 		ret = NULL; | 
					
						
							|  |  |  | 	return ret; | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-08-07 19:27:54 +00:00
										 |  |  | static char *complete_ch_3(char *line, char *word, int pos, int state) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return complete_ch_helper(line, word, pos, state, 2); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static char *complete_ch_4(char *line, char *word, int pos, int state) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return complete_ch_helper(line, word, pos, state, 3); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-10-16 21:14:05 +00:00
										 |  |  | static char *complete_mod_2(char *line, char *word, int pos, int state) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return ast_module_helper(line, word, pos, state, 1, 1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static char *complete_mod_4(char *line, char *word, int pos, int state) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return ast_module_helper(line, word, pos, state, 3, 0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | static char *complete_fn(char *line, char *word, int pos, int state) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	char *c; | 
					
						
							|  |  |  | 	char filename[256]; | 
					
						
							|  |  |  | 	if (pos != 1) | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	if (word[0] == '/') | 
					
						
							| 
									
										
										
										
											2001-10-18 16:47:57 +00:00
										 |  |  | 		strncpy(filename, word, sizeof(filename)-1); | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 	else | 
					
						
							| 
									
										
										
										
											2003-01-30 15:03:20 +00:00
										 |  |  | 		snprintf(filename, sizeof(filename), "%s/%s", (char *)ast_config_AST_MODULE_DIR, word); | 
					
						
							| 
									
										
										
										
											2001-05-09 03:11:22 +00:00
										 |  |  | 	c = (char*)filename_completion_function(filename, state); | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 	if (c && word[0] != '/') | 
					
						
							| 
									
										
										
										
											2003-01-30 15:03:20 +00:00
										 |  |  | 		c += (strlen((char*)ast_config_AST_MODULE_DIR) + 1); | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 	return c ? strdup(c) : c; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int handle_help(int fd, int argc, char *argv[]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct ast_cli_entry builtins[] = { | 
					
						
							| 
									
										
										
										
											2004-03-06 09:01:08 +00:00
										 |  |  | 	/* Keep alphabetized, with longer matches first (example: abcd before abc) */ | 
					
						
							| 
									
										
										
										
											2001-05-09 03:11:22 +00:00
										 |  |  | 	{ { "_command", "complete", NULL }, handle_commandcomplete, "Command complete", commandcomplete_help }, | 
					
						
							| 
									
										
										
										
											2003-01-30 15:03:20 +00:00
										 |  |  | 	{ { "_command", "nummatches", NULL }, handle_commandnummatches, "Returns number of command matches", commandnummatches_help }, | 
					
						
							|  |  |  | 	{ { "_command", "matchesarray", NULL }, handle_commandmatchesarray, "Returns command matches array", commandmatchesarray_help }, | 
					
						
							| 
									
										
										
										
											2004-08-07 19:27:54 +00:00
										 |  |  | 	{ { "debug", "channel", NULL }, handle_debugchan, "Enable debugging on a channel", debugchan_help, complete_ch_3 }, | 
					
						
							| 
									
										
										
										
											2003-03-12 06:00:18 +00:00
										 |  |  | 	{ { "help", NULL }, handle_help, "Display help list, or specific help on a command", help_help }, | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 	{ { "load", NULL }, handle_load, "Load a dynamic module by name", load_help, complete_fn }, | 
					
						
							| 
									
										
										
										
											2004-08-07 19:27:54 +00:00
										 |  |  | 	{ { "no", "debug", "channel", NULL }, handle_nodebugchan, "Disable debugging on a channel", nodebugchan_help, complete_ch_4 }, | 
					
						
							| 
									
										
										
										
											2004-10-16 21:14:05 +00:00
										 |  |  | 	{ { "reload", NULL }, handle_reload, "Reload configuration", reload_help, complete_mod_2 }, | 
					
						
							| 
									
										
										
										
											2004-09-13 18:19:15 +00:00
										 |  |  | 	{ { "set", "debug", NULL }, handle_set_debug, "Set level of debug chattiness", set_debug_help }, | 
					
						
							| 
									
										
										
										
											2001-05-09 03:11:22 +00:00
										 |  |  | 	{ { "set", "verbose", NULL }, handle_set_verbose, "Set level of verboseness", set_verbose_help }, | 
					
						
							| 
									
										
										
										
											1999-12-27 19:20:20 +00:00
										 |  |  | 	{ { "show", "channels", NULL }, handle_chanlist, "Display information on channels", chanlist_help }, | 
					
						
							| 
									
										
										
										
											2004-08-07 19:27:54 +00:00
										 |  |  | 	{ { "show", "channel", NULL }, handle_showchan, "Display information on a specific channel", showchan_help, complete_ch_3 }, | 
					
						
							| 
									
										
										
										
											2001-10-11 18:51:39 +00:00
										 |  |  | 	{ { "show", "modules", NULL }, handle_modlist, "List modules and info", modlist_help }, | 
					
						
							| 
									
										
										
										
											2004-10-16 21:14:05 +00:00
										 |  |  | 	{ { "show", "modules", "like", NULL }, handle_modlist, "List modules and info", modlist_help, complete_mod_4 }, | 
					
						
							| 
									
										
										
										
											2003-05-02 15:37:34 +00:00
										 |  |  | 	{ { "show", "uptime", NULL }, handle_showuptime, "Show uptime information", modlist_help }, | 
					
						
							| 
									
										
										
										
											2001-10-11 18:51:39 +00:00
										 |  |  | 	{ { "show", "version", NULL }, handle_version, "Display version info", version_help }, | 
					
						
							| 
									
										
										
										
											2004-08-07 19:27:54 +00:00
										 |  |  | 	{ { "soft", "hangup", NULL }, handle_softhangup, "Request a hangup on a given channel", softhangup_help, complete_ch_3 }, | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 	{ { "unload", NULL }, handle_unload, "Unload a dynamic module by name", unload_help, complete_fn }, | 
					
						
							|  |  |  | 	{ { NULL }, NULL, NULL, NULL } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct ast_cli_entry *find_cli(char *cmds[], int exact) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int x; | 
					
						
							|  |  |  | 	int y; | 
					
						
							|  |  |  | 	int match; | 
					
						
							|  |  |  | 	struct ast_cli_entry *e=NULL; | 
					
						
							|  |  |  | 	for (x=0;builtins[x].cmda[0];x++) { | 
					
						
							|  |  |  | 		/* start optimistic */ | 
					
						
							|  |  |  | 		match = 1; | 
					
						
							|  |  |  | 		for (y=0;match && cmds[y]; y++) { | 
					
						
							|  |  |  | 			/* If there are no more words in the candidate command, then we're
 | 
					
						
							|  |  |  | 			   there.  */ | 
					
						
							|  |  |  | 			if (!builtins[x].cmda[y] && !exact) | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			/* If there are no more words in the command (and we're looking for
 | 
					
						
							|  |  |  | 			   an exact match) or there is a difference between the two words, | 
					
						
							|  |  |  | 			   then this is not a match */ | 
					
						
							|  |  |  | 			if (!builtins[x].cmda[y] || strcasecmp(builtins[x].cmda[y], cmds[y])) | 
					
						
							|  |  |  | 				match = 0; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		/* If more words are needed to complete the command then this is not
 | 
					
						
							|  |  |  | 		   a candidate (unless we're looking for a really inexact answer  */ | 
					
						
							|  |  |  | 		if ((exact > -1) && builtins[x].cmda[y]) | 
					
						
							|  |  |  | 			match = 0; | 
					
						
							|  |  |  | 		if (match) | 
					
						
							|  |  |  | 			return &builtins[x]; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	for (e=helpers;e;e=e->next) { | 
					
						
							|  |  |  | 		match = 1; | 
					
						
							| 
									
										
										
										
											1999-12-27 19:20:20 +00:00
										 |  |  | 		for (y=0;match && cmds[y]; y++) { | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 			if (!e->cmda[y] && !exact) | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			if (!e->cmda[y] || strcasecmp(e->cmda[y], cmds[y])) | 
					
						
							|  |  |  | 				match = 0; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if ((exact > -1) && e->cmda[y]) | 
					
						
							|  |  |  | 			match = 0; | 
					
						
							|  |  |  | 		if (match) | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return e; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-14 07:44:19 +00:00
										 |  |  | static void join(char *dest, size_t destsize, char *w[]) | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	int x; | 
					
						
							|  |  |  | 	/* Join words into a string */ | 
					
						
							| 
									
										
										
										
											2004-07-14 07:44:19 +00:00
										 |  |  | 	if (!dest || destsize < 1) { | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	dest[0] = '\0'; | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 	for (x=0;w[x];x++) { | 
					
						
							|  |  |  | 		if (x) | 
					
						
							| 
									
										
										
										
											2004-07-14 07:44:19 +00:00
										 |  |  | 			strncat(dest, " ", destsize - strlen(dest) - 1); | 
					
						
							|  |  |  | 		strncat(dest, w[x], destsize - strlen(dest) - 1); | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-14 07:44:19 +00:00
										 |  |  | static void join2(char *dest, size_t destsize, char *w[]) | 
					
						
							| 
									
										
										
										
											1999-12-27 19:20:20 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	int x; | 
					
						
							|  |  |  | 	/* Join words into a string */ | 
					
						
							| 
									
										
										
										
											2004-07-14 07:44:19 +00:00
										 |  |  | 	if (!dest || destsize < 1) { | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	dest[0] = '\0'; | 
					
						
							| 
									
										
										
										
											1999-12-27 19:20:20 +00:00
										 |  |  | 	for (x=0;w[x];x++) { | 
					
						
							| 
									
										
										
										
											2004-07-14 07:44:19 +00:00
										 |  |  | 		strncat(dest, w[x], destsize - strlen(dest) - 1); | 
					
						
							| 
									
										
										
										
											1999-12-27 19:20:20 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | static char *find_best(char *argv[]) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	static char cmdline[80]; | 
					
						
							|  |  |  | 	int x; | 
					
						
							|  |  |  | 	/* See how close we get, then print the  */ | 
					
						
							|  |  |  | 	char *myargv[AST_MAX_CMD_LEN]; | 
					
						
							|  |  |  | 	for (x=0;x<AST_MAX_CMD_LEN;x++) | 
					
						
							|  |  |  | 		myargv[x]=NULL; | 
					
						
							|  |  |  | 	for (x=0;argv[x];x++) { | 
					
						
							|  |  |  | 		myargv[x] = argv[x]; | 
					
						
							|  |  |  | 		if (!find_cli(myargv, -1)) | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	join(cmdline, sizeof(cmdline), myargv); | 
					
						
							|  |  |  | 	return cmdline; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-12-27 19:20:20 +00:00
										 |  |  | int ast_cli_unregister(struct ast_cli_entry *e) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct ast_cli_entry *cur, *l=NULL; | 
					
						
							| 
									
										
										
										
											2003-08-13 15:25:16 +00:00
										 |  |  | 	ast_mutex_lock(&clilock); | 
					
						
							| 
									
										
										
										
											1999-12-27 19:20:20 +00:00
										 |  |  | 	cur = helpers; | 
					
						
							|  |  |  | 	while(cur) { | 
					
						
							|  |  |  | 		if (e == cur) { | 
					
						
							| 
									
										
										
										
											2002-05-13 22:29:39 +00:00
										 |  |  | 			if (e->inuse) { | 
					
						
							|  |  |  | 				ast_log(LOG_WARNING, "Can't remove command that is in use\n"); | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				/* Rewrite */ | 
					
						
							|  |  |  | 				if (l) | 
					
						
							|  |  |  | 					l->next = e->next; | 
					
						
							|  |  |  | 				else | 
					
						
							|  |  |  | 					helpers = e->next; | 
					
						
							|  |  |  | 				e->next = NULL; | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											1999-12-27 19:20:20 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		l = cur; | 
					
						
							|  |  |  | 		cur = cur->next; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2003-08-13 15:25:16 +00:00
										 |  |  | 	ast_mutex_unlock(&clilock); | 
					
						
							| 
									
										
										
										
											1999-12-27 19:20:20 +00:00
										 |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | int ast_cli_register(struct ast_cli_entry *e) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct ast_cli_entry *cur, *l=NULL; | 
					
						
							| 
									
										
										
										
											2002-07-29 15:38:22 +00:00
										 |  |  | 	char fulle[80] ="", fulltst[80] =""; | 
					
						
							| 
									
										
										
										
											1999-12-27 19:20:20 +00:00
										 |  |  | 	static int len; | 
					
						
							| 
									
										
										
										
											2003-08-13 15:25:16 +00:00
										 |  |  | 	ast_mutex_lock(&clilock); | 
					
						
							| 
									
										
										
										
											1999-12-27 19:20:20 +00:00
										 |  |  | 	join2(fulle, sizeof(fulle), e->cmda); | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 	if (find_cli(e->cmda, -1)) { | 
					
						
							| 
									
										
										
										
											2003-08-13 15:25:16 +00:00
										 |  |  | 		ast_mutex_unlock(&clilock); | 
					
						
							| 
									
										
										
										
											2001-10-11 18:51:39 +00:00
										 |  |  | 		ast_log(LOG_WARNING, "Command '%s' already registered (or something close enough)\n", fulle); | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	cur = helpers; | 
					
						
							|  |  |  | 	while(cur) { | 
					
						
							| 
									
										
										
										
											1999-12-27 19:20:20 +00:00
										 |  |  | 		join2(fulltst, sizeof(fulltst), cur->cmda); | 
					
						
							|  |  |  | 		len = strlen(fulltst); | 
					
						
							|  |  |  | 		if (strlen(fulle) < len) | 
					
						
							|  |  |  | 			len = strlen(fulle); | 
					
						
							|  |  |  | 		if (strncasecmp(fulle, fulltst, len) < 0) { | 
					
						
							|  |  |  | 			if (l) { | 
					
						
							|  |  |  | 				e->next = l->next; | 
					
						
							|  |  |  | 				l->next = e; | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				e->next = helpers; | 
					
						
							|  |  |  | 				helpers = e; | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		l = cur; | 
					
						
							|  |  |  | 		cur = cur->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (!cur) { | 
					
						
							|  |  |  | 		if (l) | 
					
						
							|  |  |  | 			l->next = e; | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			helpers = e; | 
					
						
							|  |  |  | 		e->next = NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2003-08-13 15:25:16 +00:00
										 |  |  | 	ast_mutex_unlock(&clilock); | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int help_workhorse(int fd, char *match[]) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	char fullcmd1[80]; | 
					
						
							|  |  |  | 	char fullcmd2[80]; | 
					
						
							|  |  |  | 	char matchstr[80]; | 
					
						
							|  |  |  | 	char *fullcmd; | 
					
						
							|  |  |  | 	struct ast_cli_entry *e, *e1, *e2; | 
					
						
							|  |  |  | 	e1 = builtins; | 
					
						
							|  |  |  | 	e2 = helpers; | 
					
						
							|  |  |  | 	if (match) | 
					
						
							|  |  |  | 		join(matchstr, sizeof(matchstr), match); | 
					
						
							|  |  |  | 	while(e1->cmda[0] || e2) { | 
					
						
							|  |  |  | 		if (e2) | 
					
						
							|  |  |  | 			join(fullcmd2, sizeof(fullcmd2), e2->cmda); | 
					
						
							|  |  |  | 		if (e1->cmda[0]) | 
					
						
							|  |  |  | 			join(fullcmd1, sizeof(fullcmd1), e1->cmda); | 
					
						
							| 
									
										
										
										
											1999-12-27 19:20:20 +00:00
										 |  |  | 		if (!e1->cmda[0] ||  | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 				(e2 && (strcmp(fullcmd2, fullcmd1) < 0))) { | 
					
						
							|  |  |  | 			/* Use e2 */ | 
					
						
							|  |  |  | 			e = e2; | 
					
						
							|  |  |  | 			fullcmd = fullcmd2; | 
					
						
							|  |  |  | 			/* Increment by going to next */ | 
					
						
							|  |  |  | 			e2 = e2->next; | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			/* Use e1 */ | 
					
						
							|  |  |  | 			e = e1; | 
					
						
							|  |  |  | 			fullcmd = fullcmd1; | 
					
						
							|  |  |  | 			e1++; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2001-05-09 03:11:22 +00:00
										 |  |  | 		/* Hide commands that start with '_' */ | 
					
						
							|  |  |  | 		if (fullcmd[0] == '_') | 
					
						
							|  |  |  | 			continue; | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 		if (match) { | 
					
						
							|  |  |  | 			if (strncasecmp(matchstr, fullcmd, strlen(matchstr))) { | 
					
						
							|  |  |  | 				continue; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2004-05-10 18:45:20 +00:00
										 |  |  | 		ast_cli(fd, "%25.25s  %s\n", fullcmd, e->summary); | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int handle_help(int fd, int argc, char *argv[]) { | 
					
						
							|  |  |  | 	struct ast_cli_entry *e; | 
					
						
							|  |  |  | 	char fullcmd[80]; | 
					
						
							|  |  |  | 	if ((argc < 1)) | 
					
						
							|  |  |  | 		return RESULT_SHOWUSAGE; | 
					
						
							|  |  |  | 	if (argc > 1) { | 
					
						
							|  |  |  | 		e = find_cli(argv + 1, 1); | 
					
						
							|  |  |  | 		if (e)  | 
					
						
							|  |  |  | 			ast_cli(fd, e->usage); | 
					
						
							|  |  |  | 		else { | 
					
						
							|  |  |  | 			if (find_cli(argv + 1, -1)) { | 
					
						
							|  |  |  | 				return help_workhorse(fd, argv + 1); | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				join(fullcmd, sizeof(fullcmd), argv+1); | 
					
						
							|  |  |  | 				ast_cli(fd, "No such command '%s'.\n", fullcmd); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		return help_workhorse(fd, NULL); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return RESULT_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static char *parse_args(char *s, int *max, char *argv[]) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	char *dup, *cur; | 
					
						
							|  |  |  | 	int x=0; | 
					
						
							|  |  |  | 	int quoted=0; | 
					
						
							|  |  |  | 	int escaped=0; | 
					
						
							|  |  |  | 	int whitespace=1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	dup = strdup(s); | 
					
						
							|  |  |  | 	if (dup) { | 
					
						
							|  |  |  | 		cur = dup; | 
					
						
							|  |  |  | 		while(*s) { | 
					
						
							|  |  |  | 			switch(*s) { | 
					
						
							|  |  |  | 			case '"': | 
					
						
							|  |  |  | 				/* If it's escaped, put a literal quote */ | 
					
						
							|  |  |  | 				if (escaped)  | 
					
						
							|  |  |  | 					goto normal; | 
					
						
							|  |  |  | 				else  | 
					
						
							|  |  |  | 					quoted = !quoted; | 
					
						
							| 
									
										
										
										
											2001-05-09 03:11:22 +00:00
										 |  |  | 				if (quoted && whitespace) { | 
					
						
							|  |  |  | 					/* If we're starting a quote, coming off white space start a new word, too */ | 
					
						
							|  |  |  | 					argv[x++] = cur; | 
					
						
							|  |  |  | 					whitespace=0; | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 				escaped = 0; | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			case ' ': | 
					
						
							|  |  |  | 			case '\t': | 
					
						
							|  |  |  | 				if (!quoted && !escaped) { | 
					
						
							|  |  |  | 					/* If we're not quoted, mark this as whitespace, and
 | 
					
						
							|  |  |  | 					   end the previous argument */ | 
					
						
							|  |  |  | 					whitespace = 1; | 
					
						
							|  |  |  | 					*(cur++) = '\0'; | 
					
						
							|  |  |  | 				} else | 
					
						
							|  |  |  | 					/* Otherwise, just treat it as anything else */  | 
					
						
							|  |  |  | 					goto normal; | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			case '\\': | 
					
						
							|  |  |  | 				/* If we're escaped, print a literal, otherwise enable escaping */ | 
					
						
							|  |  |  | 				if (escaped) { | 
					
						
							|  |  |  | 					goto normal; | 
					
						
							|  |  |  | 				} else { | 
					
						
							|  |  |  | 					escaped=1; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			default: | 
					
						
							|  |  |  | normal: | 
					
						
							|  |  |  | 				if (whitespace) { | 
					
						
							|  |  |  | 					if (x >= AST_MAX_ARGS -1) { | 
					
						
							|  |  |  | 						ast_log(LOG_WARNING, "Too many arguments, truncating\n"); | 
					
						
							|  |  |  | 						break; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					/* Coming off of whitespace, start the next argument */ | 
					
						
							|  |  |  | 					argv[x++] = cur; | 
					
						
							|  |  |  | 					whitespace=0; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				*(cur++) = *s; | 
					
						
							|  |  |  | 				escaped=0; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			s++; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		/* Null terminate */ | 
					
						
							|  |  |  | 		*(cur++) = '\0'; | 
					
						
							|  |  |  | 		argv[x] = NULL; | 
					
						
							|  |  |  | 		*max = x; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return dup; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-01-30 15:03:20 +00:00
										 |  |  | /* This returns the number of unique matches for the generator */ | 
					
						
							|  |  |  | int ast_cli_generatornummatches(char *text, char *word) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int matches = 0, i = 0; | 
					
						
							| 
									
										
										
										
											2003-04-08 13:45:36 +00:00
										 |  |  | 	char *buf, *oldbuf = NULL; | 
					
						
							| 
									
										
										
										
											2003-01-30 15:03:20 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	while ( (buf = ast_cli_generator(text, word, i)) ) { | 
					
						
							| 
									
										
										
										
											2003-04-08 13:45:36 +00:00
										 |  |  | 		if (++i > 1 && strcmp(buf,oldbuf) == 0)  { | 
					
						
							| 
									
										
										
										
											2003-01-30 15:03:20 +00:00
										 |  |  | 				continue; | 
					
						
							| 
									
										
										
										
											2003-04-08 13:45:36 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2003-01-30 15:03:20 +00:00
										 |  |  | 		oldbuf = buf; | 
					
						
							|  |  |  | 		matches++; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2003-11-09 20:31:54 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-01-30 15:03:20 +00:00
										 |  |  | 	return matches; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | char **ast_cli_completion_matches(char *text, char *word) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	char **match_list = NULL, *retstr, *prevstr; | 
					
						
							|  |  |  | 	size_t match_list_len, max_equal, which, i; | 
					
						
							|  |  |  | 	int matches = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	match_list_len = 1; | 
					
						
							|  |  |  | 	while ((retstr = ast_cli_generator(text, word, matches)) != NULL) { | 
					
						
							|  |  |  | 		if (matches + 1 >= match_list_len) { | 
					
						
							|  |  |  | 			match_list_len <<= 1; | 
					
						
							|  |  |  | 			match_list = realloc(match_list, match_list_len * sizeof(char *)); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		match_list[++matches] = retstr; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!match_list) | 
					
						
							|  |  |  | 		return (char **) NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	which = 2; | 
					
						
							|  |  |  | 	prevstr = match_list[1]; | 
					
						
							|  |  |  | 	max_equal = strlen(prevstr); | 
					
						
							|  |  |  | 	for (; which <= matches; which++) { | 
					
						
							|  |  |  | 		for (i = 0; i < max_equal && prevstr[i] == match_list[which][i]; i++) | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 		max_equal = i; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	retstr = malloc(max_equal + 1); | 
					
						
							|  |  |  | 	(void) strncpy(retstr, match_list[1], max_equal); | 
					
						
							|  |  |  | 	retstr[max_equal] = '\0'; | 
					
						
							|  |  |  | 	match_list[0] = retstr; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (matches + 1 >= match_list_len) | 
					
						
							|  |  |  | 		match_list = realloc(match_list, (match_list_len + 1) * sizeof(char *)); | 
					
						
							|  |  |  | 	match_list[matches + 1] = (char *) NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return (match_list); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-05-09 03:11:22 +00:00
										 |  |  | static char *__ast_cli_generator(char *text, char *word, int state, int lock) | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	char *argv[AST_MAX_ARGS]; | 
					
						
							|  |  |  | 	struct ast_cli_entry *e, *e1, *e2; | 
					
						
							|  |  |  | 	int x; | 
					
						
							|  |  |  | 	int matchnum=0; | 
					
						
							|  |  |  | 	char *dup, *res; | 
					
						
							|  |  |  | 	char fullcmd1[80]; | 
					
						
							|  |  |  | 	char fullcmd2[80]; | 
					
						
							|  |  |  | 	char matchstr[80]; | 
					
						
							|  |  |  | 	char *fullcmd; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if ((dup = parse_args(text, &x, argv))) { | 
					
						
							|  |  |  | 		join(matchstr, sizeof(matchstr), argv); | 
					
						
							| 
									
										
										
										
											2001-05-09 03:11:22 +00:00
										 |  |  | 		if (lock) | 
					
						
							| 
									
										
										
										
											2003-08-13 15:25:16 +00:00
										 |  |  | 			ast_mutex_lock(&clilock); | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 		e1 = builtins; | 
					
						
							|  |  |  | 		e2 = helpers; | 
					
						
							|  |  |  | 		while(e1->cmda[0] || e2) { | 
					
						
							|  |  |  | 			if (e2) | 
					
						
							|  |  |  | 				join(fullcmd2, sizeof(fullcmd2), e2->cmda); | 
					
						
							|  |  |  | 			if (e1->cmda[0]) | 
					
						
							|  |  |  | 				join(fullcmd1, sizeof(fullcmd1), e1->cmda); | 
					
						
							| 
									
										
										
										
											2001-10-11 18:51:39 +00:00
										 |  |  | 			if (!e1->cmda[0] ||  | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 					(e2 && (strcmp(fullcmd2, fullcmd1) < 0))) { | 
					
						
							|  |  |  | 				/* Use e2 */ | 
					
						
							|  |  |  | 				e = e2; | 
					
						
							|  |  |  | 				fullcmd = fullcmd2; | 
					
						
							|  |  |  | 				/* Increment by going to next */ | 
					
						
							|  |  |  | 				e2 = e2->next; | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				/* Use e1 */ | 
					
						
							|  |  |  | 				e = e1; | 
					
						
							|  |  |  | 				fullcmd = fullcmd1; | 
					
						
							|  |  |  | 				e1++; | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2004-08-07 19:27:54 +00:00
										 |  |  | 			if ((fullcmd[0] != '_') && !strncasecmp(matchstr, fullcmd, strlen(matchstr))) { | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 				/* We contain the first part of one or more commands */ | 
					
						
							|  |  |  | 				matchnum++; | 
					
						
							|  |  |  | 				if (matchnum > state) { | 
					
						
							|  |  |  | 					/* Now, what we're supposed to return is the next word... */ | 
					
						
							| 
									
										
										
										
											2004-05-06 20:49:24 +00:00
										 |  |  | 					if (!ast_strlen_zero(word) && x>0) { | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 						res = e->cmda[x-1]; | 
					
						
							|  |  |  | 					} else { | 
					
						
							|  |  |  | 						res = e->cmda[x]; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					if (res) { | 
					
						
							| 
									
										
										
										
											2001-05-09 03:11:22 +00:00
										 |  |  | 						if (lock) | 
					
						
							| 
									
										
										
										
											2003-08-13 15:25:16 +00:00
										 |  |  | 							ast_mutex_unlock(&clilock); | 
					
						
							| 
									
										
										
										
											2003-04-08 13:45:36 +00:00
										 |  |  | 						free(dup); | 
					
						
							| 
									
										
										
										
											2003-04-30 19:04:00 +00:00
										 |  |  | 						return res ? strdup(res) : NULL; | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			if (e->generator && !strncasecmp(matchstr, fullcmd, strlen(fullcmd))) { | 
					
						
							|  |  |  | 				/* We have a command in its entirity within us -- theoretically only one
 | 
					
						
							|  |  |  | 				   command can have this occur */ | 
					
						
							| 
									
										
										
										
											2004-08-07 19:27:54 +00:00
										 |  |  | 				fullcmd = e->generator(matchstr, word, (!ast_strlen_zero(word) ? (x - 1) : (x)), state); | 
					
						
							| 
									
										
										
										
											2001-05-09 03:11:22 +00:00
										 |  |  | 				if (lock) | 
					
						
							| 
									
										
										
										
											2003-08-13 15:25:16 +00:00
										 |  |  | 					ast_mutex_unlock(&clilock); | 
					
						
							| 
									
										
										
										
											2004-04-08 05:28:53 +00:00
										 |  |  | 				free(dup); | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 				return fullcmd; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			 | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2001-05-09 03:11:22 +00:00
										 |  |  | 		if (lock) | 
					
						
							| 
									
										
										
										
											2003-08-13 15:25:16 +00:00
										 |  |  | 			ast_mutex_unlock(&clilock); | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 		free(dup); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-05-09 03:11:22 +00:00
										 |  |  | char *ast_cli_generator(char *text, char *word, int state) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return __ast_cli_generator(text, word, state, 1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | int ast_cli_command(int fd, char *s) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	char *argv[AST_MAX_ARGS]; | 
					
						
							|  |  |  | 	struct ast_cli_entry *e; | 
					
						
							|  |  |  | 	int x; | 
					
						
							|  |  |  | 	char *dup; | 
					
						
							|  |  |  | 	x = AST_MAX_ARGS; | 
					
						
							|  |  |  | 	if ((dup = parse_args(s, &x, argv))) { | 
					
						
							|  |  |  | 		/* We need at least one entry, or ignore */ | 
					
						
							|  |  |  | 		if (x > 0) { | 
					
						
							| 
									
										
										
										
											2003-08-13 15:25:16 +00:00
										 |  |  | 			ast_mutex_lock(&clilock); | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 			e = find_cli(argv, 0); | 
					
						
							| 
									
										
										
										
											2002-05-13 22:29:39 +00:00
										 |  |  | 			if (e) | 
					
						
							|  |  |  | 				e->inuse++; | 
					
						
							| 
									
										
										
										
											2003-08-13 15:25:16 +00:00
										 |  |  | 			ast_mutex_unlock(&clilock); | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 			if (e) { | 
					
						
							|  |  |  | 				switch(e->handler(fd, x, argv)) { | 
					
						
							|  |  |  | 				case RESULT_SHOWUSAGE: | 
					
						
							|  |  |  | 					ast_cli(fd, e->usage); | 
					
						
							|  |  |  | 					break; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} else  | 
					
						
							| 
									
										
										
										
											1999-12-27 19:20:20 +00:00
										 |  |  | 				ast_cli(fd, "No such command '%s' (type 'help' for help)\n", find_best(argv)); | 
					
						
							| 
									
										
										
										
											2002-05-13 22:29:39 +00:00
										 |  |  | 			if (e) { | 
					
						
							| 
									
										
										
										
											2003-08-13 15:25:16 +00:00
										 |  |  | 				ast_mutex_lock(&clilock); | 
					
						
							| 
									
										
										
										
											2002-05-13 22:29:39 +00:00
										 |  |  | 				e->inuse--; | 
					
						
							| 
									
										
										
										
											2003-08-13 15:25:16 +00:00
										 |  |  | 				ast_mutex_unlock(&clilock); | 
					
						
							| 
									
										
										
										
											2002-05-13 22:29:39 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											1999-12-19 22:38:55 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		free(dup); | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		ast_log(LOG_WARNING, "Out of memory\n");	 | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } |