mirror of
https://github.com/asterisk/asterisk.git
synced 2025-10-21 12:30:41 +00:00
Bug 6886 - Add application TryExec, which does mostly the same thing, but returns differently
git-svn-id: https://origsvn.digium.com/svn/asterisk/trunk@17454 65c4cc65-6c06-0410-ace0-fbb531ad65f3
This commit is contained in:
@@ -2,8 +2,9 @@
|
|||||||
* Asterisk -- An open source telephony toolkit.
|
* Asterisk -- An open source telephony toolkit.
|
||||||
*
|
*
|
||||||
* Copyright (c) 2004 - 2005, Tilghman Lesher. All rights reserved.
|
* Copyright (c) 2004 - 2005, Tilghman Lesher. All rights reserved.
|
||||||
|
* Portions copyright (c) 2006, Philipp Dunkel.
|
||||||
*
|
*
|
||||||
* Tilghman Lesher <app_exec__v001@the-tilghman.com>
|
* Tilghman Lesher <app_exec__v002@the-tilghman.com>
|
||||||
*
|
*
|
||||||
* This code is released by the author with no restrictions on usage.
|
* This code is released by the author with no restrictions on usage.
|
||||||
*
|
*
|
||||||
@@ -19,7 +20,8 @@
|
|||||||
*
|
*
|
||||||
* \brief Exec application
|
* \brief Exec application
|
||||||
*
|
*
|
||||||
* \author Tilghman Lesher <app_exec__v001@the-tilghman.com>
|
* \author Tilghman Lesher <app_exec__v002@the-tilghman.com>
|
||||||
|
* \author Philipp Dunkel <philipp.dunkel@ebox.at>
|
||||||
*
|
*
|
||||||
* \ingroup applications
|
* \ingroup applications
|
||||||
*/
|
*/
|
||||||
@@ -43,18 +45,43 @@ ASTERISK_FILE_VERSION(__FILE__, "$Revision$")
|
|||||||
/* Maximum length of any variable */
|
/* Maximum length of any variable */
|
||||||
#define MAXRESULT 1024
|
#define MAXRESULT 1024
|
||||||
|
|
||||||
static char *tdesc = "Executes applications";
|
static char *tdesc = "Executes dialplan applications";
|
||||||
|
|
||||||
|
/*! Note
|
||||||
|
*
|
||||||
|
* The key difference between these two apps is exit status. In a
|
||||||
|
* nutshell, Exec tries to be transparent as possible, behaving
|
||||||
|
* in exactly the same way as if the application it calls was
|
||||||
|
* directly invoked from the dialplan.
|
||||||
|
*
|
||||||
|
* TryExec, on the other hand, provides a way to execute applications
|
||||||
|
* and catch any possible fatal error without actually fatally
|
||||||
|
* affecting the dialplan.
|
||||||
|
*/
|
||||||
|
|
||||||
static char *app_exec = "Exec";
|
static char *app_exec = "Exec";
|
||||||
|
static char *exec_synopsis = "Executes dialplan application";
|
||||||
static char *exec_synopsis = "Executes internal application";
|
|
||||||
|
|
||||||
static char *exec_descrip =
|
static char *exec_descrip =
|
||||||
"Usage: Exec(appname(arguments))\n"
|
"Usage: Exec(appname(arguments))\n"
|
||||||
" Allows an arbitrary application to be invoked even when not\n"
|
" Allows an arbitrary application to be invoked even when not\n"
|
||||||
|
"hardcoded into the dialplan. If the underlying application\n"
|
||||||
|
"terminates the dialplan, or if the application cannot be found,\n"
|
||||||
|
"Exec will terminate the dialplan.\n"
|
||||||
|
" To invoke external applications, see the application System.\n"
|
||||||
|
" If you would like to catch any error instead, see TryExec.\n";
|
||||||
|
|
||||||
|
static char *app_tryexec = "TryExec";
|
||||||
|
static char *tryexec_synopsis = "Executes dialplan application, always returning";
|
||||||
|
static char *tryexec_descrip =
|
||||||
|
"Usage: TryExec(appname(arguments))\n"
|
||||||
|
" Allows an arbitrary application to be invoked even when not\n"
|
||||||
"hardcoded into the dialplan. To invoke external applications\n"
|
"hardcoded into the dialplan. To invoke external applications\n"
|
||||||
"see the application System. Returns whatever value the\n"
|
"see the application System. Always returns to the dialplan.\n"
|
||||||
"app returns or a non-zero value if the app cannot be found.\n";
|
"The channel variable TRYSTATUS will be set to:\n"
|
||||||
|
" SUCCESS if the application returned zero\n"
|
||||||
|
" FAILED if the application returned non-zero\n"
|
||||||
|
" NOAPP if the application was not found or was not specified\n"
|
||||||
|
" NOMEMORY if there was not enough memory to execute.\n";
|
||||||
|
|
||||||
LOCAL_USER_DECL;
|
LOCAL_USER_DECL;
|
||||||
|
|
||||||
@@ -62,13 +89,11 @@ static int exec_exec(struct ast_channel *chan, void *data)
|
|||||||
{
|
{
|
||||||
int res=0;
|
int res=0;
|
||||||
struct localuser *u;
|
struct localuser *u;
|
||||||
char *s, *appname, *endargs, args[MAXRESULT];
|
char *s, *appname, *endargs, args[MAXRESULT] = "";
|
||||||
struct ast_app *app;
|
struct ast_app *app;
|
||||||
|
|
||||||
LOCAL_USER_ADD(u);
|
LOCAL_USER_ADD(u);
|
||||||
|
|
||||||
memset(args, 0, MAXRESULT);
|
|
||||||
|
|
||||||
/* Check and parse arguments */
|
/* Check and parse arguments */
|
||||||
if (data) {
|
if (data) {
|
||||||
if ((s = ast_strdupa(data))) {
|
if ((s = ast_strdupa(data))) {
|
||||||
@@ -96,11 +121,51 @@ static int exec_exec(struct ast_channel *chan, void *data)
|
|||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int tryexec_exec(struct ast_channel *chan, void *data)
|
||||||
|
{
|
||||||
|
int res=0;
|
||||||
|
struct localuser *u;
|
||||||
|
char *s, *appname, *endargs, args[MAXRESULT] = "";
|
||||||
|
struct ast_app *app;
|
||||||
|
|
||||||
|
LOCAL_USER_ADD(u);
|
||||||
|
|
||||||
|
/* Check and parse arguments */
|
||||||
|
if (data) {
|
||||||
|
if ((s = ast_strdupa(data))) {
|
||||||
|
appname = strsep(&s, "(");
|
||||||
|
if (s) {
|
||||||
|
endargs = strrchr(s, ')');
|
||||||
|
if (endargs)
|
||||||
|
*endargs = '\0';
|
||||||
|
pbx_substitute_variables_helper(chan, s, args, MAXRESULT - 1);
|
||||||
|
}
|
||||||
|
if (appname) {
|
||||||
|
app = pbx_findapp(appname);
|
||||||
|
if (app) {
|
||||||
|
res = pbx_exec(chan, app, args);
|
||||||
|
pbx_builtin_setvar_helper(chan, "TRYSTATUS", res ? "FAILED" : "SUCCESS");
|
||||||
|
} else {
|
||||||
|
ast_log(LOG_WARNING, "Could not find application (%s)\n", appname);
|
||||||
|
pbx_builtin_setvar_helper(chan, "TRYSTATUS", "NOAPP");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
ast_log(LOG_ERROR, "Out of memory\n");
|
||||||
|
pbx_builtin_setvar_helper(chan, "TRYSTATUS", "NOMEMORY");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
LOCAL_USER_REMOVE(u);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
int unload_module(void)
|
int unload_module(void)
|
||||||
{
|
{
|
||||||
int res;
|
int res;
|
||||||
|
|
||||||
res = ast_unregister_application(app_exec);
|
res = ast_unregister_application(app_exec);
|
||||||
|
res |= ast_unregister_application(app_tryexec);
|
||||||
|
|
||||||
STANDARD_HANGUP_LOCALUSERS;
|
STANDARD_HANGUP_LOCALUSERS;
|
||||||
|
|
||||||
@@ -109,7 +174,9 @@ int unload_module(void)
|
|||||||
|
|
||||||
int load_module(void)
|
int load_module(void)
|
||||||
{
|
{
|
||||||
return ast_register_application(app_exec, exec_exec, exec_synopsis, exec_descrip);
|
int res = ast_register_application(app_exec, exec_exec, exec_synopsis, exec_descrip);
|
||||||
|
res |= ast_register_application(app_tryexec, tryexec_exec, tryexec_synopsis, tryexec_descrip);
|
||||||
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
char *description(void)
|
char *description(void)
|
||||||
|
Reference in New Issue
Block a user