tippecanoe/json.c

328 lines
6.3 KiB
C
Raw Normal View History

2014-02-05 23:16:23 +00:00
#include <stdio.h>
#include <stdlib.h>
2014-02-05 23:33:43 +00:00
#include <ctype.h>
2014-02-06 01:21:19 +00:00
#include <stdarg.h>
2014-02-05 23:16:23 +00:00
typedef enum json_type {
2014-02-05 23:33:43 +00:00
JSON_STRING, JSON_NUMBER, JSON_ARRAY, JSON_HASH, JSON_NULL, JSON_TRUE, JSON_FALSE,
2014-02-05 23:16:23 +00:00
} json_type;
typedef struct json_array {
struct json_object *object;
struct json_array *next;
} json_array;
typedef struct json_hash {
struct json_object *key;
struct json_object *value;
struct json_hash *next;
} json_hash;
typedef struct json_object {
json_type type;
struct json_object *parent;
char *string;
double number;
json_array *array;
2014-02-06 00:10:22 +00:00
json_hash *hash;
2014-02-05 23:16:23 +00:00
} json_object;
2014-02-05 23:33:43 +00:00
2014-02-06 01:21:19 +00:00
void json_print(json_object *j);
2014-02-06 00:10:22 +00:00
static json_object *add_object(json_type type, json_object *parent) {
2014-02-05 23:33:43 +00:00
json_object *o = malloc(sizeof(struct json_object));
o->type = type;
o->parent = parent;
2014-02-06 00:10:22 +00:00
o->array = NULL;
o->hash = NULL;
if (parent != NULL) {
if (parent->type == JSON_ARRAY) {
json_array *a = malloc(sizeof(json_array));
a->next = parent->array;
a->object = o;
parent->array = a;
}
if (parent->type == JSON_HASH) {
if (parent->hash != NULL && parent->hash->value == NULL) {
parent->hash->value = o;
} else {
json_hash *h = malloc(sizeof(json_hash));
h->next = parent->hash;
h->key = o;
parent->hash = h;
}
}
}
2014-02-05 23:33:43 +00:00
return o;
}
2014-02-06 01:21:19 +00:00
static void json_error(char *s, ...) {
va_list ap;
va_start(ap, s);
vfprintf(stderr, s, ap);
va_end(ap);
2014-02-05 23:33:43 +00:00
exit(EXIT_FAILURE);
}
int peek(FILE *f) {
int c = getc(f);
ungetc(c, f);
return c;
}
struct string {
char *buf;
int n;
int nalloc;
};
static void string_init(struct string *s) {
s->nalloc = 500;
s->buf = malloc(s->nalloc);
s->n = 0;
s->buf[0] = '\0';
}
static void string_append(struct string *s, char c) {
if (s->n + 2 >= s->nalloc) {
s->nalloc += 500;
s->buf = realloc(s->buf, s->nalloc);
}
s->buf[s->n++] = c;
s->buf[s->n] = '\0';
}
static void string_free(struct string *s) {
free(s->buf);
}
2014-02-06 00:51:28 +00:00
json_object *json_parse(FILE *f, json_object *current) {
2014-02-05 23:33:43 +00:00
int c = getc(f);
if (c == EOF) {
return NULL;
}
2014-02-05 23:42:15 +00:00
while (c == ' ' || c == '\t' || c == '\r' || c == '\n') {
2014-02-05 23:33:43 +00:00
c = getc(f);
if (c == EOF) {
return NULL;
}
}
if (c == '[') {
2014-02-06 00:51:28 +00:00
return json_parse(f, add_object(JSON_ARRAY, current));
2014-02-05 23:33:43 +00:00
} else if (c == ']') {
if (current->parent == NULL || current->parent->type != JSON_ARRAY) {
2014-02-06 01:21:19 +00:00
json_error("] without [\n");
2014-02-05 23:33:43 +00:00
}
return current->parent;
}
if (c == '{') {
2014-02-06 00:51:28 +00:00
return json_parse(f, add_object(JSON_HASH, current));
2014-02-05 23:33:43 +00:00
} else if (c == '}') {
if (current->parent == NULL || current->parent->type != JSON_HASH) {
2014-02-06 01:21:19 +00:00
json_error("} without {\n");
2014-02-05 23:33:43 +00:00
}
return current->parent;
}
2014-02-05 23:42:15 +00:00
if (c == 'n') {
if (getc(f) != 'u' || getc(f) != 'l' || getc(f) != 'l') {
2014-02-06 01:21:19 +00:00
json_error("misspelled null\n");
2014-02-05 23:42:15 +00:00
}
2014-02-06 00:10:22 +00:00
return add_object(JSON_NULL, current);
2014-02-05 23:42:15 +00:00
}
if (c == 't') {
if (getc(f) != 'r' || getc(f) != 'u' || getc(f) != 'e') {
2014-02-06 01:21:19 +00:00
json_error("misspelled true\n");
2014-02-05 23:42:15 +00:00
}
2014-02-06 00:10:22 +00:00
return add_object(JSON_TRUE, current);
2014-02-05 23:42:15 +00:00
}
if (c == 'f') {
if (getc(f) != 'a' || getc(f) != 'l' || getc(f) != 's' || getc(f) != 'e') {
2014-02-06 01:21:19 +00:00
json_error("misspelled false\n");
2014-02-05 23:42:15 +00:00
}
2014-02-06 00:10:22 +00:00
return add_object(JSON_FALSE, current);
2014-02-05 23:42:15 +00:00
}
2014-02-06 00:00:01 +00:00
if (c == ',') {
if (current->parent == NULL ||
2014-02-06 01:21:19 +00:00
(current->parent->type != JSON_ARRAY &&
2014-02-06 00:00:01 +00:00
current->parent->type != JSON_HASH)) {
2014-02-06 01:21:19 +00:00
json_error(", not in array or hash: %d\n", current->parent->type);
2014-02-06 00:00:01 +00:00
}
2014-02-06 01:21:19 +00:00
return json_parse(f, current->parent);
2014-02-06 00:00:01 +00:00
}
if (c == ':') {
if (current->parent == NULL || current->parent->type != JSON_HASH) {
2014-02-06 01:21:19 +00:00
json_error(": not in hash: %d\n", current->parent);
2014-02-06 00:00:01 +00:00
}
2014-02-06 00:10:22 +00:00
if (current->parent->hash == NULL || current->parent->hash->value != NULL) {
2014-02-06 01:21:19 +00:00
json_error(": without key: %d\n", current->parent);
2014-02-06 00:10:22 +00:00
}
2014-02-06 00:00:01 +00:00
2014-02-06 01:21:19 +00:00
return json_parse(f, current->parent);
2014-02-06 00:00:01 +00:00
}
if (c == '-' || (c >= '0' && c <= '9')) {
struct string val;
string_init(&val);
string_append(&val, c);
if (c >= '1' && c <= '9') {
c = peek(f);
while (c >= '0' && c <= '9') {
string_append(&val, getc(f));
c = peek(f);
}
}
if (peek(f) == '.') {
string_append(&val, getc(f));
c = peek(f);
while (c >= '0' && c <= '9') {
string_append(&val, getc(f));
c = peek(f);
}
}
c = peek(f);
if (c == 'e' || c == 'E') {
string_append(&val, getc(f));
c = peek(f);
if (c == '+' || c == '-') {
string_append(&val, getc(f));
}
c = peek(f);
while (c >= '0' && c <= '9') {
string_append(&val, getc(f));
c = peek(f);
}
}
json_object *n = add_object(JSON_NUMBER, current);
n->number = atof(val.buf);
string_free(&val);
return n;
}
2014-02-06 00:45:20 +00:00
if (c == '"') {
struct string val;
string_init(&val);
while ((c = getc(f)) != EOF) {
if (c == '"') {
break;
} else if (c == '\\') {
c = getc(f);
if (c == '"') {
string_append(&val, '"');
} else if (c == '\\') {
string_append(&val, '\\');
} else if (c == '/') {
string_append(&val, '/');
} else if (c == 'b') {
string_append(&val, '\b');
} else if (c == 'f') {
string_append(&val, '\f');
} else if (c == 'n') {
string_append(&val, '\n');
} else if (c == 'r') {
string_append(&val, '\r');
} else if (c == 't') {
string_append(&val, '\t');
} else if (c == 'u') {
/* XXX */
} else {
2014-02-06 01:21:19 +00:00
json_error("unknown string escape %c\n", c);
2014-02-06 00:45:20 +00:00
}
} else {
string_append(&val, c);
}
}
json_object *s = add_object(JSON_STRING, current);
s->string = val.buf;
return s;
}
2014-02-06 01:21:19 +00:00
json_error("unrecognized character %c\n", c);
2014-02-05 23:33:43 +00:00
}
2014-02-06 00:51:28 +00:00
void json_print(json_object *j) {
2014-02-06 01:21:19 +00:00
if (j == NULL) {
printf("NULL");
} else if (j->type == JSON_STRING) {
printf("\"%s\"", j->string);
2014-02-06 00:51:28 +00:00
} else if (j->type == JSON_NUMBER) {
2014-02-06 01:21:19 +00:00
printf("%f", j->number);
2014-02-06 00:51:28 +00:00
} else if (j->type == JSON_NULL) {
2014-02-06 01:21:19 +00:00
printf("null");
2014-02-06 00:51:28 +00:00
} else if (j->type == JSON_TRUE) {
2014-02-06 01:21:19 +00:00
printf("true");
2014-02-06 00:51:28 +00:00
} else if (j->type == JSON_FALSE) {
2014-02-06 01:21:19 +00:00
printf("false");
2014-02-06 00:51:28 +00:00
} else if (j->type == JSON_HASH) {
2014-02-06 01:21:19 +00:00
printf("{");
json_hash *h = j->hash;
while (h != NULL) {
json_print(h->key);
printf(":");
json_print(h->value);
printf(",");
h = h->next;
}
printf("}");
2014-02-06 00:51:28 +00:00
} else if (j->type == JSON_ARRAY) {
2014-02-06 01:21:19 +00:00
printf("[");
json_array *a = j->array;
while (a != NULL) {
json_print(a->object);
printf(",");
a = a->next;
}
printf("]");
2014-02-06 00:51:28 +00:00
} else {
2014-02-06 01:21:19 +00:00
printf("what type? %d", j->type);
2014-02-06 00:51:28 +00:00
}
}
int main() {
json_object *j = NULL;
while ((j = json_parse(stdin, j)) != NULL) {
2014-02-06 01:21:19 +00:00
#if 0
2014-02-06 00:51:28 +00:00
if (j->parent != NULL) {
printf("parent: ");
json_print(j->parent);
}
2014-02-06 01:21:19 +00:00
#endif
2014-02-06 00:51:28 +00:00
json_print(j);
printf("\n");
}
}