mirror of
				https://github.com/asterisk/asterisk.git
				synced 2025-10-31 18:55:19 +00:00 
			
		
		
		
	introduce a bit of regexp support in the internal CLI api.
Now you can specify a cli command as
	"console autoanswer [on|off]"
which means the on|off argument is optional, or
	"console {mute|unmute}"
which means the mute|unmute argument is mandatory.
The blocks in [] or {} do not necessarily need to be at the
end of the string.
Completions for the variant parts are generated automatically.
This should significantly simplify the implementation of
the various handlers.
git-svn-id: https://origsvn.digium.com/svn/asterisk/trunk@47787 65c4cc65-6c06-0410-ace0-fbb531ad65f3
			
			
This commit is contained in:
		| @@ -174,7 +174,7 @@ struct ast_cli_entry { | |||||||
| 	int inuse; /*!< For keeping track of usage */ | 	int inuse; /*!< For keeping track of usage */ | ||||||
| 	struct module *module;	/*!< module this belongs to */ | 	struct module *module;	/*!< module this belongs to */ | ||||||
| 	char *_full_cmd;	/*!< built at load time from cmda[] */ | 	char *_full_cmd;	/*!< built at load time from cmda[] */ | ||||||
|  | 	int cmdlen;		/*!< len up to the first invalid char [<{% */ | ||||||
| 	/*! \brief This gets set in ast_cli_register() | 	/*! \brief This gets set in ast_cli_register() | ||||||
| 	  It then gets set to something different when the deprecated command | 	  It then gets set to something different when the deprecated command | ||||||
| 	  is run for the first time (ie; after we warn the user that it's deprecated) | 	  is run for the first time (ie; after we warn the user that it's deprecated) | ||||||
|   | |||||||
							
								
								
									
										257
									
								
								main/cli.c
									
									
									
									
									
								
							
							
						
						
									
										257
									
								
								main/cli.c
									
									
									
									
									
								
							| @@ -197,14 +197,13 @@ static char *handle_verbose(struct ast_cli_entry *e, int cmd, struct ast_cli_arg | |||||||
| 	int oldval = option_verbose; | 	int oldval = option_verbose; | ||||||
| 	int newlevel; | 	int newlevel; | ||||||
| 	int atleast = 0; | 	int atleast = 0; | ||||||
| 	static char *choices[] = { "off", "atleast", NULL }; |  | ||||||
| 	int fd = a->fd; | 	int fd = a->fd; | ||||||
| 	int argc = a->argc; | 	int argc = a->argc; | ||||||
| 	char **argv = a->argv; | 	char **argv = a->argv; | ||||||
|  |  | ||||||
| 	switch (cmd) { | 	switch (cmd) { | ||||||
| 	case CLI_INIT: | 	case CLI_INIT: | ||||||
| 		e->command = "core set verbose"; | 		e->command = "core set verbose [off|atleast]"; | ||||||
| 		e->usage = | 		e->usage = | ||||||
| 			"Usage: core set verbose [atleast] <level>\n" | 			"Usage: core set verbose [atleast] <level>\n" | ||||||
| 			"       core set verbose off\n" | 			"       core set verbose off\n" | ||||||
| @@ -214,26 +213,24 @@ static char *handle_verbose(struct ast_cli_entry *e, int cmd, struct ast_cli_arg | |||||||
| 		return NULL; | 		return NULL; | ||||||
|  |  | ||||||
| 	case CLI_GENERATE: | 	case CLI_GENERATE: | ||||||
| 		if (a->pos > e->args) | 		return NULL; | ||||||
| 			return NULL; |  | ||||||
| 		return ast_cli_complete(a->word, choices, a->n); |  | ||||||
| 	} | 	} | ||||||
| 	/* all the above return, so we proceed with the handler. | 	/* all the above return, so we proceed with the handler. | ||||||
| 	 * we are guaranteed to be called with argc >= e->args; | 	 * we are guaranteed to be called with argc >= e->args; | ||||||
| 	 */ | 	 */ | ||||||
|  |  | ||||||
| 	if (argc < e->args + 1) | 	if (argc < e->args) | ||||||
| 		return CLI_SHOWUSAGE; | 		return CLI_SHOWUSAGE; | ||||||
|  |  | ||||||
| 	if (argc == e->args + 1 && !strcasecmp(argv[e->args], "off")) { | 	if (argc == e->args && !strcasecmp(argv[e->args - 1], "off")) { | ||||||
| 		newlevel = 0; | 		newlevel = 0; | ||||||
| 		goto done; | 		goto done; | ||||||
| 	} | 	} | ||||||
| 	if (!strcasecmp(argv[e->args], "atleast")) | 	if (!strcasecmp(argv[e->args-1], "atleast")) | ||||||
| 		atleast = 1; | 		atleast = 1; | ||||||
| 	if (argc != e->args + atleast + 1) | 	if (argc != e->args + atleast) | ||||||
| 		return CLI_SHOWUSAGE; | 		return CLI_SHOWUSAGE; | ||||||
| 	if (sscanf(argv[e->args + atleast], "%d", &newlevel) != 1) | 	if (sscanf(argv[e->args + atleast - 1], "%d", &newlevel) != 1) | ||||||
| 		return CLI_SHOWUSAGE; | 		return CLI_SHOWUSAGE; | ||||||
|  |  | ||||||
| done: | done: | ||||||
| @@ -257,14 +254,13 @@ static char *handle_set_debug(struct ast_cli_entry *e, int cmd, struct ast_cli_a | |||||||
| 	int newlevel; | 	int newlevel; | ||||||
| 	int atleast = 0; | 	int atleast = 0; | ||||||
| 	char *filename = '\0'; | 	char *filename = '\0'; | ||||||
| 	static char *choices[] = { "off", "atleast", NULL }; |  | ||||||
| 	int fd = a->fd; | 	int fd = a->fd; | ||||||
| 	int argc = a->argc; | 	int argc = a->argc; | ||||||
| 	char **argv = a->argv; | 	char **argv = a->argv; | ||||||
|  |  | ||||||
| 	switch (cmd) { | 	switch (cmd) { | ||||||
| 	case CLI_INIT: | 	case CLI_INIT: | ||||||
| 		e->command = "core set debug"; | 		e->command = "core set debug [off|atleast]"; | ||||||
| 		e->usage = | 		e->usage = | ||||||
| 			"Usage: core set debug [atleast] <level> [filename]\n" | 			"Usage: core set debug [atleast] <level> [filename]\n" | ||||||
| 			"       core set debug off\n" | 			"       core set debug off\n" | ||||||
| @@ -275,32 +271,26 @@ static char *handle_set_debug(struct ast_cli_entry *e, int cmd, struct ast_cli_a | |||||||
| 		return NULL; | 		return NULL; | ||||||
|  |  | ||||||
| 	case CLI_GENERATE: | 	case CLI_GENERATE: | ||||||
| 		if (a->pos > e->args) | 		return NULL; | ||||||
| 			return NULL; |  | ||||||
| 		return ast_cli_complete(a->word, choices, a->n); |  | ||||||
| 	} | 	} | ||||||
| 	/* all the above return, so we proceed with the handler. | 	if (argc < e->args) | ||||||
| 	 * we are guaranteed to be called with argc >= e->args; |  | ||||||
| 	 */ |  | ||||||
|  |  | ||||||
| 	if (argc < e->args + 1) |  | ||||||
| 		return CLI_SHOWUSAGE; | 		return CLI_SHOWUSAGE; | ||||||
|  |  | ||||||
| 	if (argc == e->args + 1 && !strcasecmp(argv[e->args], "off")) { | 	if (argc == e->args && !strcasecmp(argv[e->args-1], "off")) { | ||||||
| 		newlevel = 0; | 		newlevel = 0; | ||||||
| 		goto done; | 		goto done; | ||||||
| 	} | 	} | ||||||
| 	if (!strcasecmp(argv[e->args], "atleast")) | 	if (!strcasecmp(argv[e->args-1], "atleast")) | ||||||
| 		atleast = 1; | 		atleast = 1; | ||||||
| 	if (argc < e->args + atleast + 1 || argc > e->args + atleast + 2) | 	if (argc < e->args + atleast || argc > e->args + atleast + 1) | ||||||
| 		return CLI_SHOWUSAGE; | 		return CLI_SHOWUSAGE; | ||||||
| 	if (sscanf(argv[e->args + atleast], "%d", &newlevel) != 1) | 	if (sscanf(argv[e->args + atleast-1], "%d", &newlevel) != 1) | ||||||
| 		return CLI_SHOWUSAGE; | 		return CLI_SHOWUSAGE; | ||||||
|  |  | ||||||
| 	if (argc == e->args + atleast + 1) { | 	if (argc == e->args + atleast) { | ||||||
| 		debug_filename[0] = '\0'; | 		debug_filename[0] = '\0'; | ||||||
| 	} else { | 	} else { | ||||||
| 		ast_copy_string(debug_filename, argv[e->args + atleast + 1], sizeof(debug_filename)); | 		ast_copy_string(debug_filename, argv[e->args + atleast], sizeof(debug_filename)); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| done: | done: | ||||||
| @@ -545,7 +535,7 @@ static char *handle_chanlist(struct ast_cli_entry *e, int cmd, struct ast_cli_ar | |||||||
|  |  | ||||||
| 	switch (cmd) { | 	switch (cmd) { | ||||||
| 	case CLI_INIT: | 	case CLI_INIT: | ||||||
| 		e->command = "core show channels"; | 		e->command = "core show channels [concise|verbose]"; | ||||||
| 		e->usage = | 		e->usage = | ||||||
| 			"Usage: core show channels [concise|verbose]\n" | 			"Usage: core show channels [concise|verbose]\n" | ||||||
| 			"       Lists currently defined channels and some information about them. If\n" | 			"       Lists currently defined channels and some information about them. If\n" | ||||||
| @@ -554,23 +544,21 @@ static char *handle_chanlist(struct ast_cli_entry *e, int cmd, struct ast_cli_ar | |||||||
| 			"       more and longer fields.\n"; | 			"       more and longer fields.\n"; | ||||||
| 		return NULL; | 		return NULL; | ||||||
|  |  | ||||||
| 	case CLI_GENERATE: { | 	case CLI_GENERATE: | ||||||
| 		static char *choices[] = { "concise", "verbose", NULL }; | 		return NULL; | ||||||
| 		return (a->pos != e->args) ? NULL : ast_cli_complete(a->word, choices, a->n); |  | ||||||
| 		} |  | ||||||
| 	} | 	} | ||||||
| 	fd = a->fd; | 	fd = a->fd; | ||||||
| 	argc = a->argc; | 	argc = a->argc; | ||||||
| 	argv = a->argv; | 	argv = a->argv; | ||||||
|  |  | ||||||
| 	if (a->argc == e->args + 1) { | 	if (a->argc == e->args) { | ||||||
| 		if (!strcasecmp(argv[3],"concise")) | 		if (!strcasecmp(argv[e->args-1],"concise")) | ||||||
| 			concise = 1; | 			concise = 1; | ||||||
| 		else if (!strcasecmp(argv[3],"verbose")) | 		else if (!strcasecmp(argv[e->args-1],"verbose")) | ||||||
| 			verbose = 1; | 			verbose = 1; | ||||||
| 		else | 		else | ||||||
| 			return CLI_SHOWUSAGE; | 			return CLI_SHOWUSAGE; | ||||||
| 	} else if (a->argc != e->args) | 	} else if (a->argc != e->args - 1) | ||||||
| 		return CLI_SHOWUSAGE; | 		return CLI_SHOWUSAGE; | ||||||
|  |  | ||||||
| 	if (!concise && !verbose) | 	if (!concise && !verbose) | ||||||
| @@ -1107,18 +1095,40 @@ static struct ast_cli_entry cli_cli[] = { | |||||||
| 	softhangup_help, complete_ch_3 }, | 	softhangup_help, complete_ch_3 }, | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  | /*! | ||||||
|  |  * Some regexp characters in cli arguments are reserved and used as separators. | ||||||
|  |  */ | ||||||
|  | static const char cli_rsvd[] = "[]{}|*"; | ||||||
|  |  | ||||||
|  | /*! | ||||||
|  |  * initialize the _full_cmd string and related parameters, | ||||||
|  |  * return 0 on success, -1 on error. | ||||||
|  |  */ | ||||||
|  | static int set_full_cmd(struct ast_cli_entry *e) | ||||||
|  | { | ||||||
|  | 	int i; | ||||||
|  | 	char buf[80]; | ||||||
|  |  | ||||||
|  | 	ast_join(buf, sizeof(buf), e->cmda); | ||||||
|  | 	e->_full_cmd = strdup(buf); | ||||||
|  | 	if (!e->_full_cmd) { | ||||||
|  | 		ast_log(LOG_WARNING, "-- cannot allocate <%s>\n", buf); | ||||||
|  | 		return -1; | ||||||
|  | 	} | ||||||
|  | 	e->cmdlen = strcspn(e->_full_cmd, cli_rsvd); | ||||||
|  | 	for (i = 0; e->cmda[i]; i++) | ||||||
|  | 		; | ||||||
|  | 	e->args = i; | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
| /*! \brief initialize the _full_cmd string in * each of the builtins. */ | /*! \brief initialize the _full_cmd string in * each of the builtins. */ | ||||||
| void ast_builtins_init(void) | void ast_builtins_init(void) | ||||||
| { | { | ||||||
| 	struct ast_cli_entry *e; | 	struct ast_cli_entry *e; | ||||||
|  |  | ||||||
| 	for (e = builtins; e->cmda[0] != NULL; e++) { | 	for (e = builtins; e->cmda[0] != NULL; e++) | ||||||
| 		char buf[80]; | 		set_full_cmd(e); | ||||||
| 		ast_join(buf, sizeof(buf), e->cmda); |  | ||||||
| 		e->_full_cmd = strdup(buf); |  | ||||||
| 		if (!e->_full_cmd) |  | ||||||
| 			ast_log(LOG_WARNING, "-- cannot allocate <%s>\n", buf); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	ast_cli_register_multiple(cli_cli, sizeof(cli_cli) / sizeof(struct ast_cli_entry)); | 	ast_cli_register_multiple(cli_cli, sizeof(cli_cli) / sizeof(struct ast_cli_entry)); | ||||||
| } | } | ||||||
| @@ -1159,12 +1169,77 @@ static struct ast_cli_entry *cli_next(struct cli_iterator *i) | |||||||
| 	return e; | 	return e; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | /*! | ||||||
|  |  * match a word in the CLI entry. | ||||||
|  |  * returns -1 on mismatch, 0 on match of an optional word, | ||||||
|  |  * 1 on match of a full word. | ||||||
|  |  */ | ||||||
|  | static int word_match(const char *cmd, const char *cli_word) | ||||||
|  | { | ||||||
|  | 	int l; | ||||||
|  | 	char *pos; | ||||||
|  |  | ||||||
|  | 	if (ast_strlen_zero(cmd) || ast_strlen_zero(cli_word)) | ||||||
|  | 		return -1; | ||||||
|  | 	if (!strchr(cli_rsvd, cli_word[0])) /* normal match */ | ||||||
|  | 		return (strcasecmp(cmd, cli_word) == 0) ? 1 : -1; | ||||||
|  | 	/* regexp match, takes [foo|bar] or {foo|bar} */ | ||||||
|  | 	l = strlen(cmd); | ||||||
|  | 	pos = strcasestr(cli_word, cmd); | ||||||
|  | 	if (pos == NULL) /* not found, say ok if optional */ | ||||||
|  | 		return cli_word[0] == '[' ? 0 : -1; | ||||||
|  | 	if (pos == cli_word)	/* no valid match at the beginning */ | ||||||
|  | 		return -1; | ||||||
|  | 	if (strchr(cli_rsvd, pos[-1]) && strchr(cli_rsvd, pos[l])) | ||||||
|  | 		return 1;	/* valid match */ | ||||||
|  | 	return -1;	/* not found */ | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /*! \brief if word is a valid prefix for token, returns the pos-th | ||||||
|  |  * match as a malloced string, or NULL otherwise. | ||||||
|  |  * Always tell in *actual how many matches we got. | ||||||
|  |  */ | ||||||
|  | static char *is_prefix(const char *word, const char *token, | ||||||
|  | 	int pos, int *actual) | ||||||
|  | { | ||||||
|  | 	int lw; | ||||||
|  | 	char *s, *t1; | ||||||
|  |  | ||||||
|  | 	*actual = 0; | ||||||
|  | 	if (ast_strlen_zero(token)) | ||||||
|  | 		return NULL; | ||||||
|  | 	if (ast_strlen_zero(word)) | ||||||
|  | 		word = "";	/* dummy */ | ||||||
|  | 	lw = strlen(word); | ||||||
|  | 	if (strcspn(word, cli_rsvd) != lw) | ||||||
|  | 		return NULL;	/* no match if word has reserved chars */ | ||||||
|  | 	if (strchr(cli_rsvd, token[0]) == NULL) {	/* regular match */ | ||||||
|  | 		if (strncasecmp(token, word, lw))	/* no match */ | ||||||
|  | 			return NULL; | ||||||
|  | 		*actual = 1; | ||||||
|  | 		return (pos != 0) ? NULL : strdup(token); | ||||||
|  | 	} | ||||||
|  | 	/* now handle regexp match */ | ||||||
|  | 	t1 = ast_strdupa(token + 1);	/* copy, skipping first char */ | ||||||
|  | 	while (pos >= 0 && (s = strsep(&t1, cli_rsvd)) && *s) { | ||||||
|  | 		if (strncasecmp(s, word, lw))	/* no match */ | ||||||
|  | 			continue; | ||||||
|  | 		(*actual)++; | ||||||
|  | 		if (pos-- == 0) | ||||||
|  | 			return strdup(s); | ||||||
|  | 	} | ||||||
|  | 	return NULL; | ||||||
|  | } | ||||||
|  |  | ||||||
| /*! | /*! | ||||||
|  * \brief locate a cli command in the 'helpers' list (which must be locked). |  * \brief locate a cli command in the 'helpers' list (which must be locked). | ||||||
|  * exact has 3 values: |  * exact has 3 values: | ||||||
|  *      0       returns if the search key is equal or longer than the entry. |  *      0       returns if the search key is equal or longer than the entry. | ||||||
|  |  *		note that trailing optional arguments are skipped. | ||||||
|  *      -1      true if the mismatch is on the last word XXX not true! |  *      -1      true if the mismatch is on the last word XXX not true! | ||||||
|  *      1       true only on complete, exact match. |  *      1       true only on complete, exact match. | ||||||
|  |  * | ||||||
|  |  * The search compares word by word taking care of regexps in e->cmda | ||||||
|  */ |  */ | ||||||
| static struct ast_cli_entry *find_cli(char *const cmds[], int match_type) | static struct ast_cli_entry *find_cli(char *const cmds[], int match_type) | ||||||
| { | { | ||||||
| @@ -1173,32 +1248,37 @@ static struct ast_cli_entry *find_cli(char *const cmds[], int match_type) | |||||||
| 	struct cli_iterator i = { NULL, NULL}; | 	struct cli_iterator i = { NULL, NULL}; | ||||||
|  |  | ||||||
| 	while( (e = cli_next(&i)) ) { | 	while( (e = cli_next(&i)) ) { | ||||||
| 		int y; | 		/* word-by word regexp comparison */ | ||||||
| 		for (y = 0 ; cmds[y] && e->cmda[y]; y++) { | 		char * const *src = cmds; | ||||||
| 			if (strcasecmp(e->cmda[y], cmds[y])) | 		char * const *dst = e->cmda; | ||||||
|  | 		int n = 0; | ||||||
|  | 		for (;; dst++, src += n) { | ||||||
|  | 			n = word_match(*src, *dst); | ||||||
|  | 			if (n < 0) | ||||||
| 				break; | 				break; | ||||||
| 		} | 		} | ||||||
| 		if (e->cmda[y] == NULL) {	/* no more words in candidate */ | 		if (ast_strlen_zero(*dst) || ((*dst)[0] == '[' && ast_strlen_zero(dst[1]))) { | ||||||
| 			if (cmds[y] == NULL)	/* this is an exact match, cannot do better */ | 			/* no more words in 'e' */ | ||||||
|  | 			if (ast_strlen_zero(*src))	/* exact match, cannot do better */ | ||||||
| 				break; | 				break; | ||||||
| 			/* here the search key is longer than the candidate */ | 			/* Here, cmds has more words than the entry 'e' */ | ||||||
| 			if (match_type != 0)	/* but we look for almost exact match... */ | 			if (match_type != 0)	/* but we look for almost exact match... */ | ||||||
| 				continue;	/* so we skip this one. */ | 				continue;	/* so we skip this one. */ | ||||||
| 			/* otherwise we like it (case 0) */ | 			/* otherwise we like it (case 0) */ | ||||||
| 		} else {			/* still words in candidate */ | 		} else {	/* still words in 'e' */ | ||||||
| 			if (cmds[y] == NULL)	/* search key is shorter, not good */ | 			if (ast_strlen_zero(*src)) | ||||||
|  | 				continue; /* cmds is shorter than 'e', not good */ | ||||||
|  | 			/* Here we have leftover words in cmds and 'e', | ||||||
|  | 			 * but there is a mismatch. We only accept this one if match_type == -1 | ||||||
|  | 			 * and this is the last word for both. | ||||||
|  | 			 */ | ||||||
|  | 			if (match_type != -1 || !ast_strlen_zero(src[1]) || | ||||||
|  | 			    !ast_strlen_zero(dst[1]))	/* not the one we look for */ | ||||||
| 				continue; | 				continue; | ||||||
| 			/* if we get here, both words exist but there is a mismatch */ | 			/* good, we are in case match_type == -1 and mismatch on last word */ | ||||||
| 			if (match_type == 0)	/* not the one we look for */ |  | ||||||
| 				continue; |  | ||||||
| 			if (match_type == 1)	/* not the one we look for */ |  | ||||||
| 				continue; |  | ||||||
| 			if (cmds[y+1] != NULL || e->cmda[y+1] != NULL)	/* not the one we look for */ |  | ||||||
| 				continue; |  | ||||||
| 			/* we are in case match_type == -1 and mismatch on last word */ |  | ||||||
| 		} | 		} | ||||||
| 		if (y > matchlen) {	/* remember the candidate */ | 		if (src - cmds > matchlen) {	/* remember the candidate */ | ||||||
| 			matchlen = y; | 			matchlen = src - cmds; | ||||||
| 			cand = e; | 			cand = e; | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| @@ -1251,7 +1331,6 @@ static int __ast_cli_unregister(struct ast_cli_entry *e, struct ast_cli_entry *e | |||||||
| static int __ast_cli_register(struct ast_cli_entry *e, struct ast_cli_entry *ed) | static int __ast_cli_register(struct ast_cli_entry *e, struct ast_cli_entry *ed) | ||||||
| { | { | ||||||
| 	struct ast_cli_entry *cur; | 	struct ast_cli_entry *cur; | ||||||
| 	char fulle[80] =""; |  | ||||||
| 	int i, lf, ret = -1; | 	int i, lf, ret = -1; | ||||||
|  |  | ||||||
| 	if (e->handler == NULL) {	/* new style entry, run the handler to init fields */ | 	if (e->handler == NULL) {	/* new style entry, run the handler to init fields */ | ||||||
| @@ -1274,22 +1353,20 @@ static int __ast_cli_register(struct ast_cli_entry *e, struct ast_cli_entry *ed) | |||||||
| 		} | 		} | ||||||
| 		*dst++ = NULL; | 		*dst++ = NULL; | ||||||
| 	} | 	} | ||||||
| 	for (i = 0; e->cmda[i]; i++) | 	if (set_full_cmd(e)) | ||||||
| 		; | 		goto done; | ||||||
| 	e->args = i; |  | ||||||
| 	ast_join(fulle, sizeof(fulle), e->cmda); |  | ||||||
| 	AST_LIST_LOCK(&helpers); | 	AST_LIST_LOCK(&helpers); | ||||||
| 	 | 	 | ||||||
| 	if (find_cli(e->cmda, 1)) { | 	if (find_cli(e->cmda, 1)) { | ||||||
| 		AST_LIST_UNLOCK(&helpers); | 		AST_LIST_UNLOCK(&helpers); | ||||||
| 		ast_log(LOG_WARNING, "Command '%s' already registered (or something close enough)\n", fulle); | 		ast_log(LOG_WARNING, "Command '%s' already registered (or something close enough)\n", e->_full_cmd); | ||||||
|  | 		free(e->_full_cmd); | ||||||
|  | 		e->_full_cmd = NULL; | ||||||
| 		goto done; | 		goto done; | ||||||
| 	} | 	} | ||||||
| 	e->_full_cmd = ast_strdup(fulle); | 	if (!ed) { | ||||||
| 	if (!e->_full_cmd) | 		e->deprecated = 0; | ||||||
| 		goto done; | 	} else { | ||||||
|  |  | ||||||
| 	if (ed) { |  | ||||||
| 		e->deprecated = 1; | 		e->deprecated = 1; | ||||||
| 		e->summary = ed->summary; | 		e->summary = ed->summary; | ||||||
| 		e->usage = ed->usage; | 		e->usage = ed->usage; | ||||||
| @@ -1299,16 +1376,14 @@ static int __ast_cli_register(struct ast_cli_entry *e, struct ast_cli_entry *ed) | |||||||
| 		   To show command B, you just need to always use ed->_full_cmd. | 		   To show command B, you just need to always use ed->_full_cmd. | ||||||
| 		 */ | 		 */ | ||||||
| 		e->_deprecated_by = S_OR(ed->_deprecated_by, ed->_full_cmd); | 		e->_deprecated_by = S_OR(ed->_deprecated_by, ed->_full_cmd); | ||||||
| 	} else { |  | ||||||
| 		e->deprecated = 0; |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	lf = strlen(fulle); | 	lf = e->cmdlen; | ||||||
| 	AST_LIST_TRAVERSE_SAFE_BEGIN(&helpers, cur, list) { | 	AST_LIST_TRAVERSE_SAFE_BEGIN(&helpers, cur, list) { | ||||||
| 		int len = strlen(cur->_full_cmd); | 		int len = cur->cmdlen; | ||||||
| 		if (lf < len) | 		if (lf < len) | ||||||
| 			len = lf; | 			len = lf; | ||||||
| 		if (strncasecmp(fulle, cur->_full_cmd, len) < 0) { | 		if (strncasecmp(e->_full_cmd, cur->_full_cmd, len) < 0) { | ||||||
| 			AST_LIST_INSERT_BEFORE_CURRENT(&helpers, e, list);  | 			AST_LIST_INSERT_BEFORE_CURRENT(&helpers, e, list);  | ||||||
| 			break; | 			break; | ||||||
| 		} | 		} | ||||||
| @@ -1390,7 +1465,7 @@ static int help1(int fd, char *match[], int locked) | |||||||
| 			continue; | 			continue; | ||||||
| 		if (match && strncasecmp(matchstr, e->_full_cmd, len)) | 		if (match && strncasecmp(matchstr, e->_full_cmd, len)) | ||||||
| 			continue; | 			continue; | ||||||
| 		ast_cli(fd, "%25.25s  %s\n", e->_full_cmd, S_OR(e->summary, "<no description available>")); | 		ast_cli(fd, "%30.30s %s\n", e->_full_cmd, S_OR(e->summary, "<no description available>")); | ||||||
| 		found++; | 		found++; | ||||||
| 	} | 	} | ||||||
| 	AST_LIST_UNLOCK(&helpers); | 	AST_LIST_UNLOCK(&helpers); | ||||||
| @@ -1558,6 +1633,9 @@ char **ast_cli_completion_matches(const char *text, const char *word) | |||||||
| 	return match_list; | 	return match_list; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * generate the entry at position 'state' | ||||||
|  |  */ | ||||||
| static char *__ast_cli_generator(const char *text, const char *word, int state, int lock) | static char *__ast_cli_generator(const char *text, const char *word, int state, int lock) | ||||||
| { | { | ||||||
| 	char *argv[AST_MAX_ARGS]; | 	char *argv[AST_MAX_ARGS]; | ||||||
| @@ -1584,15 +1662,24 @@ static char *__ast_cli_generator(const char *text, const char *word, int state, | |||||||
| 	if (lock) | 	if (lock) | ||||||
| 		AST_LIST_LOCK(&helpers); | 		AST_LIST_LOCK(&helpers); | ||||||
| 	while ( (e = cli_next(&i)) ) { | 	while ( (e = cli_next(&i)) ) { | ||||||
| 		int lc = strlen(e->_full_cmd); | 		/* XXX repeated code */ | ||||||
| 		if (e->_full_cmd[0] != '_' && lc > 0 && matchlen <= lc && | 		int src = 0, dst = 0, n = 0; | ||||||
| 				!strncasecmp(matchstr, e->_full_cmd, matchlen)) { | 		for (;; dst++, src += n) { | ||||||
| 			/* Found initial part, return a copy of the next word... */ | 			n = word_match(argv[src], e->cmda[dst]); | ||||||
| 			if (e->cmda[argindex] && ++matchnum > state) { | 			if (n < 0) | ||||||
| 				ret = strdup(e->cmda[argindex]); /* we need a malloced string */ |  | ||||||
| 				break; | 				break; | ||||||
| 			} | 		} | ||||||
| 		} else if (!strncasecmp(matchstr, e->_full_cmd, lc) && matchstr[lc] < 33) { |  | ||||||
|  | 		if (src < argindex)	/* not a match */ | ||||||
|  | 			continue; | ||||||
|  | 		ret = is_prefix(argv[src], e->cmda[dst], state - matchnum, &n); | ||||||
|  | 		matchnum += n;	/* this many matches here */ | ||||||
|  | 		if (ret) { | ||||||
|  | 			if (matchnum > state) | ||||||
|  | 				break; | ||||||
|  | 			free(ret); | ||||||
|  | 			ret = NULL; | ||||||
|  | 		} else if (ast_strlen_zero(e->cmda[dst])) { | ||||||
| 			/* This entry is a prefix of the command string entered | 			/* This entry is a prefix of the command string entered | ||||||
| 			 * (only one entry in the list should have this property). | 			 * (only one entry in the list should have this property). | ||||||
| 			 * Run the generator if one is available. In any case we are done. | 			 * Run the generator if one is available. In any case we are done. | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user