mirror of https://github.com/asterisk/asterisk
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
391 lines
13 KiB
391 lines
13 KiB
/*
|
|
* Asterisk -- An open source telephony toolkit.
|
|
*
|
|
* Copyright (C) 1999 - 2005, Digium, Inc.
|
|
*
|
|
* Mark Spencer <markster@digium.com>
|
|
*
|
|
* See http://www.asterisk.org for more information about
|
|
* the Asterisk project. Please do not directly contact
|
|
* any of the maintainers of this project for assistance;
|
|
* the project provides a web site, mailing lists and IRC
|
|
* channels for your use.
|
|
*
|
|
* This program is free software, distributed under the terms of
|
|
* the GNU General Public License Version 2. See the LICENSE file
|
|
* at the top of the source tree.
|
|
*/
|
|
|
|
/*! \file
|
|
* \brief Standard Command Line Interface
|
|
*/
|
|
|
|
#ifndef _ASTERISK_CLI_H
|
|
#define _ASTERISK_CLI_H
|
|
|
|
#if defined(__cplusplus) || defined(c_plusplus)
|
|
extern "C" {
|
|
#endif
|
|
|
|
#include "asterisk/linkedlists.h"
|
|
#include "asterisk/strings.h"
|
|
|
|
void ast_cli(int fd, const char *fmt, ...)
|
|
__attribute__((format(printf, 2, 3)));
|
|
|
|
/* dont check permissions while passing this option as a 'uid'
|
|
* to the cli_has_permissions() function. */
|
|
#define CLI_NO_PERMS -1
|
|
|
|
#define RESULT_SUCCESS 0
|
|
#define RESULT_SHOWUSAGE 1
|
|
#define RESULT_FAILURE 2
|
|
|
|
#define CLI_SUCCESS (char *)RESULT_SUCCESS
|
|
#define CLI_SHOWUSAGE (char *)RESULT_SHOWUSAGE
|
|
#define CLI_FAILURE (char *)RESULT_FAILURE
|
|
|
|
#define AST_MAX_CMD_LEN 16
|
|
|
|
#define AST_MAX_ARGS 64
|
|
|
|
#define AST_CLI_COMPLETE_EOF "_EOF_"
|
|
|
|
/*!
|
|
* In many cases we need to print singular or plural
|
|
* words depending on a count. This macro helps us e.g.
|
|
* printf("we have %d object%s", n, ESS(n));
|
|
*/
|
|
#define ESS(x) ((x) == 1 ? "" : "s")
|
|
|
|
/*!
|
|
* \brief Return Yes or No depending on the argument.
|
|
*
|
|
* Note that this should probably still be used for CLI commands instead of
|
|
* AST_YESNO(), in the off chance we someday want to translate the CLI.
|
|
*
|
|
* \param x Boolean value
|
|
* \return "Yes" if x is true (non-zero)
|
|
* \return "No" if x is false (zero)
|
|
*/
|
|
#define AST_CLI_YESNO(x) AST_YESNO(x)
|
|
|
|
/*! \brief return On or Off depending on the argument.
|
|
* This is used in many places in CLI command, having a function to generate
|
|
* this helps maintaining a consistent output (and possibly emitting the
|
|
* output in other languages, at some point).
|
|
*/
|
|
#define AST_CLI_ONOFF(x) (x) ? "On" : "Off"
|
|
|
|
/*! \page CLI_command_API CLI command API
|
|
|
|
CLI commands are described by a struct ast_cli_entry that contains
|
|
all the components for their implementation.
|
|
|
|
In the "old-style" format, the record must contain:
|
|
- a NULL-terminated array of words constituting the command, e.g.
|
|
{ "set", "debug", "on", NULL },
|
|
- a summary string (short) and a usage string (longer);
|
|
- a handler which implements the command itself, invoked with
|
|
a file descriptor and argc/argv as typed by the user
|
|
- a 'generator' function which, given a partial string, can
|
|
generate legal completions for it.
|
|
An example is
|
|
|
|
int old_setdebug(int fd, int argc, char *argv[]);
|
|
char *dbg_complete(const char *line, const char *word, int pos, int n);
|
|
|
|
{ { "set", "debug", "on", NULL }, do_setdebug, "Enable debugging",
|
|
set_debug_usage, dbg_complete },
|
|
|
|
In the "new-style" format, all the above functionalities are implemented
|
|
by a single function, and the arguments tell which output is required.
|
|
The prototype is the following:
|
|
|
|
char *new_setdebug(const struct ast_cli_entry *e, int cmd, struct ast_cli_args *a);
|
|
|
|
...
|
|
// this is how we create the entry to register
|
|
AST_CLI_DEFINE(new_setdebug, "short description")
|
|
...
|
|
|
|
To help the transition, we make the pointer to the struct ast_cli_entry
|
|
available to old-style handlers via argv[-1].
|
|
|
|
An example of new-style handler is the following
|
|
|
|
\code
|
|
static char *test_new_cli(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a)
|
|
{
|
|
static const char * const choices[] = { "one", "two", "three", NULL };
|
|
|
|
switch (cmd) {
|
|
case CLI_INIT:
|
|
e->command = "do this well";
|
|
e->usage =
|
|
"Usage: do this well <arg>\n"
|
|
" typically multiline with body indented\n";
|
|
return NULL;
|
|
|
|
case CLI_GENERATE:
|
|
if (a->pos > e->args)
|
|
return NULL;
|
|
return ast_cli_complete(a->word, choices, a->n);
|
|
|
|
default:
|
|
// we are guaranteed to be called with argc >= e->args;
|
|
if (a->argc > e->args + 1) // we accept one extra argument
|
|
return CLI_SHOWUSAGE;
|
|
ast_cli(a->fd, "done this well for %s\n", e->args[argc-1]);
|
|
return CLI_SUCCESS;
|
|
}
|
|
}
|
|
|
|
\endcode
|
|
|
|
*/
|
|
|
|
/*! \brief calling arguments for new-style handlers.
|
|
* \arg \ref CLI_command_API
|
|
*/
|
|
enum ast_cli_command {
|
|
CLI_INIT = -2, /* return the usage string */
|
|
CLI_GENERATE = -3, /* behave as 'generator', remap argv to struct ast_cli_args */
|
|
CLI_HANDLER = -4, /* run the normal handler */
|
|
};
|
|
|
|
/* argument for new-style CLI handler */
|
|
struct ast_cli_args {
|
|
const int fd;
|
|
const int argc;
|
|
const char * const *argv;
|
|
const char *line; /* the current input line */
|
|
const char *word; /* the word we want to complete */
|
|
const int pos; /* position of the word to complete */
|
|
const int n; /* the iteration count (n-th entry we generate) */
|
|
};
|
|
|
|
/*! \brief descriptor for a cli entry.
|
|
* \arg \ref CLI_command_API
|
|
*/
|
|
struct ast_cli_entry {
|
|
const char * const cmda[AST_MAX_CMD_LEN]; /*!< words making up the command.
|
|
* set the first entry to NULL for a new-style entry.
|
|
*/
|
|
|
|
const char * const summary; /*!< Summary of the command (< 60 characters) */
|
|
const char * usage; /*!< Detailed usage information */
|
|
|
|
int inuse; /*!< For keeping track of usage */
|
|
struct ast_module *module; /*!< module this belongs to */
|
|
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()
|
|
*/
|
|
int args; /*!< number of non-null entries in cmda */
|
|
char *command; /*!< command, non-null for new-style entries */
|
|
char *(*handler)(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a);
|
|
/*! For linking */
|
|
AST_LIST_ENTRY(ast_cli_entry) list;
|
|
};
|
|
|
|
#if defined(__cplusplus) || defined(c_plusplus)
|
|
#define AST_CLI_DEFINE(fn, txt) { { "" }, txt, NULL, 0, NULL, NULL, 0, 0, NULL, fn }
|
|
#else
|
|
/* XXX the parser in gcc 2.95 gets confused if you don't put a space
|
|
* between the last arg before VA_ARGS and the comma */
|
|
#define AST_CLI_DEFINE(fn, txt , ... ) { .handler = fn, .summary = txt, ## __VA_ARGS__ }
|
|
#endif
|
|
|
|
/*!
|
|
* Helper function to generate cli entries from a NULL-terminated array.
|
|
* Returns the n-th matching entry from the array, or NULL if not found.
|
|
* Can be used to implement generate() for static entries as below
|
|
* (in this example we complete the word in position 2):
|
|
\code
|
|
char *my_generate(const char *line, const char *word, int pos, int n)
|
|
{
|
|
static const char * const choices[] = { "one", "two", "three", NULL };
|
|
if (pos == 2)
|
|
return ast_cli_complete(word, choices, n);
|
|
else
|
|
return NULL;
|
|
}
|
|
\endcode
|
|
*/
|
|
char *ast_cli_complete(const char *word, const char * const choices[], int pos);
|
|
|
|
/*!
|
|
* \brief Interprets a command
|
|
* Interpret a command s, sending output to fd if uid:gid has permissions
|
|
* to run this command. uid = CLI_NO_PERMS to avoid checking user permissions
|
|
* gid = CLI_NO_PERMS to avoid checking group permissions.
|
|
* \param uid User ID that is trying to run the command.
|
|
* \param gid Group ID that is trying to run the command.
|
|
* \param fd pipe
|
|
* \param s incoming string
|
|
* \retval 0 on success
|
|
* \retval -1 on failure
|
|
*/
|
|
int ast_cli_command_full(int uid, int gid, int fd, const char *s);
|
|
|
|
#define ast_cli_command(fd,s) ast_cli_command_full(CLI_NO_PERMS, CLI_NO_PERMS, fd, s)
|
|
|
|
/*!
|
|
* \brief Executes multiple CLI commands
|
|
* Interpret strings separated by NULL and execute each one, sending output to fd
|
|
* if uid has permissions, uid = CLI_NO_PERMS to avoid checking users permissions.
|
|
* gid = CLI_NO_PERMS to avoid checking group permissions.
|
|
* \param uid User ID that is trying to run the command.
|
|
* \param gid Group ID that is trying to run the command.
|
|
* \param fd pipe
|
|
* \param size is the total size of the string
|
|
* \param s incoming string
|
|
* \retval number of commands executed
|
|
*/
|
|
int ast_cli_command_multiple_full(int uid, int gid, int fd, size_t size, const char *s);
|
|
|
|
#define ast_cli_command_multiple(fd,size,s) ast_cli_command_multiple_full(CLI_NO_PERMS, CLI_NO_PERMS, fd, size, s)
|
|
|
|
/*! \brief Registers a command or an array of commands
|
|
* \param e which cli entry to register.
|
|
* Register your own command
|
|
* \retval 0 on success
|
|
* \retval -1 on failure
|
|
*/
|
|
#define ast_cli_register(e) __ast_cli_register(e, AST_MODULE_SELF)
|
|
|
|
int __ast_cli_register(struct ast_cli_entry *e, struct ast_module *mod);
|
|
|
|
/*!
|
|
* \brief Register multiple commands
|
|
* \param e pointer to first cli entry to register
|
|
* \param len number of entries to register
|
|
*/
|
|
#define ast_cli_register_multiple(e, len) \
|
|
__ast_cli_register_multiple(e, len, AST_MODULE_SELF)
|
|
|
|
int __ast_cli_register_multiple(struct ast_cli_entry *e, int len, struct ast_module *mod);
|
|
|
|
/*!
|
|
* \brief Unregisters a command or an array of commands
|
|
* \param e which cli entry to unregister
|
|
* Unregister your own command. You must pass a completed ast_cli_entry structure
|
|
* \return 0
|
|
*/
|
|
int ast_cli_unregister(struct ast_cli_entry *e);
|
|
|
|
/*!
|
|
* \brief Unregister multiple commands
|
|
* \param e pointer to first cli entry to unregister
|
|
* \param len number of entries to unregister
|
|
*/
|
|
int ast_cli_unregister_multiple(struct ast_cli_entry *e, int len);
|
|
|
|
/*!
|
|
* \brief Readline madness
|
|
* Useful for readline, that's about it
|
|
* \retval 0 on success
|
|
* \retval -1 on failure
|
|
*
|
|
* Only call this function to proxy the CLI generator to
|
|
* another.
|
|
*/
|
|
char *ast_cli_generator(const char *, const char *, int);
|
|
|
|
/*!
|
|
* \brief Generates a NULL-terminated array of strings that
|
|
* 1) begin with the string in the second parameter, and
|
|
* 2) are valid in a command after the string in the first parameter.
|
|
*
|
|
* The first entry (offset 0) of the result is the longest common substring
|
|
* in the results, useful to extend the string that has been completed.
|
|
* Subsequent entries are all possible values, followed by a NULL.
|
|
* All strings and the array itself are malloc'ed and must be freed
|
|
* by the caller.
|
|
*
|
|
* \warning This function cannot be called recursively so it will always
|
|
* fail if called from a CLI_GENERATE callback.
|
|
*/
|
|
char **ast_cli_completion_matches(const char *, const char *);
|
|
|
|
/*!
|
|
* \brief Generates a vector of strings for CLI completion.
|
|
*
|
|
* \param text Complete input being matched.
|
|
* \param word Current word being matched
|
|
*
|
|
* The results contain strings that both:
|
|
* 1) Begin with the string in \a word.
|
|
* 2) Are valid in a command after the string in \a text.
|
|
*
|
|
* The first entry (offset 0) of the result is the longest common substring
|
|
* in the results, useful to extend the string that has been completed.
|
|
* Subsequent entries are all possible values.
|
|
*
|
|
* \note All strings and the vector itself are malloc'ed and must be freed
|
|
* by the caller.
|
|
*
|
|
* \note The vector is sorted and does not contain any duplicates.
|
|
*
|
|
* \warning This function cannot be called recursively so it will always
|
|
* fail if called from a CLI_GENERATE callback.
|
|
*/
|
|
struct ast_vector_string *ast_cli_completion_vector(const char *text, const char *word);
|
|
|
|
/*!
|
|
* \brief Add a result to a request for completion options.
|
|
*
|
|
* \param value A completion option text.
|
|
*
|
|
* \retval 0 Success
|
|
* \retval -1 Failure
|
|
*
|
|
* This is an alternative to returning individual values from CLI_GENERATE. Instead
|
|
* of repeatedly being asked for the next match and having to start over, you can
|
|
* call this function repeatedly from your own stateful loop. When all matches have
|
|
* been added you can return NULL from the CLI_GENERATE function.
|
|
*
|
|
* \note This function always eventually results in calling ast_free on \a value.
|
|
*/
|
|
int ast_cli_completion_add(char *value);
|
|
|
|
/*!
|
|
* \brief Command completion for the list of active channels.
|
|
*
|
|
* This can be called from a CLI command completion function that wants to
|
|
* complete from the list of active channels. 'rpos' is the required
|
|
* position in the command. This function will return NULL immediately if
|
|
* 'rpos' is not the same as the current position, 'pos'.
|
|
*/
|
|
char *ast_complete_channels(const char *line, const char *word, int pos, int state, int rpos);
|
|
|
|
/*!
|
|
* \since 13.8
|
|
* \brief Print on cli a duration in seconds in format
|
|
* %s year(s), %s week(s), %s day(s), %s hour(s), %s second(s)
|
|
*
|
|
* \param ast_cli_args fd to print by ast_cli
|
|
* \param duration The time (in seconds) to print
|
|
* \param prefix A Prefix string to add before of duration formatted
|
|
*/
|
|
void ast_cli_print_timestr_fromseconds(int fd, int seconds, const char *prefix);
|
|
|
|
/*!
|
|
* \brief Allow a CLI command to be executed while Asterisk is shutting down.
|
|
*
|
|
* CLI commands by defeault are disabled when Asterisk is shutting down. This is
|
|
* to ensure the safety of the shutdown since CLI commands may attempt to access
|
|
* resources that have been freed as a result of the shutdown.
|
|
*
|
|
* If a CLI command should be allowed at shutdown, then the best way to enable this
|
|
* is to call ast_cli_allow_at_shutdown during the CLI_INIT state of the CLI handler.
|
|
*/
|
|
int ast_cli_allow_at_shutdown(struct ast_cli_entry *e);
|
|
|
|
#if defined(__cplusplus) || defined(c_plusplus)
|
|
}
|
|
#endif
|
|
|
|
#endif /* _ASTERISK_CLI_H */
|