mirror of https://github.com/grpc/grpc.git
The C based gRPC (C++, Python, Ruby, Objective-C, PHP, C#)
https://grpc.io/
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
206 lines
6.5 KiB
206 lines
6.5 KiB
/* |
|
* |
|
* Copyright 2015, Google Inc. |
|
* All rights reserved. |
|
* |
|
* Redistribution and use in source and binary forms, with or without |
|
* modification, are permitted provided that the following conditions are |
|
* met: |
|
* |
|
* * Redistributions of source code must retain the above copyright |
|
* notice, this list of conditions and the following disclaimer. |
|
* * Redistributions in binary form must reproduce the above |
|
* copyright notice, this list of conditions and the following disclaimer |
|
* in the documentation and/or other materials provided with the |
|
* distribution. |
|
* * Neither the name of Google Inc. nor the names of its |
|
* contributors may be used to endorse or promote products derived from |
|
* this software without specific prior written permission. |
|
* |
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
|
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
|
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR |
|
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT |
|
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, |
|
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT |
|
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, |
|
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY |
|
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
|
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
|
* |
|
*/ |
|
|
|
#include <string.h> |
|
|
|
#include <grpc/support/alloc.h> |
|
#include <grpc/support/log.h> |
|
#include <grpc/support/string_util.h> |
|
#include <grpc/support/useful.h> |
|
#include "src/core/lib/json/json.h" |
|
#include "src/core/lib/support/string.h" |
|
|
|
#include "test/core/util/test_config.h" |
|
|
|
typedef struct testing_pair { |
|
const char *input; |
|
const char *output; |
|
} testing_pair; |
|
|
|
static testing_pair testing_pairs[] = { |
|
/* Testing valid parsing. */ |
|
/* Testing trivial parses, with de-indentation. */ |
|
{" 0 ", "0"}, |
|
{" 1 ", "1"}, |
|
{" \" \" ", "\" \""}, |
|
{" \"a\" ", "\"a\""}, |
|
{" true ", "true"}, |
|
/* Testing the parser's ability to decode trivial UTF-16. */ |
|
{"\"\\u0020\\\\\\u0010\\u000a\\u000D\"", "\" \\\\\\u0010\\n\\r\""}, |
|
/* Testing various UTF-8 sequences. */ |
|
{"\"ßâñć௵⇒\"", "\"\\u00df\\u00e2\\u00f1\\u0107\\u0bf5\\u21d2\""}, |
|
{"\"\\u00df\\u00e2\\u00f1\\u0107\\u0bf5\\u21d2\"", |
|
"\"\\u00df\\u00e2\\u00f1\\u0107\\u0bf5\\u21d2\""}, |
|
/* Testing UTF-8 character "𝄞", U+11D1E. */ |
|
{"\"\xf0\x9d\x84\x9e\"", "\"\\ud834\\udd1e\""}, |
|
{"\"\\ud834\\udd1e\"", "\"\\ud834\\udd1e\""}, |
|
{"{\"\\ud834\\udd1e\":0}", "{\"\\ud834\\udd1e\":0}"}, |
|
/* Testing nested empty containers. */ |
|
{ |
|
" [ [ ] , { } , [ ] ] ", "[[],{},[]]", |
|
}, |
|
/* Testing escapes and control chars in key strings. */ |
|
{" { \"\\u007f\x7f\\n\\r\\\"\\f\\b\\\\a , b\": 1, \"\": 0 } ", |
|
"{\"\\u007f\\u007f\\n\\r\\\"\\f\\b\\\\a , b\":1,\"\":0}"}, |
|
/* Testing the writer's ability to cut off invalid UTF-8 sequences. */ |
|
{"\"abc\xf0\x9d\x24\"", "\"abc\""}, |
|
{"\"\xff\"", "\"\""}, |
|
/* Testing valid number parsing. */ |
|
{"[0, 42 , 0.0123, 123.456]", "[0,42,0.0123,123.456]"}, |
|
{"[1e4,-53.235e-31, 0.3e+3]", "[1e4,-53.235e-31,0.3e+3]"}, |
|
/* Testing keywords parsing. */ |
|
{"[true, false, null]", "[true,false,null]"}, |
|
|
|
/* Testing invalid parsing. */ |
|
|
|
/* Testing plain invalid things, exercising the state machine. */ |
|
{"\\", NULL}, |
|
{"nu ll", NULL}, |
|
{"{\"foo\": bar}", NULL}, |
|
{"{\"foo\": bar\"x\"}", NULL}, |
|
{"fals", NULL}, |
|
{"0,0 ", NULL}, |
|
{"\"foo\",[]", NULL}, |
|
/* Testing unterminated string. */ |
|
{"\"\\x", NULL}, |
|
/* Testing invalid UTF-16 number. */ |
|
{"\"\\u123x", NULL}, |
|
{"{\"\\u123x", NULL}, |
|
/* Testing imbalanced surrogate pairs. */ |
|
{"\"\\ud834f", NULL}, |
|
{"{\"\\ud834f\":0}", NULL}, |
|
{"\"\\ud834\\n", NULL}, |
|
{"{\"\\ud834\\n\":0}", NULL}, |
|
{"\"\\udd1ef", NULL}, |
|
{"{\"\\udd1ef\":0}", NULL}, |
|
{"\"\\ud834\\ud834\"", NULL}, |
|
{"{\"\\ud834\\ud834\"\":0}", NULL}, |
|
{"\"\\ud834\\u1234\"", NULL}, |
|
{"{\"\\ud834\\u1234\"\":0}", NULL}, |
|
{"\"\\ud834]\"", NULL}, |
|
{"{\"\\ud834]\"\":0}", NULL}, |
|
{"\"\\ud834 \"", NULL}, |
|
{"{\"\\ud834 \"\":0}", NULL}, |
|
{"\"\\ud834\\\\\"", NULL}, |
|
{"{\"\\ud834\\\\\"\":0}", NULL}, |
|
/* Testing embedded invalid whitechars. */ |
|
{"\"\n\"", NULL}, |
|
{"\"\t\"", NULL}, |
|
/* Testing empty json data. */ |
|
{"", NULL}, |
|
/* Testing extra characters after end of parsing. */ |
|
{"{},", NULL}, |
|
/* Testing imbalanced containers. */ |
|
{"{}}", NULL}, |
|
{"[]]", NULL}, |
|
{"{{}", NULL}, |
|
{"[[]", NULL}, |
|
{"[}", NULL}, |
|
{"{]", NULL}, |
|
/* Testing bad containers. */ |
|
{"{x}", NULL}, |
|
{"{x=0,y}", NULL}, |
|
/* Testing trailing comma. */ |
|
{"{,}", NULL}, |
|
{"[1,2,3,4,]", NULL}, |
|
{"{\"a\": 1, }", NULL}, |
|
/* Testing after-ending characters. */ |
|
{"{}x", NULL}, |
|
/* Testing having a key syntax in an array. */ |
|
{"[\"x\":0]", NULL}, |
|
/* Testing invalid numbers. */ |
|
{"1.", NULL}, |
|
{"1e", NULL}, |
|
{".12", NULL}, |
|
{"1.x", NULL}, |
|
{"1.12x", NULL}, |
|
{"1ex", NULL}, |
|
{"1e12x", NULL}, |
|
{".12x", NULL}, |
|
{"000", NULL}, |
|
}; |
|
|
|
static void test_pairs() { |
|
unsigned i; |
|
|
|
for (i = 0; i < GPR_ARRAY_SIZE(testing_pairs); i++) { |
|
testing_pair *pair = testing_pairs + i; |
|
char *scratchpad = gpr_strdup(pair->input); |
|
grpc_json *json; |
|
|
|
gpr_log(GPR_INFO, "parsing string %i - should %s", i, |
|
pair->output ? "succeed" : "fail"); |
|
json = grpc_json_parse_string(scratchpad); |
|
|
|
if (pair->output) { |
|
char *output; |
|
|
|
GPR_ASSERT(json); |
|
output = grpc_json_dump_to_string(json, 0); |
|
GPR_ASSERT(output); |
|
gpr_log(GPR_INFO, "succeeded with output = %s", output); |
|
GPR_ASSERT(strcmp(output, pair->output) == 0); |
|
|
|
grpc_json_destroy(json); |
|
gpr_free(output); |
|
} else { |
|
gpr_log(GPR_INFO, "failed"); |
|
GPR_ASSERT(!json); |
|
} |
|
|
|
gpr_free(scratchpad); |
|
} |
|
} |
|
|
|
static void test_atypical() { |
|
char *scratchpad = gpr_strdup("[[],[],[]]"); |
|
grpc_json *json = grpc_json_parse_string(scratchpad); |
|
grpc_json *brother; |
|
|
|
GPR_ASSERT(json); |
|
GPR_ASSERT(json->child); |
|
brother = json->child->next; |
|
grpc_json_destroy(json->child); |
|
GPR_ASSERT(json->child == brother); |
|
grpc_json_destroy(json->child->next); |
|
grpc_json_destroy(json); |
|
gpr_free(scratchpad); |
|
} |
|
|
|
int main(int argc, char **argv) { |
|
grpc_test_init(argc, argv); |
|
test_pairs(); |
|
test_atypical(); |
|
gpr_log(GPR_INFO, "json_test success"); |
|
return 0; |
|
}
|
|
|