mirror of
https://github.com/asterisk/asterisk.git
synced 2025-10-24 05:38:11 +00:00
func_odbc: Introduce minargs config and expose ARGC in addition to ARGn.
minargs enables enforcing of minimum count of arguments to pass to func_odbc, so if you're unconditionally using ARG1 through ARG4 then this should be set to 4. func_odbc will generate an error in this case, so for example [FOO] minargs = 4 and ODBC_FOO(a,b,c) in dialplan will now error out instead of using a potentially leaked ARG4 from Gosub(). ARGC is needed if you're using optional argument, to verify whether or not an argument has been passed, else it's possible to use a leaked ARGn from Gosub (app_stack). So now you can safely do ${IF($[${ARGC}>3]?${ARGV}:default value)} kind of thing. Change-Id: I6ca0b137d90b03f6aa9c496991f6cbf1518f6c24 Signed-off-by: Jaco Kroon <jaco@uls.co.za>
This commit is contained in:
committed by
George Joseph
parent
6e695c867f
commit
b0f349a330
@@ -23,6 +23,10 @@
|
|||||||
; For substitution, you have ${ARG1}, ${ARG2} ... ${ARGn}
|
; For substitution, you have ${ARG1}, ${ARG2} ... ${ARGn}
|
||||||
; for the arguments to each SQL statement.
|
; for the arguments to each SQL statement.
|
||||||
;
|
;
|
||||||
|
; Additionally you can use ${ARGC} to determine the number of arguments that
|
||||||
|
; was actually passed (or risk using leaked ARGn variables from the channel).
|
||||||
|
; Also reference the minargs configuration option.
|
||||||
|
;
|
||||||
; In addition, for write statements, you have ${VAL1}, ${VAL2} ... ${VALn}
|
; In addition, for write statements, you have ${VAL1}, ${VAL2} ... ${VALn}
|
||||||
; parsed, just like arguments, for the values. In addition, if you want the
|
; parsed, just like arguments, for the values. In addition, if you want the
|
||||||
; whole value, never mind the parsing, you can get that with ${VALUE}.
|
; whole value, never mind the parsing, you can get that with ${VALUE}.
|
||||||
@@ -87,6 +91,13 @@
|
|||||||
; These additional rows can be returned by using the name of the
|
; These additional rows can be returned by using the name of the
|
||||||
; function which was called to retrieve the first row as an
|
; function which was called to retrieve the first row as an
|
||||||
; argument to ODBC_FETCH().
|
; argument to ODBC_FETCH().
|
||||||
|
; minargs The minimum number of ARGUMENTS that has to be passed to the
|
||||||
|
; function. If fewer arguments than this is passed, then the call
|
||||||
|
; will fail. It is important to note that unlike Gosub() and friends,
|
||||||
|
; func_odbc will not mask out ARGn variables that it's not actively
|
||||||
|
; using, as such, without this, it's entirely possible to use say
|
||||||
|
; ARG2 from the Gosub() inside func_odbc when the intent was to
|
||||||
|
; use an argument passed to func_odbc, but it simply was never passed.
|
||||||
|
|
||||||
|
|
||||||
; ODBC_SQL - Allow an SQL statement to be built entirely in the dialplan
|
; ODBC_SQL - Allow an SQL statement to be built entirely in the dialplan
|
||||||
|
20
doc/CHANGES-staging/func_odbc_ARGC_minargs.txt
Normal file
20
doc/CHANGES-staging/func_odbc_ARGC_minargs.txt
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
Subject: func_odbc
|
||||||
|
|
||||||
|
Introduce an ARGC variable for func_odbc functions, along with a minargs
|
||||||
|
per-function configuration option.
|
||||||
|
|
||||||
|
minargs enables enforcing of minimum count of arguments to pass to
|
||||||
|
func_odbc, so if you're unconditionally using ARG1 through ARG4 then
|
||||||
|
this should be set to 4. func_odbc will generate an error in this case,
|
||||||
|
so for example
|
||||||
|
|
||||||
|
[FOO]
|
||||||
|
minargs = 4
|
||||||
|
|
||||||
|
and ODBC_FOO(a,b,c) in dialplan will now error out instead of using a
|
||||||
|
potentially leaked ARG4 from Gosub().
|
||||||
|
|
||||||
|
ARGC is needed if you're using optional argument, to verify whether or
|
||||||
|
not an argument has been passed, else it's possible to use a leaked ARGn
|
||||||
|
from Gosub (app_stack). So now you can safely do
|
||||||
|
${IF($[${ARGC}>3]?${ARGV}:default value)} kind of thing.
|
@@ -120,6 +120,7 @@ struct acf_odbc_query {
|
|||||||
char *sql_insert;
|
char *sql_insert;
|
||||||
unsigned int flags;
|
unsigned int flags;
|
||||||
int rowlimit;
|
int rowlimit;
|
||||||
|
int minargs;
|
||||||
struct ast_custom_function *acf;
|
struct ast_custom_function *acf;
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -545,6 +546,14 @@ static int acf_odbc_write(struct ast_channel *chan, const char *cmd, char *s, co
|
|||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
AST_STANDARD_APP_ARGS(args, s);
|
||||||
|
if (args.argc < query->minargs) {
|
||||||
|
ast_log(LOG_ERROR, "%d arguments supplied to '%s' requiring minimum %d\n",
|
||||||
|
args.argc, cmd, query->minargs);
|
||||||
|
AST_RWLIST_UNLOCK(&queries);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
if (!chan) {
|
if (!chan) {
|
||||||
if (!(chan = ast_dummy_channel_alloc())) {
|
if (!(chan = ast_dummy_channel_alloc())) {
|
||||||
AST_RWLIST_UNLOCK(&queries);
|
AST_RWLIST_UNLOCK(&queries);
|
||||||
@@ -578,7 +587,8 @@ static int acf_odbc_write(struct ast_channel *chan, const char *cmd, char *s, co
|
|||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
AST_STANDARD_APP_ARGS(args, s);
|
snprintf(varname, sizeof(varname), "%u", args.argc);
|
||||||
|
pbx_builtin_pushvar_helper(chan, "ARGC", varname);
|
||||||
for (i = 0; i < args.argc; i++) {
|
for (i = 0; i < args.argc; i++) {
|
||||||
snprintf(varname, sizeof(varname), "ARG%d", i + 1);
|
snprintf(varname, sizeof(varname), "ARG%d", i + 1);
|
||||||
pbx_builtin_pushvar_helper(chan, varname, args.field[i]);
|
pbx_builtin_pushvar_helper(chan, varname, args.field[i]);
|
||||||
@@ -603,6 +613,8 @@ static int acf_odbc_write(struct ast_channel *chan, const char *cmd, char *s, co
|
|||||||
chan = ast_channel_unref(chan);
|
chan = ast_channel_unref(chan);
|
||||||
} else {
|
} else {
|
||||||
/* Restore prior values */
|
/* Restore prior values */
|
||||||
|
pbx_builtin_setvar_helper(chan, "ARGC", NULL);
|
||||||
|
|
||||||
for (i = 0; i < args.argc; i++) {
|
for (i = 0; i < args.argc; i++) {
|
||||||
snprintf(varname, sizeof(varname), "ARG%d", i + 1);
|
snprintf(varname, sizeof(varname), "ARG%d", i + 1);
|
||||||
pbx_builtin_setvar_helper(chan, varname, NULL);
|
pbx_builtin_setvar_helper(chan, varname, NULL);
|
||||||
@@ -756,6 +768,14 @@ static int acf_odbc_read(struct ast_channel *chan, const char *cmd, char *s, cha
|
|||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
AST_STANDARD_APP_ARGS(args, s);
|
||||||
|
if (args.argc < query->minargs) {
|
||||||
|
ast_log(LOG_ERROR, "%d arguments supplied to '%s' requiring minimum %d\n",
|
||||||
|
args.argc, cmd, query->minargs);
|
||||||
|
AST_RWLIST_UNLOCK(&queries);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
if (!chan) {
|
if (!chan) {
|
||||||
if (!(chan = ast_dummy_channel_alloc())) {
|
if (!(chan = ast_dummy_channel_alloc())) {
|
||||||
AST_RWLIST_UNLOCK(&queries);
|
AST_RWLIST_UNLOCK(&queries);
|
||||||
@@ -768,7 +788,8 @@ static int acf_odbc_read(struct ast_channel *chan, const char *cmd, char *s, cha
|
|||||||
ast_autoservice_start(chan);
|
ast_autoservice_start(chan);
|
||||||
}
|
}
|
||||||
|
|
||||||
AST_STANDARD_APP_ARGS(args, s);
|
snprintf(varname, sizeof(varname), "%u", args.argc);
|
||||||
|
pbx_builtin_pushvar_helper(chan, "ARGC", varname);
|
||||||
for (x = 0; x < args.argc; x++) {
|
for (x = 0; x < args.argc; x++) {
|
||||||
snprintf(varname, sizeof(varname), "ARG%d", x + 1);
|
snprintf(varname, sizeof(varname), "ARG%d", x + 1);
|
||||||
pbx_builtin_pushvar_helper(chan, varname, args.field[x]);
|
pbx_builtin_pushvar_helper(chan, varname, args.field[x]);
|
||||||
@@ -780,6 +801,8 @@ static int acf_odbc_read(struct ast_channel *chan, const char *cmd, char *s, cha
|
|||||||
chan = ast_channel_unref(chan);
|
chan = ast_channel_unref(chan);
|
||||||
} else {
|
} else {
|
||||||
/* Restore prior values */
|
/* Restore prior values */
|
||||||
|
pbx_builtin_setvar_helper(chan, "ARGC", NULL);
|
||||||
|
|
||||||
for (x = 0; x < args.argc; x++) {
|
for (x = 0; x < args.argc; x++) {
|
||||||
snprintf(varname, sizeof(varname), "ARG%d", x + 1);
|
snprintf(varname, sizeof(varname), "ARG%d", x + 1);
|
||||||
pbx_builtin_setvar_helper(chan, varname, NULL);
|
pbx_builtin_setvar_helper(chan, varname, NULL);
|
||||||
@@ -1290,6 +1313,10 @@ static int init_acf_query(struct ast_config *cfg, char *catg, struct acf_odbc_qu
|
|||||||
sscanf(tmp, "%30d", &((*query)->rowlimit));
|
sscanf(tmp, "%30d", &((*query)->rowlimit));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if ((tmp = ast_variable_retrieve(cfg, catg, "minargs"))) {
|
||||||
|
sscanf(tmp, "%30d", &((*query)->minargs));
|
||||||
|
}
|
||||||
|
|
||||||
(*query)->acf = ast_calloc(1, sizeof(struct ast_custom_function));
|
(*query)->acf = ast_calloc(1, sizeof(struct ast_custom_function));
|
||||||
if (!(*query)->acf) {
|
if (!(*query)->acf) {
|
||||||
free_acf_query(*query);
|
free_acf_query(*query);
|
||||||
|
Reference in New Issue
Block a user