<p>N A has uploaded this change for <strong>review</strong>.</p><p><a href="https://gerrit.asterisk.org/c/asterisk/+/19834">View Change</a></p><pre style="font-family: monospace,monospace; white-space: pre-wrap;">func_json: Enhance parsing capabilities of JSON_DECODE<br><br>Adds support for arrays to JSON_DECODE by allowing the<br>user to print out entire arrays or index a particular<br>key or print the number of keys in a JSON array.<br><br>Additionally, adds support for recursively iterating a<br>JSON tree in a single function call, making it easier<br>to parse JSON results with multiple levels. A maximum<br>depth is imposed to prevent potentially blowing<br>the stack.<br><br>Also fixes a bug with the unit tests causing an empty<br>string to be printed instead of the actual test result.<br><br>ASTERISK-29913 #close<br><br>Change-Id: I603940b216a3911b498fc6583b18934011ef5d5b<br>---<br>A doc/CHANGES-staging/func_json_additions.txt<br>M funcs/func_json.c<br>2 files changed, 221 insertions(+), 40 deletions(-)<br><br></pre><pre style="font-family: monospace,monospace; white-space: pre-wrap;">git pull ssh://gerrit.asterisk.org:29418/asterisk refs/changes/34/19834/1</pre><pre style="font-family: monospace,monospace; white-space: pre-wrap;"><span>diff --git a/doc/CHANGES-staging/func_json_additions.txt b/doc/CHANGES-staging/func_json_additions.txt</span><br><span>new file mode 100644</span><br><span>index 0000000..963f0b1</span><br><span>--- /dev/null</span><br><span>+++ b/doc/CHANGES-staging/func_json_additions.txt</span><br><span>@@ -0,0 +1,5 @@</span><br><span style="color: hsl(120, 100%, 40%);">+Subject: func_json</span><br><span style="color: hsl(120, 100%, 40%);">+</span><br><span style="color: hsl(120, 100%, 40%);">+Additional parsing capabilities have been added to the</span><br><span style="color: hsl(120, 100%, 40%);">+JSON_DECODE function, including support for arrays</span><br><span style="color: hsl(120, 100%, 40%);">+and recursive indexing.</span><br><span>diff --git a/funcs/func_json.c b/funcs/func_json.c</span><br><span>index 55a0343..6672c58 100644</span><br><span>--- a/funcs/func_json.c</span><br><span>+++ b/funcs/func_json.c</span><br><span>@@ -1,7 +1,7 @@</span><br><span> /*</span><br><span>  * Asterisk -- An open source telephony toolkit.</span><br><span>  *</span><br><span style="color: hsl(0, 100%, 40%);">- * Copyright (C) 2021, Naveen Albert</span><br><span style="color: hsl(120, 100%, 40%);">+ * Copyright (C) 2021-2022, Naveen Albert</span><br><span>  *</span><br><span>  * See http://www.asterisk.org for more information about</span><br><span>  * the Asterisk project. Please do not directly contact</span><br><span>@@ -54,6 +54,21 @@</span><br><span>                      </parameter></span><br><span>                   <parameter name="item" required="true"></span><br><span>                            <para>The name of the key whose value to return.</para></span><br><span style="color: hsl(120, 100%, 40%);">+                           <para>Multiple keys can be listed separated by a hierarchy delimeter, which will recursively index into a nested JSON string to retrieve a specific subkey's value.</para></span><br><span style="color: hsl(120, 100%, 40%);">+                    </parameter></span><br><span style="color: hsl(120, 100%, 40%);">+                    <parameter name="separator" required="false"></span><br><span style="color: hsl(120, 100%, 40%);">+                               <para>A single character that delimits a key hierarchy for nested indexing. Default is a period (.)</para></span><br><span style="color: hsl(120, 100%, 40%);">+                                <para>This value should not appear in the key or hierarchy of keys itself, except to delimit the hierarchy of keys.</para></span><br><span style="color: hsl(120, 100%, 40%);">+                        </parameter></span><br><span style="color: hsl(120, 100%, 40%);">+                    <parameter name="options" required="no"></span><br><span style="color: hsl(120, 100%, 40%);">+                            <optionlist></span><br><span style="color: hsl(120, 100%, 40%);">+                                    <option name="c"></span><br><span style="color: hsl(120, 100%, 40%);">+                                             <para>For keys that reference a JSON array, return</span><br><span style="color: hsl(120, 100%, 40%);">+                                              the number of items in the array.</para></span><br><span style="color: hsl(120, 100%, 40%);">+                                                <para>This option has no effect on any other type</span><br><span style="color: hsl(120, 100%, 40%);">+                                               of value.</para></span><br><span style="color: hsl(120, 100%, 40%);">+                                        </option></span><br><span style="color: hsl(120, 100%, 40%);">+                               </optionlist></span><br><span>                  </parameter></span><br><span>           </syntax></span><br><span>              <description></span><br><span>@@ -69,19 +84,131 @@</span><br><span> </span><br><span> AST_THREADSTORAGE(result_buf);</span><br><span> </span><br><span style="color: hsl(120, 100%, 40%);">+enum json_option_flags {</span><br><span style="color: hsl(120, 100%, 40%);">+        OPT_COUNT = (1 << 0),</span><br><span style="color: hsl(120, 100%, 40%);">+};</span><br><span style="color: hsl(120, 100%, 40%);">+</span><br><span style="color: hsl(120, 100%, 40%);">+AST_APP_OPTIONS(json_options, {</span><br><span style="color: hsl(120, 100%, 40%);">+    AST_APP_OPTION('c', OPT_COUNT),</span><br><span style="color: hsl(120, 100%, 40%);">+});</span><br><span style="color: hsl(120, 100%, 40%);">+</span><br><span style="color: hsl(120, 100%, 40%);">+#define MAX_JSON_STACK 32</span><br><span style="color: hsl(120, 100%, 40%);">+</span><br><span style="color: hsl(120, 100%, 40%);">+static int parse_node(char **key, char *currentkey, char *nestchar, int count, struct ast_json *json, char *buf, size_t len, int *depth)</span><br><span style="color: hsl(120, 100%, 40%);">+{</span><br><span style="color: hsl(120, 100%, 40%);">+      const char *result = NULL;</span><br><span style="color: hsl(120, 100%, 40%);">+    char *previouskey;</span><br><span style="color: hsl(120, 100%, 40%);">+    struct ast_json *jsonval = json;</span><br><span style="color: hsl(120, 100%, 40%);">+</span><br><span style="color: hsl(120, 100%, 40%);">+    /* Prevent a huge JSON string from blowing the stack. */</span><br><span style="color: hsl(120, 100%, 40%);">+      if (*depth > MAX_JSON_STACK) {</span><br><span style="color: hsl(120, 100%, 40%);">+             ast_log(LOG_WARNING, "Max JSON stack (%d) exceeded\n", MAX_JSON_STACK);</span><br><span style="color: hsl(120, 100%, 40%);">+             return -1;</span><br><span style="color: hsl(120, 100%, 40%);">+    }</span><br><span style="color: hsl(120, 100%, 40%);">+</span><br><span style="color: hsl(120, 100%, 40%);">+   snprintf(buf, len, "%s", ""); /* clear the buffer from previous round if necessary */</span><br><span style="color: hsl(120, 100%, 40%);">+     if (!json) { /* no error or warning should be thrown */</span><br><span style="color: hsl(120, 100%, 40%);">+               ast_debug(1, "Could not find key '%s' in parsed JSON\n", currentkey);</span><br><span style="color: hsl(120, 100%, 40%);">+               return -1;</span><br><span style="color: hsl(120, 100%, 40%);">+    }</span><br><span style="color: hsl(120, 100%, 40%);">+</span><br><span style="color: hsl(120, 100%, 40%);">+   switch(ast_json_typeof(jsonval)) {</span><br><span style="color: hsl(120, 100%, 40%);">+            unsigned long int size;</span><br><span style="color: hsl(120, 100%, 40%);">+               int r;</span><br><span style="color: hsl(120, 100%, 40%);">+</span><br><span style="color: hsl(120, 100%, 40%);">+              case AST_JSON_STRING:</span><br><span style="color: hsl(120, 100%, 40%);">+                 result = ast_json_string_get(jsonval);</span><br><span style="color: hsl(120, 100%, 40%);">+                        ast_debug(1, "Got JSON string: %s\n", result);</span><br><span style="color: hsl(120, 100%, 40%);">+                      ast_copy_string(buf, result, len);</span><br><span style="color: hsl(120, 100%, 40%);">+                    break;</span><br><span style="color: hsl(120, 100%, 40%);">+                case AST_JSON_INTEGER:</span><br><span style="color: hsl(120, 100%, 40%);">+                        r = ast_json_integer_get(jsonval);</span><br><span style="color: hsl(120, 100%, 40%);">+                    ast_debug(1, "Got JSON integer: %d\n", r);</span><br><span style="color: hsl(120, 100%, 40%);">+                  snprintf(buf, len, "%d", r); /* the snprintf below is mutually exclusive with this one */</span><br><span style="color: hsl(120, 100%, 40%);">+                   break;</span><br><span style="color: hsl(120, 100%, 40%);">+                case AST_JSON_ARRAY:</span><br><span style="color: hsl(120, 100%, 40%);">+                  ast_debug(1, "Got JSON array\n");</span><br><span style="color: hsl(120, 100%, 40%);">+                   previouskey = currentkey;</span><br><span style="color: hsl(120, 100%, 40%);">+                     currentkey = strsep(key, nestchar); /* retrieve the desired index */</span><br><span style="color: hsl(120, 100%, 40%);">+                  size = ast_json_array_size(jsonval);</span><br><span style="color: hsl(120, 100%, 40%);">+                  ast_debug(1, "Parsed JSON array of size %lu, key: %s\n", size, currentkey);</span><br><span style="color: hsl(120, 100%, 40%);">+                 if (!currentkey) { /* this is the end, so just dump the array */</span><br><span style="color: hsl(120, 100%, 40%);">+                              if (count) {</span><br><span style="color: hsl(120, 100%, 40%);">+                                  ast_debug(1, "No key on which to index in the array, so returning count: %lu\n", size);</span><br><span style="color: hsl(120, 100%, 40%);">+                                     snprintf(buf, len, "%lu", size);</span><br><span style="color: hsl(120, 100%, 40%);">+                                    return 0;</span><br><span style="color: hsl(120, 100%, 40%);">+                             } else {</span><br><span style="color: hsl(120, 100%, 40%);">+                                      char *result2 = ast_json_dump_string(jsonval);</span><br><span style="color: hsl(120, 100%, 40%);">+                                        ast_debug(1, "No key on which to index in the array, so dumping '%s' array\n", previouskey);</span><br><span style="color: hsl(120, 100%, 40%);">+                                        ast_copy_string(buf, result2, len);</span><br><span style="color: hsl(120, 100%, 40%);">+                                   ast_json_free(result2);</span><br><span style="color: hsl(120, 100%, 40%);">+                               }</span><br><span style="color: hsl(120, 100%, 40%);">+                     } else if (ast_str_to_int(currentkey, &r) || r < 0) {</span><br><span style="color: hsl(120, 100%, 40%);">+                          ast_debug(1, "Requested index '%s' is not numeric or is invalid\n", currentkey);</span><br><span style="color: hsl(120, 100%, 40%);">+                    } else if (r >= size) {</span><br><span style="color: hsl(120, 100%, 40%);">+                            ast_debug(1, "Requested index '%d' does not exist in parsed array\n", r);</span><br><span style="color: hsl(120, 100%, 40%);">+                   } else {</span><br><span style="color: hsl(120, 100%, 40%);">+                              struct ast_json *json2 = ast_json_array_get(jsonval, r);</span><br><span style="color: hsl(120, 100%, 40%);">+                              if (!json2) {</span><br><span style="color: hsl(120, 100%, 40%);">+                                 ast_debug(1, "Array index %d contains empty item\n", r);</span><br><span style="color: hsl(120, 100%, 40%);">+                                    return -1;</span><br><span style="color: hsl(120, 100%, 40%);">+                            }</span><br><span style="color: hsl(120, 100%, 40%);">+                             previouskey = currentkey;</span><br><span style="color: hsl(120, 100%, 40%);">+                             currentkey = strsep(key, nestchar); /* get the next subkey */</span><br><span style="color: hsl(120, 100%, 40%);">+                         ast_debug(1, "Recursing on index %d in array (key was '%s' and is now '%s')\n", r, previouskey, currentkey);</span><br><span style="color: hsl(120, 100%, 40%);">+                                /* json2 is a borrowed ref. That's fine, since json won't get freed until recursing is over */</span><br><span style="color: hsl(120, 100%, 40%);">+                                /* If there are keys remaining, then parse the next object we can get. Otherwise, just dump the child */</span><br><span style="color: hsl(120, 100%, 40%);">+                              if (parse_node(key, currentkey, nestchar, count, currentkey ? ast_json_object_get(json2, currentkey) : json2, buf, len, depth)) { /* recurse on this node */</span><br><span style="color: hsl(120, 100%, 40%);">+                                  return -1;</span><br><span style="color: hsl(120, 100%, 40%);">+                            }</span><br><span style="color: hsl(120, 100%, 40%);">+                     }</span><br><span style="color: hsl(120, 100%, 40%);">+                     break;</span><br><span style="color: hsl(120, 100%, 40%);">+                case AST_JSON_OBJECT:</span><br><span style="color: hsl(120, 100%, 40%);">+         default:</span><br><span style="color: hsl(120, 100%, 40%);">+                      ast_debug(1, "Got generic JSON object for key %s\n", currentkey);</span><br><span style="color: hsl(120, 100%, 40%);">+                   if (!currentkey) { /* this is the end, so just dump the object */</span><br><span style="color: hsl(120, 100%, 40%);">+                             char *result2 = ast_json_dump_string(jsonval);</span><br><span style="color: hsl(120, 100%, 40%);">+                                ast_copy_string(buf, result2, len);</span><br><span style="color: hsl(120, 100%, 40%);">+                           ast_json_free(result2);</span><br><span style="color: hsl(120, 100%, 40%);">+                       } else {</span><br><span style="color: hsl(120, 100%, 40%);">+                              previouskey = currentkey;</span><br><span style="color: hsl(120, 100%, 40%);">+                             currentkey = strsep(key, nestchar); /* retrieve the desired index */</span><br><span style="color: hsl(120, 100%, 40%);">+                          ast_debug(1, "Recursing on object (key was '%s' and is now '%s')\n", previouskey, currentkey);</span><br><span style="color: hsl(120, 100%, 40%);">+                              if (parse_node(key, currentkey, nestchar, count, ast_json_object_get(jsonval, currentkey), buf, len, depth)) { /* recurse on this node */</span><br><span style="color: hsl(120, 100%, 40%);">+                                     return -1;</span><br><span style="color: hsl(120, 100%, 40%);">+                            }</span><br><span style="color: hsl(120, 100%, 40%);">+                     }</span><br><span style="color: hsl(120, 100%, 40%);">+                     break;</span><br><span style="color: hsl(120, 100%, 40%);">+        }</span><br><span style="color: hsl(120, 100%, 40%);">+     return 0;</span><br><span style="color: hsl(120, 100%, 40%);">+}</span><br><span style="color: hsl(120, 100%, 40%);">+</span><br><span> static int json_decode_read(struct ast_channel *chan, const char *cmd, char *data, char *buf, size_t len)</span><br><span> {</span><br><span style="color: hsl(0, 100%, 40%);">-        struct ast_json *json, *jsonval;</span><br><span style="color: hsl(120, 100%, 40%);">+      int count = 0;</span><br><span style="color: hsl(120, 100%, 40%);">+        struct ast_flags flags = {0};</span><br><span style="color: hsl(120, 100%, 40%);">+ struct ast_json *json = NULL;</span><br><span style="color: hsl(120, 100%, 40%);">+ char *nestchar = "."; /* default delimeter for nesting key indexing is . */</span><br><span style="color: hsl(120, 100%, 40%);">+ int res, depth = 0;</span><br><span style="color: hsl(120, 100%, 40%);">+</span><br><span>        AST_DECLARE_APP_ARGS(args,</span><br><span>           AST_APP_ARG(varname);</span><br><span>                AST_APP_ARG(key);</span><br><span style="color: hsl(120, 100%, 40%);">+             AST_APP_ARG(nestchar);</span><br><span style="color: hsl(120, 100%, 40%);">+                AST_APP_ARG(options);</span><br><span>        );</span><br><span style="color: hsl(0, 100%, 40%);">-      char *varsubst, *result2;</span><br><span style="color: hsl(0, 100%, 40%);">-       const char *result = NULL;</span><br><span style="color: hsl(120, 100%, 40%);">+    char *varsubst, *key, *currentkey, *nextkey, *firstkey, *tmp;</span><br><span>        struct ast_str *str = ast_str_thread_get(&result_buf, 16);</span><br><span> </span><br><span>   AST_STANDARD_APP_ARGS(args, data);</span><br><span> </span><br><span style="color: hsl(120, 100%, 40%);">+        if (!ast_strlen_zero(args.options)) {</span><br><span style="color: hsl(120, 100%, 40%);">+         ast_app_parse_options(json_options, &flags, NULL, args.options);</span><br><span style="color: hsl(120, 100%, 40%);">+          if (ast_test_flag(&flags, OPT_COUNT)) {</span><br><span style="color: hsl(120, 100%, 40%);">+                   count = 1;</span><br><span style="color: hsl(120, 100%, 40%);">+            }</span><br><span style="color: hsl(120, 100%, 40%);">+     }</span><br><span style="color: hsl(120, 100%, 40%);">+</span><br><span>  if (ast_strlen_zero(args.varname)) {</span><br><span>                 ast_log(LOG_WARNING, "%s requires a variable name\n", cmd);</span><br><span>                return -1;</span><br><span>@@ -90,6 +217,15 @@</span><br><span>             ast_log(LOG_WARNING, "%s requires a key\n", cmd);</span><br><span>          return -1;</span><br><span>   }</span><br><span style="color: hsl(120, 100%, 40%);">+     key = ast_strdupa(args.key);</span><br><span style="color: hsl(120, 100%, 40%);">+  if (!ast_strlen_zero(args.nestchar)) {</span><br><span style="color: hsl(120, 100%, 40%);">+                int seplen = strlen(args.nestchar);</span><br><span style="color: hsl(120, 100%, 40%);">+           if (seplen != 1) {</span><br><span style="color: hsl(120, 100%, 40%);">+                    ast_log(LOG_WARNING, "Nesting separator '%s' has length %d and is invalid (must be a single character)\n", args.nestchar, seplen);</span><br><span style="color: hsl(120, 100%, 40%);">+          } else {</span><br><span style="color: hsl(120, 100%, 40%);">+                      nestchar = args.nestchar;</span><br><span style="color: hsl(120, 100%, 40%);">+             }</span><br><span style="color: hsl(120, 100%, 40%);">+     }</span><br><span> </span><br><span>        varsubst = ast_alloca(strlen(args.varname) + 4); /* +4 for ${} and null terminator */</span><br><span>        if (!varsubst) {</span><br><span>@@ -98,45 +234,43 @@</span><br><span>      }</span><br><span>    sprintf(varsubst, "${%s}", args.varname); /* safe, because of the above allocation */</span><br><span>      ast_str_substitute_variables(&str, 0, chan, varsubst);</span><br><span style="color: hsl(120, 100%, 40%);">+</span><br><span style="color: hsl(120, 100%, 40%);">+  ast_debug(1, "Parsing JSON using nesting delimeter '%s'\n", nestchar);</span><br><span style="color: hsl(120, 100%, 40%);">+</span><br><span>   if (ast_str_strlen(str) == 0) {</span><br><span>              ast_debug(1, "Variable '%s' contains no data, nothing to search!\n", args.varname);</span><br><span>                return -1; /* empty json string */</span><br><span>   }</span><br><span> </span><br><span style="color: hsl(0, 100%, 40%);">-   ast_debug(1, "Parsing JSON: %s\n", ast_str_buffer(str));</span><br><span style="color: hsl(120, 100%, 40%);">+    /* allow for multiple key nesting */</span><br><span style="color: hsl(120, 100%, 40%);">+  currentkey = key;</span><br><span style="color: hsl(120, 100%, 40%);">+     firstkey = ast_strdupa(currentkey);</span><br><span style="color: hsl(120, 100%, 40%);">+   tmp = strstr(firstkey, nestchar);</span><br><span style="color: hsl(120, 100%, 40%);">+     if (tmp) {</span><br><span style="color: hsl(120, 100%, 40%);">+            *tmp = '\0';</span><br><span style="color: hsl(120, 100%, 40%);">+  }</span><br><span> </span><br><span style="color: hsl(120, 100%, 40%);">+ /* parse a string as JSON */</span><br><span style="color: hsl(120, 100%, 40%);">+  ast_debug(1, "Parsing JSON: %s (key: '%s')\n", ast_str_buffer(str), currentkey);</span><br><span style="color: hsl(120, 100%, 40%);">+    if (ast_strlen_zero(currentkey)) {</span><br><span style="color: hsl(120, 100%, 40%);">+            ast_debug(1, "Empty JSON key\n");</span><br><span style="color: hsl(120, 100%, 40%);">+           return -1;</span><br><span style="color: hsl(120, 100%, 40%);">+    }</span><br><span style="color: hsl(120, 100%, 40%);">+     if (ast_str_strlen(str) == 0) {</span><br><span style="color: hsl(120, 100%, 40%);">+               ast_debug(1, "JSON node '%s', contains no data, nothing to search!\n", currentkey);</span><br><span style="color: hsl(120, 100%, 40%);">+         return -1; /* empty json string */</span><br><span style="color: hsl(120, 100%, 40%);">+    }</span><br><span>    json = ast_json_load_str(str, NULL);</span><br><span style="color: hsl(0, 100%, 40%);">-</span><br><span>         if (!json) {</span><br><span>                 ast_log(LOG_WARNING, "Failed to parse as JSON: %s\n", ast_str_buffer(str));</span><br><span>                return -1;</span><br><span>   }</span><br><span> </span><br><span style="color: hsl(0, 100%, 40%);">-   jsonval = ast_json_object_get(json, args.key);</span><br><span style="color: hsl(0, 100%, 40%);">-  if (!jsonval) { /* no error or warning should be thrown */</span><br><span style="color: hsl(0, 100%, 40%);">-              ast_debug(1, "Could not find key '%s' in parsed JSON\n", args.key);</span><br><span style="color: hsl(0, 100%, 40%);">-           ast_json_unref(json);</span><br><span style="color: hsl(0, 100%, 40%);">-           return -1;</span><br><span style="color: hsl(0, 100%, 40%);">-      }</span><br><span style="color: hsl(0, 100%, 40%);">-       switch(ast_json_typeof(jsonval)) {</span><br><span style="color: hsl(0, 100%, 40%);">-              int r;</span><br><span style="color: hsl(0, 100%, 40%);">-          case AST_JSON_STRING:</span><br><span style="color: hsl(0, 100%, 40%);">-                   result = ast_json_string_get(jsonval);</span><br><span style="color: hsl(0, 100%, 40%);">-                  snprintf(buf, len, "%s", result);</span><br><span style="color: hsl(0, 100%, 40%);">-                     break;</span><br><span style="color: hsl(0, 100%, 40%);">-          case AST_JSON_INTEGER:</span><br><span style="color: hsl(0, 100%, 40%);">-                  r = ast_json_integer_get(jsonval);</span><br><span style="color: hsl(0, 100%, 40%);">-                      snprintf(buf, len, "%d", r); /* the snprintf below is mutually exclusive with this one */</span><br><span style="color: hsl(0, 100%, 40%);">-                     break;</span><br><span style="color: hsl(0, 100%, 40%);">-          default:</span><br><span style="color: hsl(0, 100%, 40%);">-                        result2 = ast_json_dump_string(jsonval);</span><br><span style="color: hsl(0, 100%, 40%);">-                        snprintf(buf, len, "%s", result2);</span><br><span style="color: hsl(0, 100%, 40%);">-                    ast_json_free(result2);</span><br><span style="color: hsl(0, 100%, 40%);">-                 break;</span><br><span style="color: hsl(0, 100%, 40%);">-  }</span><br><span style="color: hsl(120, 100%, 40%);">+     /* parse the JSON object, potentially recursively */</span><br><span style="color: hsl(120, 100%, 40%);">+  nextkey = strsep(&key, nestchar);</span><br><span style="color: hsl(120, 100%, 40%);">+ res = parse_node(&key, nextkey, nestchar, count, ast_json_object_get(json, firstkey), buf, len, &depth);</span><br><span>     ast_json_unref(json);</span><br><span style="color: hsl(0, 100%, 40%);">-</span><br><span style="color: hsl(0, 100%, 40%);">-   return 0;</span><br><span style="color: hsl(120, 100%, 40%);">+     return res;</span><br><span> }</span><br><span> </span><br><span> static struct ast_custom_function json_decode_function = {</span><br><span>@@ -151,12 +285,26 @@</span><br><span>   struct ast_channel *chan; /* dummy channel */</span><br><span>        struct ast_str *str; /* fancy string for holding comparing value */</span><br><span> </span><br><span style="color: hsl(0, 100%, 40%);">- const char *test_strings[][5] = {</span><br><span style="color: hsl(0, 100%, 40%);">-               {"{\"city\": \"Anytown\", \"state\": \"USA\"}", "city", "Anytown"},</span><br><span style="color: hsl(0, 100%, 40%);">-               {"{\"city\": \"Anytown\", \"state\": \"USA\"}", "state", "USA"},</span><br><span style="color: hsl(0, 100%, 40%);">-          {"{\"city\": \"Anytown\", \"state\": \"USA\"}", "blah", ""},</span><br><span style="color: hsl(0, 100%, 40%);">-              {"{\"key1\": \"123\", \"key2\": \"456\"}", "key1", "123"},</span><br><span style="color: hsl(0, 100%, 40%);">-                {"{\"key1\": 123, \"key2\": 456}", "key1", "123"},</span><br><span style="color: hsl(120, 100%, 40%);">+  const char *test_strings[][6] = {</span><br><span style="color: hsl(120, 100%, 40%);">+             {"{\"city\": \"Anytown\", \"state\": \"USA\"}", "", "city", "Anytown"},</span><br><span style="color: hsl(120, 100%, 40%);">+               {"{\"city\": \"Anytown\", \"state\": \"USA\"}", "", "state", "USA"},</span><br><span style="color: hsl(120, 100%, 40%);">+          {"{\"city\": \"Anytown\", \"state\": \"USA\"}", "", "blah", ""},</span><br><span style="color: hsl(120, 100%, 40%);">+              {"{\"key1\": \"123\", \"key2\": \"456\"}", "", "key1", "123"},</span><br><span style="color: hsl(120, 100%, 40%);">+                {"{\"key1\": 123, \"key2\": 456}", "", "key1", "123"},</span><br><span style="color: hsl(120, 100%, 40%);">+            {"{ \"path\": { \"to\": { \"elem\": \"someVar\" } } }", "/", "path/to/elem", "someVar"},</span><br><span style="color: hsl(120, 100%, 40%);">+              {"{ \"path\": { \"to\": { \"elem\": \"someVar\" } } }", "", "path.to.elem2", ""},</span><br><span style="color: hsl(120, 100%, 40%);">+             {"{ \"path\": { \"to\": { \"arr\": [ \"item0\", \"item1\" ] } } }", "/", "path/to/arr/2", ""}, /* nonexistent index */</span><br><span style="color: hsl(120, 100%, 40%);">+              {"{ \"path\": { \"to\": { \"arr\": [ \"item0\", \"item1\" ] } } }", "/", "path/to/arr/-1", ""}, /* bogus index */</span><br><span style="color: hsl(120, 100%, 40%);">+           {"{ \"path\": { \"to\": { \"arr\": [ \"item0\", \"item1\" ] } } }", "/", "path/to/arr/test", ""}, /* bogus index */</span><br><span style="color: hsl(120, 100%, 40%);">+         {"{ \"path\": { \"to\": { \"arr\": [ \"item0\", \"item1\" ] } } }", "", "path.to.arr.test.test2.subkey", ""}, /* bogus index */</span><br><span style="color: hsl(120, 100%, 40%);">+             {"{ \"path\": { \"to\": { \"arr\": [ \"item0\", \"item1\" ] } } }", ",c", "path.to.arr", "2"}, /* test count */</span><br><span style="color: hsl(120, 100%, 40%);">+             {"{ \"path\": { \"to\": { \"arr\": [ \"item0\", \"item1\" ] } } }", "", "path.to.arr", "[\"item0\",\"item1\"]"},</span><br><span style="color: hsl(120, 100%, 40%);">+                {"{ \"path\": { \"to\": { \"arr\": [ \"item0\", \"item1\" ] } } }", ".", "path.to.arr.1", "item1"},</span><br><span style="color: hsl(120, 100%, 40%);">+         {"{ \"path\": { \"to\": { \"arr\": [ \"item0\", \"item1\" ] } } }", "/", "path/to/arr", "[\"item0\",\"item1\"]"},</span><br><span style="color: hsl(120, 100%, 40%);">+               {"{ \"path\": { \"to\": { \"arr\": [ \"item0\", \"item1\" ] } } }", "/", "path/to/arr/1", "item1"},</span><br><span style="color: hsl(120, 100%, 40%);">+         {"{ \"path\": { \"to\": { \"arr\": [ {\"name\": \"John Smith\", \"phone\": \"123\"}, {\"name\": \"Jane Doe\", \"phone\": \"234\"} ] } } }", ",c", "path.to.arr.0.name", "John Smith"},</span><br><span style="color: hsl(120, 100%, 40%);">+          {"{ \"path\": { \"to\": { \"arr\": [ {\"name\": 1, \"phone\": 123}, {\"name\": 2, \"phone\": 234} ] } } }", ",c", "path.to.arr.0.name", "1"},</span><br><span style="color: hsl(120, 100%, 40%);">+           {"{ \"path\": { \"to\": { \"arr\": [ {\"name\": [ \"item11\", \"item12\" ], \"phone\": [ \"item13\", \"item14\" ]}, {\"name\": [ \"item15\", \"item16\" ], \"phone\": [ \"item17\", \"item18\" ]} ] } } }", ",c", "path.to.arr.0.name.1", "item12"},</span><br><span>   };</span><br><span> </span><br><span>       switch (cmd) {</span><br><span>@@ -182,7 +330,7 @@</span><br><span>         }</span><br><span> </span><br><span>        for (i = 0; i < ARRAY_LEN(test_strings); i++) {</span><br><span style="color: hsl(0, 100%, 40%);">-              char tmp[512], tmp2[512] = "";</span><br><span style="color: hsl(120, 100%, 40%);">+              char tmp[512];</span><br><span> </span><br><span>           struct ast_var_t *var = ast_var_assign("test_string", test_strings[i][0]);</span><br><span>                 if (!var) {</span><br><span>@@ -194,11 +342,11 @@</span><br><span> </span><br><span>              AST_LIST_INSERT_HEAD(ast_channel_varshead(chan), var, entries);</span><br><span> </span><br><span style="color: hsl(0, 100%, 40%);">-             snprintf(tmp, sizeof(tmp), "${JSON_DECODE(%s,%s)}", "test_string", test_strings[i][1]);</span><br><span style="color: hsl(120, 100%, 40%);">+           snprintf(tmp, sizeof(tmp), "${JSON_DECODE(%s,%s,%s)}", "test_string", test_strings[i][2], test_strings[i][1]);</span><br><span> </span><br><span>               ast_str_substitute_variables(&str, 0, chan, tmp);</span><br><span style="color: hsl(0, 100%, 40%);">-           if (strcmp(test_strings[i][2], ast_str_buffer(str))) {</span><br><span style="color: hsl(0, 100%, 40%);">-                  ast_test_status_update(test, "Format string '%s' substituted to '%s'.  Expected '%s'.\n", test_strings[i][0], tmp2, test_strings[i][2]);</span><br><span style="color: hsl(120, 100%, 40%);">+            if (strcmp(test_strings[i][3], ast_str_buffer(str))) {</span><br><span style="color: hsl(120, 100%, 40%);">+                        ast_test_status_update(test, "Format string '%s' substituted to '%s' (key: %s). Expected '%s'.\n", test_strings[i][0], ast_str_buffer(str), test_strings[i][2], test_strings[i][3]);</span><br><span>                       res = AST_TEST_FAIL;</span><br><span>                 }</span><br><span>    }</span><br><span>@@ -214,7 +362,9 @@</span><br><span> {</span><br><span>         int res;</span><br><span> </span><br><span style="color: hsl(120, 100%, 40%);">+#ifdef TEST_FRAMEWORK</span><br><span>  AST_TEST_UNREGISTER(test_JSON_DECODE);</span><br><span style="color: hsl(120, 100%, 40%);">+#endif</span><br><span>       res = ast_custom_function_unregister(&json_decode_function);</span><br><span> </span><br><span>         return res;</span><br><span>@@ -224,7 +374,9 @@</span><br><span> {</span><br><span>       int res;</span><br><span> </span><br><span style="color: hsl(120, 100%, 40%);">+#ifdef TEST_FRAMEWORK</span><br><span>  AST_TEST_REGISTER(test_JSON_DECODE);</span><br><span style="color: hsl(120, 100%, 40%);">+#endif</span><br><span>         res = ast_custom_function_register(&json_decode_function);</span><br><span> </span><br><span>   return res;</span><br><span></span><br></pre><p>To view, visit <a href="https://gerrit.asterisk.org/c/asterisk/+/19834">change 19834</a>. To unsubscribe, or for help writing mail filters, visit <a href="https://gerrit.asterisk.org/settings">settings</a>.</p><div itemscope itemtype="http://schema.org/EmailMessage"><div itemscope itemprop="action" itemtype="http://schema.org/ViewAction"><link itemprop="url" href="https://gerrit.asterisk.org/c/asterisk/+/19834"/><meta itemprop="name" content="View Change"/></div></div>

<div style="display:none"> Gerrit-Project: asterisk </div>
<div style="display:none"> Gerrit-Branch: 18 </div>
<div style="display:none"> Gerrit-Change-Id: I603940b216a3911b498fc6583b18934011ef5d5b </div>
<div style="display:none"> Gerrit-Change-Number: 19834 </div>
<div style="display:none"> Gerrit-PatchSet: 1 </div>
<div style="display:none"> Gerrit-Owner: N A <asterisk@phreaknet.org> </div>
<div style="display:none"> Gerrit-MessageType: newchange </div>