2
0
mirror of https://github.com/openvswitch/ovs synced 2025-10-23 14:57:06 +00:00
Files
openvswitch/tests/test-json.c
Ben Pfaff 1600fa6853 json: New function json_serialized_length().
This will be used for the first time in an upcoming commit.

Signed-off-by: Ben Pfaff <blp@nicira.com>
2013-04-01 13:16:59 -07:00

160 lines
3.8 KiB
C

/*
* Copyright (c) 2009, 2010, 2013 Nicira, Inc.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at:
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include <config.h>
#include "json.h"
#include <ctype.h>
#include <errno.h>
#include <getopt.h>
#include <stdio.h>
#include "util.h"
/* --pretty: If set, the JSON output is pretty-printed, instead of printed as
* compactly as possible. */
static int pretty = 0;
/* --multiple: If set, the input is a sequence of JSON objects or arrays,
* instead of exactly one object or array. */
static int multiple = 0;
static bool
print_and_free_json(struct json *json)
{
bool ok;
if (json->type == JSON_STRING) {
printf("error: %s\n", json->u.string);
ok = false;
} else {
char *s = json_to_string(json, JSSF_SORT | (pretty ? JSSF_PRETTY : 0));
ovs_assert(pretty || json_serialized_length(json) == strlen(s));
puts(s);
free(s);
ok = true;
}
json_destroy(json);
return ok;
}
static bool
refill(FILE *file, void *buffer, size_t buffer_size, size_t *n, size_t *used)
{
*used = 0;
if (feof(file)) {
*n = 0;
return false;
} else {
*n = fread(buffer, 1, buffer_size, file);
if (ferror(file)) {
ovs_fatal(errno, "Error reading input file");
}
return *n > 0;
}
}
static bool
parse_multiple(FILE *stream)
{
struct json_parser *parser;
char buffer[BUFSIZ];
size_t n, used;
bool ok;
parser = NULL;
n = used = 0;
ok = true;
while (used < n || refill(stream, buffer, sizeof buffer, &n, &used)) {
if (!parser && isspace((unsigned char) buffer[used])) {
/* Skip white space. */
used++;
} else {
if (!parser) {
parser = json_parser_create(0);
}
used += json_parser_feed(parser, &buffer[used], n - used);
if (used < n) {
if (!print_and_free_json(json_parser_finish(parser))) {
ok = false;
}
parser = NULL;
}
}
}
if (parser) {
if (!print_and_free_json(json_parser_finish(parser))) {
ok = false;
}
}
return ok;
}
int
main(int argc, char *argv[])
{
const char *input_file;
FILE *stream;
bool ok;
set_program_name(argv[0]);
for (;;) {
static const struct option options[] = {
{"pretty", no_argument, &pretty, 1},
{"multiple", no_argument, &multiple, 1},
};
int option_index = 0;
int c = getopt_long (argc, argv, "", options, &option_index);
if (c == -1) {
break;
}
switch (c) {
case 0:
break;
case '?':
exit(1);
default:
abort();
}
}
if (argc - optind != 1) {
ovs_fatal(0, "usage: %s [--pretty] [--multiple] INPUT.json",
program_name);
}
input_file = argv[optind];
stream = !strcmp(input_file, "-") ? stdin : fopen(input_file, "r");
if (!stream) {
ovs_fatal(errno, "Cannot open \"%s\"", input_file);
}
if (multiple) {
ok = parse_multiple(stream);
} else {
ok = print_and_free_json(json_from_stream(stream));
}
fclose(stream);
return !ok;
}