aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authormarkster <markster@f38db490-d61c-443f-a65b-d21fe96a405b>2005-03-04 06:36:18 +0000
committermarkster <markster@f38db490-d61c-443f-a65b-d21fe96a405b>2005-03-04 06:36:18 +0000
commitb0180cc4f641d4097a85f52b887ac0bef130604b (patch)
tree720c9e1b1d785caa6e4e27352b5c48e2b4228502
parent484d9d791ff178d551df19b08b81d98352356ae6 (diff)
Add registerable functional variables (bug #3636, with doc mods)
git-svn-id: http://svn.digium.com/svn/asterisk/trunk@5136 f38db490-d61c-443f-a65b-d21fe96a405b
-rwxr-xr-xdoc/README.variables4
-rwxr-xr-xinclude/asterisk/pbx.h13
-rwxr-xr-xpbx.c269
3 files changed, 275 insertions, 11 deletions
diff --git a/doc/README.variables b/doc/README.variables
index 4195f299b..d923ece7b 100755
--- a/doc/README.variables
+++ b/doc/README.variables
@@ -486,3 +486,7 @@ ${CDR(userfield)} The channels uses specified field.
In addition, you can set your own extra variables with a traditional
SetVAR(CDR(var)=val) to anything you want.
+
+Certain functional variables may be accessed with $(foo <args>). A list
+of these functional variables may be found by typing "show functions"
+at the Asterisk CLI.
diff --git a/include/asterisk/pbx.h b/include/asterisk/pbx.h
index 5e05c43a5..4afbd75fb 100755
--- a/include/asterisk/pbx.h
+++ b/include/asterisk/pbx.h
@@ -53,6 +53,15 @@ typedef int (*ast_state_cb_type)(char *context, char* id, int state, void *data)
typedef int (*ast_devstate_cb_type)(const char *dev, int state, void *data);
+/*! Data structure associated with an asterisk custom function */
+struct ast_custom_function_obj {
+ char *name;
+ char *desc;
+ char *syntax;
+ char *(*function)(struct ast_channel *, char *, char *, char *, size_t);
+ struct ast_custom_function_obj *next;
+};
+
/*! Data structure associated with an asterisk switch */
struct ast_switch {
/*! NULL */
@@ -587,6 +596,10 @@ int ast_goto_if_exists(struct ast_channel *chan, char* context, char *exten, int
int ast_parseable_goto(struct ast_channel *chan, const char *goto_string);
int ast_explicit_goto(struct ast_channel *chan, const char *context, const char *exten, int priority);
int ast_async_goto_if_exists(struct ast_channel *chan, char* context, char *exten, int priority);
+struct ast_custom_function_obj* ast_custom_function_find_obj(char *name);
+int ast_custom_function_unregister(struct ast_custom_function_obj *acf);
+int ast_custom_function_register(struct ast_custom_function_obj *acf);
+
#if defined(__cplusplus) || defined(c_plusplus)
}
#endif
diff --git a/pbx.c b/pbx.c
index ffc14e5c7..c7a67e1ca 100755
--- a/pbx.c
+++ b/pbx.c
@@ -11,6 +11,8 @@
* the GNU General Public License
*/
+#include <sys/types.h>
+#include <regex.h>
#include <asterisk/lock.h>
#include <asterisk/cli.h>
#include <asterisk/pbx.h>
@@ -196,6 +198,8 @@ static struct varshead globals;
static int autofallthrough = 0;
+static struct ast_custom_function_obj *acf_root = NULL;
+
static struct pbx_builtin {
char name[AST_MAX_APP];
int (*execute)(struct ast_channel *chan, void *data);
@@ -1085,6 +1089,135 @@ icky:
}
}
+static int handle_show_functions(int fd, int argc, char *argv[])
+{
+ struct ast_custom_function_obj *acfptr;
+
+ ast_cli(fd, "Installed Custom Functions:\n--------------------------------------------------------------------------------\n");
+ for (acfptr = acf_root ; acfptr ; acfptr = acfptr->next) {
+ ast_cli(fd, "%s\t(%s)\t[%s]\n", acfptr->name, acfptr->desc, acfptr->syntax);
+ }
+ ast_cli(fd, "\n");
+ return 0;
+}
+
+struct ast_custom_function_obj* ast_custom_function_find_obj(char *name)
+{
+ struct ast_custom_function_obj *acfptr;
+
+ for (acfptr = acf_root ; acfptr ; acfptr = acfptr->next) {
+ if (!strcmp(name, acfptr->name)) {
+ break;
+ }
+ }
+
+ return acfptr;
+}
+
+int ast_custom_function_unregister(struct ast_custom_function_obj *acf)
+{
+ struct ast_custom_function_obj *acfptr, *lastacf = NULL;
+
+ if (acf) {
+ for (acfptr = acf_root ; acfptr ; acfptr = acfptr->next) {
+ if (acfptr == acf) {
+ if (lastacf) {
+ lastacf->next = acf->next;
+ } else {
+ acf_root = acf->next;
+ }
+ if (option_verbose)
+ ast_verbose(VERBOSE_PREFIX_1 "Unregistered custom function %s\n", acf->name);
+ return 0;
+ }
+ lastacf = acfptr;
+ }
+ }
+ return -1;
+}
+
+int ast_custom_function_register(struct ast_custom_function_obj *acf)
+{
+ struct ast_custom_function_obj *acfptr;
+
+ if (acf) {
+ if((acfptr = ast_custom_function_find_obj(acf->name))) {
+ ast_log(LOG_ERROR, "Function %s already in use.\n", acf->name);
+ return -1;
+ }
+ acf->next = acf_root;
+ acf_root = acf;
+ if (option_verbose)
+ ast_verbose(VERBOSE_PREFIX_1 "Registered custom function %s\n", acf->name);
+ return 0;
+ }
+
+ return -1;
+}
+
+static char *ast_func(struct ast_channel *chan, char *in, char *workspace, size_t len) {
+ char *out = NULL;
+ static char *ret = "0";
+ struct ast_custom_function_obj *acfptr;
+
+ if ((out = strchr(in, ' '))) {
+ *out = '\0';
+ out++;
+
+ if((acfptr = ast_custom_function_find_obj(in))) {
+ /* run the custom function */
+ return acfptr->function(chan, in, out, workspace, len);
+ }
+ }
+ return strdup(ret);
+}
+
+static char *builtin_function_isnull(struct ast_channel *chan, char *cmd, char *data, char *buf, size_t len)
+{
+ char *ret_true = "1", *ret_false = "0";
+ return cmd ? ret_false : ret_true;
+}
+
+static char *builtin_function_exists(struct ast_channel *chan, char *cmd, char *data, char *buf, size_t len)
+{
+ char *ret_true = "1", *ret_false = "0";
+ return cmd ? ret_true : ret_false;
+}
+
+static char *builtin_function_regex(struct ast_channel *chan, char *cmd, char *data, char *buf, size_t len)
+{
+ char *ret_true = "1", *ret_false = "0", *ret;
+ char *arg, *tmp;
+ regex_t regexbuf;
+
+ ret = ret_false; /* convince me otherwise */
+ if ((tmp = ast_strdupa(data)) && (arg = strchr(tmp, '"')) && (data = strchr(arg+1, '"'))) {
+ arg++;
+ *data = '\0';
+ data++;
+
+ if(data[0] == ' ') {
+ data++;
+ } else {
+ ast_log(LOG_WARNING, "Malformed input missing space in %s\n", cmd);
+ ret = ret_false;
+ }
+
+ if (regcomp(&regexbuf, arg, REG_EXTENDED | REG_NOSUB)) {
+ ast_log(LOG_WARNING, "Malformed regex input %s\n", cmd);
+ ret = NULL;
+ }
+ ret = regexec(&regexbuf, data, 0, NULL, 0) ? ret_false : ret_true;
+ regfree(&regexbuf);
+
+
+ } else {
+ ast_log(LOG_WARNING, "Malformed input %s\n", cmd);
+ }
+
+ return ret;
+}
+
static void pbx_substitute_variables_helper_full(struct ast_channel *c, const char *cp1, char *cp2, int count, struct varshead *headp)
{
char *cp4;
@@ -1092,9 +1225,9 @@ static void pbx_substitute_variables_helper_full(struct ast_channel *c, const ch
int length;
char workspace[4096];
char ltmp[4096], var[4096];
- char *nextvar, *nextexp;
+ char *nextvar, *nextexp, *nextfunc;
char *vars, *vare;
- int pos, brackets, needsub, len;
+ int pos, brackets, needsub, len, needfunc;
/* Substitutes variables into cp2, based on string cp1, and assuming cp2 to be
zero-filled */
@@ -1108,13 +1241,27 @@ static void pbx_substitute_variables_helper_full(struct ast_channel *c, const ch
/* Look for an expression */
nextexp = strstr(whereweare, "$[");
+
+ /* Look for a function */
+ nextfunc = strstr(whereweare, "$(");
/* Pick the first one only */
- if (nextvar && nextexp) {
- if (nextvar < nextexp)
- nextexp = NULL;
- else
- nextvar = NULL;
+ len = 0;
+ if (nextvar)
+ len++;
+ if(nextexp)
+ len++;
+ if(nextfunc)
+ len++;
+
+ if (len > 1) {
+ if(nextfunc) {
+ nextvar = nextexp = NULL;
+ } else if (nextvar) {
+ nextexp = nextfunc = NULL;
+ } else if (nextexp) {
+ nextvar = nextfunc = NULL;
+ }
}
/* If there is one, we only go that far */
@@ -1122,7 +1269,9 @@ static void pbx_substitute_variables_helper_full(struct ast_channel *c, const ch
pos = nextvar - whereweare;
else if (nextexp)
pos = nextexp - whereweare;
-
+ else if (nextfunc) {
+ pos = nextfunc - whereweare;
+ }
/* Can't copy more than 'count' bytes */
if (pos > count)
pos = count;
@@ -1193,7 +1342,8 @@ static void pbx_substitute_variables_helper_full(struct ast_channel *c, const ch
vars = vare = nextexp + 2;
brackets = 1;
needsub = 0;
-
+ needfunc = 0;
+
/* Find the end of it */
while(brackets && *vare) {
if ((vare[0] == '$') && (vare[1] == '[')) {
@@ -1207,6 +1357,9 @@ static void pbx_substitute_variables_helper_full(struct ast_channel *c, const ch
} else if ((vare[0] == '$') && (vare[1] == '{')) {
needsub++;
vare++;
+ } else if ((vare[0] == '$') && (vare[1] == '(')) {
+ needfunc++;
+ vare++;
}
vare++;
}
@@ -1223,7 +1376,7 @@ static void pbx_substitute_variables_helper_full(struct ast_channel *c, const ch
var[len] = '\0';
/* Substitute if necessary */
- if (needsub) {
+ if (needsub || needfunc) {
memset(ltmp, 0, sizeof(ltmp));
pbx_substitute_variables_helper(c, var, ltmp, sizeof(ltmp) - 1);
vars = ltmp;
@@ -1245,6 +1398,63 @@ static void pbx_substitute_variables_helper_full(struct ast_channel *c, const ch
cp2 += length;
free(cp4);
}
+ } else if (nextfunc) {
+ /* We have a function. Find the start and end */
+ vars = vare = nextfunc + 2;
+ brackets = 1;
+ needsub = 0;
+
+ /* Find the end of it */
+ while(brackets && *vare) {
+ if ((vare[0] == '$') && (vare[1] == '(')) {
+ needsub++;
+ brackets++;
+ vare++;
+ } else if (vare[0] == '(') {
+ brackets++;
+ } else if (vare[0] == ')') {
+ brackets--;
+ } else if ((vare[0] == '$') && (vare[1] == '{')) {
+ needsub++;
+ vare++;
+ }
+ vare++;
+ }
+ if (brackets)
+ ast_log(LOG_NOTICE, "Error in extension logic (missing ')')\n");
+ len = vare - vars - 1;
+
+ /* Skip totally over variable name */
+ whereweare += ( len + 3);
+
+ /* Store variable name (and truncate) */
+ memset(var, 0, sizeof(var));
+ strncpy(var, vars, sizeof(var) - 1);
+ var[len] = '\0';
+
+ /* Substitute if necessary */
+ if (needsub) {
+ memset(ltmp, 0, sizeof(ltmp));
+ pbx_substitute_variables_helper(c, var, ltmp, sizeof(ltmp) - 1);
+ vars = ltmp;
+ } else {
+ vars = var;
+ }
+
+ /* Evaluate expression */
+ cp4 = ast_func(c, vars, workspace, sizeof(workspace));
+
+ ast_log(LOG_DEBUG, "Expression is '%s'\n", cp4);
+
+ if (cp4) {
+ length = strlen(cp4);
+ if (length > count)
+ length = count;
+ memcpy(cp2, cp4, length);
+ count -= length;
+ cp2 += length;
+ free(cp4);
+ }
} else
break;
@@ -1266,7 +1476,7 @@ static void pbx_substitute_variables(char *passdata, int datalen, struct ast_cha
memset(passdata, 0, datalen);
/* No variables or expressions in e->data, so why scan it? */
- if (!strstr(e->data,"${") && !strstr(e->data,"$[")) {
+ if (!strstr(e->data,"${") && !strstr(e->data,"$[") && !strstr(e->data,"$(")) {
strncpy(passdata, e->data, datalen - 1);
passdata[datalen-1] = '\0';
return;
@@ -2593,6 +2803,10 @@ static char show_application_help[] =
"Usage: show application <application> [<application> [<application> [...]]]\n"
" Describes a particular application.\n";
+static char show_functions_help[] =
+"Usage: show functions\n"
+" List builtin functions accessable as $(function args)";
+
static char show_applications_help[] =
"Usage: show applications [{like|describing} <text>]\n"
" List applications which are currently available.\n"
@@ -3169,6 +3383,30 @@ static int handle_show_dialplan(int fd, int argc, char *argv[])
return RESULT_SUCCESS;
}
+
+/* custom commands */
+
+static struct ast_custom_function_obj regex_function = {
+ .name = "regex",
+ .desc = "Regular Expression: Returns 1 if data matches regular expression.",
+ .syntax = "$(regex \"<regular expression>\" <data>)",
+ .function = builtin_function_regex,
+};
+
+static struct ast_custom_function_obj isnull_function = {
+ .name = "isnull",
+ .desc = "NULL Test: Returns 1 if NULL or 0 otherwise",
+ .syntax = "$(isnull <data>)",
+ .function = builtin_function_isnull,
+};
+
+static struct ast_custom_function_obj exists_function = {
+ .name = "exists",
+ .desc = "Existance Test: Returns 1 if exists, 0 otherwise",
+ .syntax = "$(exists <data>)",
+ .function = builtin_function_exists,
+};
+
/*
* CLI entries for upper commands ...
*/
@@ -3177,6 +3415,11 @@ static struct ast_cli_entry show_applications_cli =
handle_show_applications, "Shows registered applications",
show_applications_help, complete_show_applications };
+static struct ast_cli_entry show_functions_cli =
+ { { "show", "functions", NULL },
+ handle_show_functions, "Shows registered functions",
+ show_functions_help};
+
static struct ast_cli_entry show_application_cli =
{ { "show", "application", NULL },
handle_show_application, "Describe a specific application",
@@ -5530,10 +5773,14 @@ int load_pbx(void)
}
AST_LIST_HEAD_INIT(&globals);
ast_cli_register(&show_applications_cli);
+ ast_cli_register(&show_functions_cli);
ast_cli_register(&show_application_cli);
ast_cli_register(&show_dialplan_cli);
ast_cli_register(&show_switches_cli);
ast_cli_register(&show_hints_cli);
+ ast_custom_function_register(&regex_function);
+ ast_custom_function_register(&isnull_function);
+ ast_custom_function_register(&exists_function);
/* Register builtin applications */
for (x=0; x<sizeof(builtins) / sizeof(struct pbx_builtin); x++) {