|
|
|
/*
|
|
|
|
*
|
|
|
|
* Copyright 2014, 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/useful.h>
|
|
|
|
#include <grpc/support/log.h>
|
|
|
|
#include "src/core/json/json.h"
|
|
|
|
#include "src/core/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" },
|
|
|
|
{ " \"a\" ", "\"a\"" },
|
|
|
|
{ " true ", "true" },
|
|
|
|
/* Testing the parser's ability to decode trivial UTF-16. */
|
|
|
|
{ "\"\\u0020\\\\\\u0010\\u000a\\u000D\"", "\" \\\\\\u0010\\n\\r\"" },
|
|
|
|
/* Testing UTF-8 character "𝄞", U+11D1E. */
|
|
|
|
{ "\"\xf0\x9d\x84\x9e\"", "\"\\ud834\\udd1e\"" },
|
|
|
|
{ "\"\\ud834\\udd1e\"", "\"\\ud834\\udd1e\"" },
|
|
|
|
/* Testing nested empty containers. */
|
|
|
|
{ " [ [ ] , { } , [ ] ] ", "[[],{},[]]", },
|
|
|
|
/* Testing escapes and control chars in key strings. */
|
|
|
|
{ " { \"\\n\\\\a , b\": [] } ", "{\"\\n\\\\a , b\":[]}" },
|
|
|
|
/* 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 },
|
|
|
|
{ "fals", NULL },
|
|
|
|
/* Testing unterminated string. */
|
|
|
|
{ "\"\\x", NULL },
|
|
|
|
/* Testing invalid UTF-16 number. */
|
|
|
|
{ "\"\\u123x", NULL },
|
|
|
|
/* Testing imbalanced surrogate pairs. */
|
|
|
|
{ "\"\\ud834f", NULL },
|
|
|
|
{ "\"\\ud834\\n", NULL },
|
|
|
|
{ "\"\\udd1ef", NULL },
|
|
|
|
{ "\"\\ud834\\ud834\"", NULL },
|
|
|
|
{ "\"\\ud834\\u1234\"", 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 trailing comma. */
|
|
|
|
{ "{,}", NULL },
|
|
|
|
{ "[1,2,3,4,]", 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() {
|
|
|
|
int 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);
|
|
|
|
}
|
|
|
|
|
|
|
|
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);
|
|
|
|
json->child = brother;
|
|
|
|
grpc_json_destroy(json);
|
|
|
|
}
|
|
|
|
|
|
|
|
int main(int argc, char **argv) {
|
|
|
|
grpc_test_init(argc, argv);
|
|
|
|
test_pairs();
|
|
|
|
test_atypical();
|
|
|
|
gpr_log(GPR_INFO, "json_test success");
|
|
|
|
return 0;
|
|
|
|
}
|