mirror of
https://github.com/asterisk/asterisk.git
synced 2025-10-26 14:27:14 +00:00
core: Add PARSE_TIMELEN support to ast_parse_arg and ACO.
This adds support for parsing timelen values from config files. This includes support for all flags which apply to PARSE_INT32. Support for this parser is added to ACO via the OPT_TIMELEN_T option type. Fixes an issue where extra characters provided to ast_app_parse_timelen were ignored, they now cause an error. Testing is included. ASTERISK-27117 #close Change-Id: I6b333feca7e3f83b4ef5bf2636fc0fd613742554
This commit is contained in:
13
main/app.c
13
main/app.c
@@ -3060,19 +3060,32 @@ int ast_app_parse_timelen(const char *timestr, int *result, enum ast_timelen uni
|
||||
case 'h':
|
||||
case 'H':
|
||||
unit = TIMELEN_HOURS;
|
||||
if (u[1] != '\0') {
|
||||
return -1;
|
||||
}
|
||||
break;
|
||||
case 's':
|
||||
case 'S':
|
||||
unit = TIMELEN_SECONDS;
|
||||
if (u[1] != '\0') {
|
||||
return -1;
|
||||
}
|
||||
break;
|
||||
case 'm':
|
||||
case 'M':
|
||||
if (toupper(u[1]) == 'S') {
|
||||
unit = TIMELEN_MILLISECONDS;
|
||||
if (u[2] != '\0') {
|
||||
return -1;
|
||||
}
|
||||
} else if (u[1] == '\0') {
|
||||
unit = TIMELEN_MINUTES;
|
||||
} else {
|
||||
return -1;
|
||||
}
|
||||
break;
|
||||
default:
|
||||
return -1;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@@ -3741,6 +3741,55 @@ uint32_done:
|
||||
break;
|
||||
}
|
||||
|
||||
case PARSE_TIMELEN:
|
||||
{
|
||||
int x = 0;
|
||||
int *result = p_result;
|
||||
int def = result ? *result : 0;
|
||||
int high = INT_MAX;
|
||||
int low = INT_MIN;
|
||||
enum ast_timelen defunit;
|
||||
|
||||
defunit = va_arg(ap, enum ast_timelen);
|
||||
/* optional arguments: default value and/or (low, high) */
|
||||
if (flags & PARSE_DEFAULT) {
|
||||
def = va_arg(ap, int);
|
||||
}
|
||||
if (flags & (PARSE_IN_RANGE | PARSE_OUT_RANGE)) {
|
||||
low = va_arg(ap, int);
|
||||
high = va_arg(ap, int);
|
||||
}
|
||||
if (ast_strlen_zero(arg)) {
|
||||
error = 1;
|
||||
goto timelen_done;
|
||||
}
|
||||
error = ast_app_parse_timelen(arg, &x, defunit);
|
||||
if (error || x < INT_MIN || x > INT_MAX) {
|
||||
/* Parse error, or type out of int bounds */
|
||||
error = 1;
|
||||
goto timelen_done;
|
||||
}
|
||||
error = (x < low) || (x > high);
|
||||
if (flags & PARSE_RANGE_DEFAULTS) {
|
||||
if (x < low) {
|
||||
def = low;
|
||||
} else if (x > high) {
|
||||
def = high;
|
||||
}
|
||||
}
|
||||
if (flags & PARSE_OUT_RANGE) {
|
||||
error = !error;
|
||||
}
|
||||
timelen_done:
|
||||
if (result) {
|
||||
*result = error ? def : x;
|
||||
}
|
||||
|
||||
ast_debug(3, "extract timelen from [%s] in [%d, %d] gives [%d](%d)\n",
|
||||
arg, low, high, result ? *result : x, error);
|
||||
break;
|
||||
}
|
||||
|
||||
case PARSE_DOUBLE:
|
||||
{
|
||||
double *result = p_result;
|
||||
|
||||
@@ -34,6 +34,7 @@
|
||||
#include "asterisk/config_options.h"
|
||||
#include "asterisk/stringfields.h"
|
||||
#include "asterisk/acl.h"
|
||||
#include "asterisk/app.h"
|
||||
#include "asterisk/frame.h"
|
||||
#include "asterisk/xmldoc.h"
|
||||
#include "asterisk/cli.h"
|
||||
@@ -118,6 +119,7 @@ static void config_option_destroy(void *obj)
|
||||
|
||||
static int int_handler_fn(const struct aco_option *opt, struct ast_variable *var, void *obj);
|
||||
static int uint_handler_fn(const struct aco_option *opt, struct ast_variable *var, void *obj);
|
||||
static int timelen_handler_fn(const struct aco_option *opt, struct ast_variable *var, void *obj);
|
||||
static int double_handler_fn(const struct aco_option *opt, struct ast_variable *var, void *obj);
|
||||
static int sockaddr_handler_fn(const struct aco_option *opt, struct ast_variable *var, void *obj);
|
||||
static int stringfield_handler_fn(const struct aco_option *opt, struct ast_variable *var, void *obj);
|
||||
@@ -151,6 +153,7 @@ static aco_option_handler ast_config_option_default_handler(enum aco_option_type
|
||||
case OPT_SOCKADDR_T: return sockaddr_handler_fn;
|
||||
case OPT_STRINGFIELD_T: return stringfield_handler_fn;
|
||||
case OPT_UINT_T: return uint_handler_fn;
|
||||
case OPT_TIMELEN_T: return timelen_handler_fn;
|
||||
|
||||
case OPT_CUSTOM_T: return NULL;
|
||||
}
|
||||
@@ -1378,6 +1381,39 @@ static int uint_handler_fn(const struct aco_option *opt, struct ast_variable *va
|
||||
return res;
|
||||
}
|
||||
|
||||
/*! \brief Default option handler for timelen signed integers
|
||||
* \note For a description of the opt->flags and opt->args values, see the documentation for
|
||||
* enum aco_option_type in config_options.h
|
||||
*/
|
||||
static int timelen_handler_fn(const struct aco_option *opt, struct ast_variable *var, void *obj)
|
||||
{
|
||||
int *field = (int *)(obj + opt->args[0]);
|
||||
unsigned int flags = PARSE_TIMELEN | opt->flags;
|
||||
int res = 0;
|
||||
if (opt->flags & PARSE_IN_RANGE) {
|
||||
if (opt->flags & PARSE_DEFAULT) {
|
||||
res = ast_parse_arg(var->value, flags, field, (enum ast_timelen) opt->args[1], (int) opt->args[2], (int) opt->args[3], opt->args[4]);
|
||||
} else {
|
||||
res = ast_parse_arg(var->value, flags, field, (enum ast_timelen) opt->args[1], (int) opt->args[2], (int) opt->args[3]);
|
||||
}
|
||||
if (res) {
|
||||
if (opt->flags & PARSE_RANGE_DEFAULTS) {
|
||||
ast_log(LOG_WARNING, "Failed to set %s=%s. Set to %d instead due to range limit (%d, %d)\n", var->name, var->value, *field, (int) opt->args[2], (int) opt->args[3]);
|
||||
res = 0;
|
||||
} else if (opt->flags & PARSE_DEFAULT) {
|
||||
ast_log(LOG_WARNING, "Failed to set %s=%s, Set to default value %d instead.\n", var->name, var->value, *field);
|
||||
res = 0;
|
||||
}
|
||||
}
|
||||
} else if ((opt->flags & PARSE_DEFAULT) && ast_parse_arg(var->value, flags, field, (enum ast_timelen) opt->args[1], (int) opt->args[2])) {
|
||||
ast_log(LOG_WARNING, "Attempted to set %s=%s, but set it to %d instead due to default)\n", var->name, var->value, *field);
|
||||
} else {
|
||||
res = ast_parse_arg(var->value, flags, field, (enum ast_timelen) opt->args[1]);
|
||||
}
|
||||
|
||||
return res;
|
||||
}
|
||||
|
||||
/*! \brief Default option handler for doubles
|
||||
* \note For a description of the opt->flags and opt->args values, see the documentation for
|
||||
* enum aco_option_type in config_options.h
|
||||
|
||||
Reference in New Issue
Block a user