[Asterisk-code-review] pbx variables.c: Misc fixes in variable substitution. (asterisk[15])
Richard Mudgett
asteriskteam at digium.com
Mon Jan 22 12:38:03 CST 2018
Richard Mudgett has uploaded this change for review. ( https://gerrit.asterisk.org/8039
Change subject: pbx_variables.c: Misc fixes in variable substitution.
......................................................................
pbx_variables.c: Misc fixes in variable substitution.
* Copy more than one character at a time when there is nothing to
substitute.
* Fix off by one error if a '}' or ']' is missing.
* Eliminated the requirement that the "used" parameter had to point to a
variable. The current callers were always declaring a variable to meet
the requirement and discarding the value put into that variable. Now it
can be NULL.
* In ast_str_substitute_variables_full() fixed using the bogus channel to
evaluate a function. We were not using the bogus channel we just created
to help evaluate a subexpression.
Change-Id: Ia83d99f4f16abe47f329eb39b6ff2013ae7c9854
---
M include/asterisk/pbx.h
M main/pbx_variables.c
2 files changed, 157 insertions(+), 90 deletions(-)
git pull ssh://gerrit.asterisk.org:29418/asterisk refs/changes/39/8039/1
diff --git a/include/asterisk/pbx.h b/include/asterisk/pbx.h
index c8c171a..a40c6a4 100644
--- a/include/asterisk/pbx.h
+++ b/include/asterisk/pbx.h
@@ -1432,7 +1432,7 @@
* \param c Channel variables from which to extract values, and channel to pass to any dialplan functions.
* \param headp If no channel is specified, a channel list from which to extract variable values
* \param templ Variable template to expand.
- * \param used Number of bytes read from the template.
+ * \param used Number of bytes read from the template. (May be NULL)
*/
void ast_str_substitute_variables_full(struct ast_str **buf, ssize_t maxlen, struct ast_channel *c, struct varshead *headp, const char *templ, size_t *used);
/*! @} */
diff --git a/main/pbx_variables.c b/main/pbx_variables.c
index eede213..1dc0fc0 100644
--- a/main/pbx_variables.c
+++ b/main/pbx_variables.c
@@ -396,51 +396,74 @@
void ast_str_substitute_variables_full(struct ast_str **buf, ssize_t maxlen, struct ast_channel *c, struct varshead *headp, const char *templ, size_t *used)
{
/* Substitutes variables into buf, based on string templ */
- char *cp4 = NULL;
const char *whereweare;
- int orig_size = 0;
- int offset, offset2, isfunction;
- const char *nextvar, *nextexp, *nextthing;
- const char *vars, *vare;
- char *finalvars;
- int pos, brackets, needsub, len;
- struct ast_str *substr1 = ast_str_create(16), *substr2 = NULL, *substr3 = ast_str_create(16);
+ struct ast_str *substr1 = ast_str_create(16);
+ struct ast_str *substr2 = NULL;
+ struct ast_str *substr3 = ast_str_create(16);
ast_str_reset(*buf);
+
+ if (!substr1 || !substr3) {
+ if (used) {
+ *used = ast_str_strlen(*buf);
+ }
+ ast_free(substr1);
+ ast_free(substr3);
+ return;
+ }
+
whereweare = templ;
while (!ast_strlen_zero(whereweare)) {
+ const char *nextvar = NULL;
+ const char *nextexp = NULL;
+ const char *nextthing;
+ const char *vars;
+ const char *vare;
+ char *finalvars;
+ int pos;
+ int brackets;
+ int needsub;
+ int len;
+
/* reset our buffer */
ast_str_reset(substr3);
- /* Assume we're copying the whole remaining string */
- pos = strlen(whereweare);
- nextvar = NULL;
- nextexp = NULL;
+ /* Determine how much simply needs to be copied to the output buf. */
nextthing = strchr(whereweare, '$');
if (nextthing) {
+ pos = nextthing - whereweare;
switch (nextthing[1]) {
case '{':
+ /* Variable substitution */
nextvar = nextthing;
- pos = nextvar - whereweare;
break;
case '[':
+ /* Expression substitution */
nextexp = nextthing;
- pos = nextexp - whereweare;
break;
default:
- pos = 1;
+ /* '$' is not part of a substitution so include it too. */
+ ++pos;
+ break;
}
+ } else {
+ /* We're copying the whole remaining string */
+ pos = strlen(whereweare);
}
if (pos) {
/* Copy that many bytes */
ast_str_append_substr(buf, maxlen, whereweare, pos);
- templ += pos;
whereweare += pos;
}
if (nextvar) {
+ int offset;
+ int offset2;
+ int isfunction;
+ int res;
+
/* We have a variable. Find the start and end, and determine
if we are going to have to recursively call ourselves on the
contents */
@@ -452,33 +475,42 @@
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] == '['))
+ } else if ((vare[0] == '$') && (vare[1] == '[')) {
needsub++;
+ vare++;
+ }
vare++;
}
- if (brackets)
+ len = vare - vars;
+ if (brackets) {
ast_log(LOG_WARNING, "Error in extension logic (missing '}')\n");
- len = vare - vars - 1;
+ } else {
+ /* Don't count the closing '}' in the length. */
+ --len;
+ }
/* Skip totally over variable string */
- whereweare += (len + 3);
+ whereweare = vare;
- /* Store variable name (and truncate) */
+ /* Store variable name expression to lookup. */
ast_str_set_substr(&substr1, 0, vars, len);
ast_debug(5, "Evaluating '%s' (from '%s' len %d)\n", ast_str_buffer(substr1), vars, len);
/* Substitute if necessary */
if (needsub) {
- size_t my_used;
-
if (!substr2) {
substr2 = ast_str_create(16);
+ if (!substr2) {
+ continue;
+ }
}
- ast_str_substitute_variables_full(&substr2, 0, c, headp, ast_str_buffer(substr1), &my_used);
+ ast_str_substitute_variables_full(&substr2, 0, c, headp, ast_str_buffer(substr1), NULL);
finalvars = ast_str_buffer(substr2);
} else {
finalvars = ast_str_buffer(substr1);
@@ -488,28 +520,32 @@
if (isfunction) {
/* Evaluate function */
if (c || !headp) {
- cp4 = ast_func_read2(c, finalvars, &substr3, 0) ? NULL : ast_str_buffer(substr3);
+ res = ast_func_read2(c, finalvars, &substr3, 0);
} else {
struct varshead old;
- struct ast_channel *bogus = ast_dummy_channel_alloc();
+ struct ast_channel *bogus;
+
+ bogus = ast_dummy_channel_alloc();
if (bogus) {
- memcpy(&old, ast_channel_varshead(bogus), sizeof(old));
- memcpy(ast_channel_varshead(bogus), headp, sizeof(*ast_channel_varshead(bogus)));
- cp4 = ast_func_read2(c, finalvars, &substr3, 0) ? NULL : ast_str_buffer(substr3);
+ old = *ast_channel_varshead(bogus);
+ *ast_channel_varshead(bogus) = *headp;
+ res = ast_func_read2(bogus, finalvars, &substr3, 0);
/* Don't deallocate the varshead that was passed in */
- memcpy(ast_channel_varshead(bogus), &old, sizeof(*ast_channel_varshead(bogus)));
+ *ast_channel_varshead(bogus) = old;
ast_channel_unref(bogus);
} else {
- ast_log(LOG_ERROR, "Unable to allocate bogus channel for variable substitution. Function results may be blank.\n");
+ ast_log(LOG_ERROR, "Unable to allocate bogus channel for function value substitution.\n");
+ res = -1;
}
}
- ast_debug(2, "Function %s result is '%s'\n", finalvars, cp4 ? cp4 : "(null)");
+ ast_debug(2, "Function %s result is '%s'\n",
+ finalvars, res ? "" : ast_str_buffer(substr3));
} else {
/* Retrieve variable value */
ast_str_retrieve_variable(&substr3, 0, c, headp, finalvars);
- cp4 = ast_str_buffer(substr3);
+ res = 0;
}
- if (cp4) {
+ if (!res) {
ast_str_substring(substr3, offset, offset2);
ast_str_append(buf, maxlen, "%s", ast_str_buffer(substr3));
}
@@ -537,24 +573,29 @@
}
vare++;
}
- if (brackets)
+ len = vare - vars;
+ if (brackets) {
ast_log(LOG_WARNING, "Error in extension logic (missing ']')\n");
- len = vare - vars - 1;
+ } else {
+ /* Don't count the closing ']' in the length. */
+ --len;
+ }
/* Skip totally over expression */
- whereweare += (len + 3);
+ whereweare = vare;
- /* Store variable name (and truncate) */
+ /* Store expression to evaluate. */
ast_str_set_substr(&substr1, 0, vars, len);
/* Substitute if necessary */
if (needsub) {
- size_t my_used;
-
if (!substr2) {
substr2 = ast_str_create(16);
+ if (!substr2) {
+ continue;
+ }
}
- ast_str_substitute_variables_full(&substr2, 0, c, headp, ast_str_buffer(substr1), &my_used);
+ ast_str_substitute_variables_full(&substr2, 0, c, headp, ast_str_buffer(substr1), NULL);
finalvars = ast_str_buffer(substr2);
} else {
finalvars = ast_str_buffer(substr1);
@@ -566,7 +607,9 @@
ast_str_append(buf, maxlen, "%s", ast_str_buffer(substr3));
}
}
- *used = ast_str_strlen(*buf) - orig_size;
+ if (used) {
+ *used = ast_str_strlen(*buf);
+ }
ast_free(substr1);
ast_free(substr2);
ast_free(substr3);
@@ -574,49 +617,58 @@
void ast_str_substitute_variables(struct ast_str **buf, ssize_t maxlen, struct ast_channel *chan, const char *templ)
{
- size_t used;
- ast_str_substitute_variables_full(buf, maxlen, chan, NULL, templ, &used);
+ ast_str_substitute_variables_full(buf, maxlen, chan, NULL, templ, NULL);
}
void ast_str_substitute_variables_varshead(struct ast_str **buf, ssize_t maxlen, struct varshead *headp, const char *templ)
{
- size_t used;
- ast_str_substitute_variables_full(buf, maxlen, NULL, headp, templ, &used);
+ ast_str_substitute_variables_full(buf, maxlen, NULL, headp, templ, NULL);
}
void pbx_substitute_variables_helper_full(struct ast_channel *c, struct varshead *headp, const char *cp1, char *cp2, int count, size_t *used)
{
/* Substitutes variables into cp2, based on string cp1, cp2 NO LONGER NEEDS TO BE ZEROED OUT!!!! */
- char *cp4 = NULL;
- const char *whereweare, *orig_cp2 = cp2;
- int length, offset, offset2, isfunction;
+ const char *whereweare;
+ const char *orig_cp2 = cp2;
char *workspace = NULL;
- char *ltmp = NULL, *var = NULL;
- char *nextvar, *nextexp, *nextthing;
- char *vars, *vare;
- int pos, brackets, needsub, len;
+ char *ltmp = NULL;
+ char *var = NULL;
*cp2 = 0; /* just in case nothing ends up there */
whereweare = cp1;
while (!ast_strlen_zero(whereweare) && count) {
- /* Assume we're copying the whole remaining string */
- pos = strlen(whereweare);
- nextvar = NULL;
- nextexp = NULL;
+ char *nextvar = NULL;
+ char *nextexp = NULL;
+ char *nextthing;
+ char *vars;
+ char *vare;
+ int length;
+ int pos;
+ int brackets;
+ int needsub;
+ int len;
+
+ /* Determine how much simply needs to be copied to the output buf. */
nextthing = strchr(whereweare, '$');
if (nextthing) {
+ pos = nextthing - whereweare;
switch (nextthing[1]) {
case '{':
+ /* Variable substitution */
nextvar = nextthing;
- pos = nextvar - whereweare;
break;
case '[':
+ /* Expression substitution */
nextexp = nextthing;
- pos = nextexp - whereweare;
break;
default:
- pos = 1;
+ /* '$' is not part of a substitution so include it too. */
+ ++pos;
+ break;
}
+ } else {
+ /* We're copying the whole remaining string */
+ pos = strlen(whereweare);
}
if (pos) {
@@ -634,6 +686,11 @@
}
if (nextvar) {
+ int offset;
+ int offset2;
+ int isfunction;
+ char *cp4;
+
/* We have a variable. Find the start and end, and determine
if we are going to have to recursively call ourselves on the
contents */
@@ -645,35 +702,41 @@
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] == '['))
+ } else if ((vare[0] == '$') && (vare[1] == '[')) {
needsub++;
+ vare++;
+ }
vare++;
}
- if (brackets)
+ len = vare - vars;
+ if (brackets) {
ast_log(LOG_WARNING, "Error in extension logic (missing '}')\n");
- len = vare - vars - 1;
+ } else {
+ /* Don't count the closing '}' in the length. */
+ --len;
+ }
/* Skip totally over variable string */
- whereweare += (len + 3);
+ whereweare = vare;
if (!var)
var = ast_alloca(VAR_BUF_SIZE);
- /* Store variable name (and truncate) */
+ /* Store variable name expression to lookup (and truncate). */
ast_copy_string(var, vars, len + 1);
/* Substitute if necessary */
if (needsub) {
- size_t my_used;
-
if (!ltmp) {
ltmp = ast_alloca(VAR_BUF_SIZE);
}
- pbx_substitute_variables_helper_full(c, headp, var, ltmp, VAR_BUF_SIZE - 1, &my_used);
+ pbx_substitute_variables_helper_full(c, headp, var, ltmp, VAR_BUF_SIZE - 1, NULL);
vars = ltmp;
} else {
vars = var;
@@ -691,16 +754,19 @@
cp4 = ast_func_read(c, vars, workspace, VAR_BUF_SIZE) ? NULL : workspace;
else {
struct varshead old;
- struct ast_channel *c = ast_dummy_channel_alloc();
- if (c) {
- memcpy(&old, ast_channel_varshead(c), sizeof(old));
- memcpy(ast_channel_varshead(c), headp, sizeof(*ast_channel_varshead(c)));
- cp4 = ast_func_read(c, vars, workspace, VAR_BUF_SIZE) ? NULL : workspace;
+ struct ast_channel *bogus;
+
+ bogus = ast_dummy_channel_alloc();
+ if (bogus) {
+ old = *ast_channel_varshead(bogus);
+ *ast_channel_varshead(bogus) = *headp;
+ cp4 = ast_func_read(bogus, vars, workspace, VAR_BUF_SIZE) ? NULL : workspace;
/* Don't deallocate the varshead that was passed in */
- memcpy(ast_channel_varshead(c), &old, sizeof(*ast_channel_varshead(c)));
- c = ast_channel_unref(c);
+ *ast_channel_varshead(bogus) = old;
+ ast_channel_unref(bogus);
} else {
- ast_log(LOG_ERROR, "Unable to allocate bogus channel for variable substitution. Function results may be blank.\n");
+ ast_log(LOG_ERROR, "Unable to allocate bogus channel for function value substitution.\n");
+ cp4 = NULL;
}
}
ast_debug(2, "Function %s result is '%s'\n", vars, cp4 ? cp4 : "(null)");
@@ -743,34 +809,35 @@
}
vare++;
}
- if (brackets)
+ len = vare - vars;
+ if (brackets) {
ast_log(LOG_WARNING, "Error in extension logic (missing ']')\n");
- len = vare - vars - 1;
+ } else {
+ /* Don't count the closing ']' in the length. */
+ --len;
+ }
/* Skip totally over expression */
- whereweare += (len + 3);
+ whereweare = vare;
if (!var)
var = ast_alloca(VAR_BUF_SIZE);
- /* Store variable name (and truncate) */
+ /* Store expression to evaluate (and truncate). */
ast_copy_string(var, vars, len + 1);
/* Substitute if necessary */
if (needsub) {
- size_t my_used;
-
if (!ltmp) {
ltmp = ast_alloca(VAR_BUF_SIZE);
}
- pbx_substitute_variables_helper_full(c, headp, var, ltmp, VAR_BUF_SIZE - 1, &my_used);
+ pbx_substitute_variables_helper_full(c, headp, var, ltmp, VAR_BUF_SIZE - 1, NULL);
vars = ltmp;
} else {
vars = var;
}
length = ast_expr(vars, cp2, count, c);
-
if (length) {
ast_debug(1, "Expression result is '%s'\n", cp2);
count -= length;
@@ -779,19 +846,19 @@
}
}
}
- *used = cp2 - orig_cp2;
+ if (used) {
+ *used = cp2 - orig_cp2;
+ }
}
void pbx_substitute_variables_helper(struct ast_channel *c, const char *cp1, char *cp2, int count)
{
- size_t used;
- pbx_substitute_variables_helper_full(c, (c) ? ast_channel_varshead(c) : NULL, cp1, cp2, count, &used);
+ pbx_substitute_variables_helper_full(c, (c) ? ast_channel_varshead(c) : NULL, cp1, cp2, count, NULL);
}
void pbx_substitute_variables_varshead(struct varshead *headp, const char *cp1, char *cp2, int count)
{
- size_t used;
- pbx_substitute_variables_helper_full(NULL, headp, cp1, cp2, count, &used);
+ pbx_substitute_variables_helper_full(NULL, headp, cp1, cp2, count, NULL);
}
/*! \brief CLI support for listing global variables in a parseable way */
--
To view, visit https://gerrit.asterisk.org/8039
To unsubscribe, visit https://gerrit.asterisk.org/settings
Gerrit-Project: asterisk
Gerrit-Branch: 15
Gerrit-MessageType: newchange
Gerrit-Change-Id: Ia83d99f4f16abe47f329eb39b6ff2013ae7c9854
Gerrit-Change-Number: 8039
Gerrit-PatchSet: 1
Gerrit-Owner: Richard Mudgett <rmudgett at digium.com>
-------------- next part --------------
An HTML attachment was scrubbed...
URL: <http://lists.digium.com/pipermail/asterisk-code-review/attachments/20180122/b90b256a/attachment-0001.html>
More information about the asterisk-code-review
mailing list