summaryrefslogtreecommitdiffhomepage
path: root/src/nxt_conf_json.c
diff options
context:
space:
mode:
Diffstat (limited to 'src/nxt_conf_json.c')
-rw-r--r--src/nxt_conf_json.c1888
1 files changed, 0 insertions, 1888 deletions
diff --git a/src/nxt_conf_json.c b/src/nxt_conf_json.c
deleted file mode 100644
index 45682245..00000000
--- a/src/nxt_conf_json.c
+++ /dev/null
@@ -1,1888 +0,0 @@
-
-/*
- * Copyright (C) Igor Sysoev
- * Copyright (C) Valentin V. Bartenev
- * Copyright (C) NGINX, Inc.
- */
-
-#include <nxt_main.h>
-#include <nxt_conf.h>
-#if 0
-#include <math.h>
-#include <float.h>
-#endif
-
-
-#define NXT_CONF_JSON_STR_SIZE 14
-
-
-typedef enum {
- NXT_CONF_JSON_NULL = 0,
- NXT_CONF_JSON_BOOLEAN,
- NXT_CONF_JSON_INTEGER,
- NXT_CONF_JSON_NUMBER,
- NXT_CONF_JSON_SHORT_STRING,
- NXT_CONF_JSON_STRING,
- NXT_CONF_JSON_ARRAY,
- NXT_CONF_JSON_OBJECT,
-} nxt_conf_json_type_t;
-
-
-typedef enum {
- NXT_CONF_JSON_OP_PASS = 0,
- NXT_CONF_JSON_OP_CREATE,
- NXT_CONF_JSON_OP_REPLACE,
- NXT_CONF_JSON_OP_DELETE,
-} nxt_conf_json_op_action_t;
-
-
-typedef struct nxt_conf_json_array_s nxt_conf_json_array_t;
-typedef struct nxt_conf_json_object_s nxt_conf_json_object_t;
-
-
-struct nxt_conf_json_value_s {
- union {
- uint32_t boolean; /* 1 bit. */
- int64_t integer;
- double number;
- u_char str[1 + NXT_CONF_JSON_STR_SIZE];
- nxt_str_t *string;
- nxt_conf_json_array_t *array;
- nxt_conf_json_object_t *object;
- } u;
-
- nxt_conf_json_type_t type:8; /* 3 bits. */
-};
-
-
-struct nxt_conf_json_array_s {
- nxt_uint_t count;
- nxt_conf_json_value_t elements[];
-};
-
-
-typedef struct {
- nxt_conf_json_value_t name;
- nxt_conf_json_value_t value;
-} nxt_conf_json_obj_member_t;
-
-
-struct nxt_conf_json_object_s {
- nxt_uint_t count;
- nxt_conf_json_obj_member_t members[];
-};
-
-
-struct nxt_conf_json_op_s {
- uint32_t index;
- uint32_t action; /* nxt_conf_json_op_action_t */
- void *ctx;
- nxt_conf_json_op_t *next;
-};
-
-
-static u_char *nxt_conf_json_skip_space(u_char *start, u_char *end);
-static u_char *nxt_conf_json_parse_value(nxt_mp_t *mp,
- nxt_conf_json_value_t *value, u_char *start, u_char *end);
-static u_char *nxt_conf_json_parse_object(nxt_mp_t *mp,
- nxt_conf_json_value_t *value, u_char *start, u_char *end);
-static nxt_int_t nxt_conf_json_object_hash_add(nxt_mp_t *mp,
- nxt_lvlhsh_t *lvlhsh, nxt_conf_json_obj_member_t *member);
-static nxt_int_t nxt_conf_json_object_hash_test(nxt_lvlhsh_query_t *lhq,
- void *data);
-static void *nxt_conf_json_object_hash_alloc(void *data, size_t size);
-static void nxt_conf_json_object_hash_free(void *data, void *p);
-static u_char *nxt_conf_json_parse_array(nxt_mp_t *mp,
- nxt_conf_json_value_t *value, u_char *start, u_char *end);
-static u_char *nxt_conf_json_parse_string(nxt_mp_t *mp,
- nxt_conf_json_value_t *value, u_char *start, u_char *end);
-static u_char *nxt_conf_json_parse_number(nxt_mp_t *mp,
- nxt_conf_json_value_t *value, u_char *start, u_char *end);
-
-static nxt_int_t nxt_conf_json_copy_value(nxt_mp_t *mp, nxt_conf_json_op_t *op,
- nxt_conf_json_value_t *dst, nxt_conf_json_value_t *src);
-static nxt_int_t nxt_conf_json_copy_object(nxt_mp_t *mp, nxt_conf_json_op_t *op,
- nxt_conf_json_value_t *dst, nxt_conf_json_value_t *src);
-
-static size_t nxt_conf_json_integer_length(nxt_conf_json_value_t *value);
-static u_char *nxt_conf_json_integer_print(u_char *p,
- nxt_conf_json_value_t *value);
-static size_t nxt_conf_json_string_length(nxt_conf_json_value_t *value);
-static u_char *nxt_conf_json_string_print(u_char *p,
- nxt_conf_json_value_t *value);
-static size_t nxt_conf_json_array_length(nxt_conf_json_value_t *value,
- nxt_conf_json_pretty_t *pretty);
-static u_char *nxt_conf_json_array_print(u_char *p,
- nxt_conf_json_value_t *value, nxt_conf_json_pretty_t *pretty);
-static size_t nxt_conf_json_object_length(nxt_conf_json_value_t *value,
- nxt_conf_json_pretty_t *pretty);
-static u_char *nxt_conf_json_object_print(u_char *p,
- nxt_conf_json_value_t *value, nxt_conf_json_pretty_t *pretty);
-
-static size_t nxt_conf_json_escape_length(u_char *p, size_t size);
-static u_char *nxt_conf_json_escape(u_char *dst, u_char *src, size_t size);
-
-
-#define nxt_conf_json_newline(p) \
- ((p)[0] = '\r', (p)[1] = '\n', (p) + 2)
-
-
-nxt_inline u_char *
-nxt_conf_json_indentation(u_char *p, uint32_t level)
-{
- while (level) {
- *p++ = '\t';
- level--;
- }
-
- return p;
-}
-
-
-nxt_inline void
-nxt_conf_json_value_get_string(nxt_conf_json_value_t *value, nxt_str_t *str)
-{
- if (value->type == NXT_CONF_JSON_SHORT_STRING) {
- str->length = value->u.str[0];
- str->start = &value->u.str[1];
-
- } else {
- *str = *value->u.string;
- }
-}
-
-
-typedef struct {
- u_char *start;
- u_char *end;
- nxt_bool_t last;
-} nxt_conf_path_parse_t;
-
-
-static void nxt_conf_json_path_next_token(nxt_conf_path_parse_t *parse,
- nxt_str_t *token);
-
-
-nxt_conf_json_value_t *
-nxt_conf_json_get_value(nxt_conf_json_value_t *value, nxt_str_t *path)
-{
- nxt_str_t token;
- nxt_conf_path_parse_t parse;
-
- parse.start = path->start;
- parse.end = path->start + path->length;
- parse.last = 0;
-
- do {
- nxt_conf_json_path_next_token(&parse, &token);
-
- if (token.length == 0) {
-
- if (parse.last) {
- break;
- }
-
- return NULL;
- }
-
- value = nxt_conf_json_object_get_member(value, &token, NULL);
-
- if (value == NULL) {
- return NULL;
- }
-
- } while (parse.last == 0);
-
- return value;
-}
-
-
-static void
-nxt_conf_json_path_next_token(nxt_conf_path_parse_t *parse, nxt_str_t *token)
-{
- u_char *p, *end;
-
- end = parse->end;
- p = parse->start + 1;
-
- token->start = p;
-
- while (p < end && *p != '/') {
- p++;
- }
-
- parse->start = p;
- parse->last = (p >= end);
-
- token->length = p - token->start;
-}
-
-
-nxt_conf_json_value_t *
-nxt_conf_json_object_get_member(nxt_conf_json_value_t *value, nxt_str_t *name,
- uint32_t *index)
-{
- nxt_str_t str;
- nxt_uint_t n;
- nxt_conf_json_object_t *object;
- nxt_conf_json_obj_member_t *member;
-
- if (value->type != NXT_CONF_JSON_OBJECT) {
- return NULL;
- }
-
- object = value->u.object;
-
- for (n = 0; n < object->count; n++) {
- member = &object->members[n];
-
- nxt_conf_json_value_get_string(&member->name, &str);
-
- if (nxt_strstr_eq(&str, name)) {
-
- if (index != NULL) {
- *index = n;
- }
-
- return &member->value;
- }
- }
-
- return NULL;
-}
-
-
-nxt_int_t
-nxt_conf_json_object_map(nxt_conf_json_value_t *value,
- nxt_conf_json_object_map_t *map, void *data)
-{
- nxt_uint_t i;
- nxt_conf_json_value_t *v;
-
- union {
- uint8_t ui8;
- int32_t i32;
- int64_t i64;
- nxt_int_t i;
- ssize_t size;
- off_t off;
- double dbl;
- nxt_str_t str;
- void *v;
- } *ptr;
-
- for (i = 0; map[i].name.length != 0; i++) {
-
- v = nxt_conf_json_object_get_member(value, &map[i].name, NULL);
-
- if (v == NULL || v->type == NXT_CONF_JSON_NULL) {
- continue;
- }
-
- ptr = nxt_pointer_to(data, map[i].offset);
-
- switch (map[i].type) {
-
- case NXT_CONF_JSON_MAP_INT8:
-
- if (v->type != NXT_CONF_JSON_BOOLEAN) {
- return NXT_ERROR;
- }
-
- ptr->ui8 = v->u.boolean;
-
- break;
-
- case NXT_CONF_JSON_MAP_INT32:
- case NXT_CONF_JSON_MAP_INT64:
- case NXT_CONF_JSON_MAP_INT:
- case NXT_CONF_JSON_MAP_SIZE:
- case NXT_CONF_JSON_MAP_OFF:
-
- if (v->type != NXT_CONF_JSON_INTEGER) {
- return NXT_ERROR;
- }
-
- switch (map[i].type) {
-
- case NXT_CONF_JSON_MAP_INT32:
- ptr->ui8 = v->u.integer;
- break;
-
- case NXT_CONF_JSON_MAP_INT64:
- ptr->i64 = v->u.integer;
- break;
-
- case NXT_CONF_JSON_MAP_INT:
- ptr->i = v->u.integer;
- break;
-
- case NXT_CONF_JSON_MAP_SIZE:
- ptr->size = v->u.integer;
- break;
-
- case NXT_CONF_JSON_MAP_OFF:
- ptr->off = v->u.integer;
- break;
-
- default:
- nxt_unreachable();
- }
-
- break;
-
- case NXT_CONF_JSON_MAP_DOUBLE:
-
- if (v->type == NXT_CONF_JSON_NUMBER) {
- ptr->dbl = v->u.number;
-
- } else if (v->type == NXT_CONF_JSON_INTEGER) {
- ptr->dbl = v->u.integer;
-
- } else {
- return NXT_ERROR;
- }
-
- break;
-
- case NXT_CONF_JSON_MAP_STR:
-
- if (v->type != NXT_CONF_JSON_SHORT_STRING
- && v->type != NXT_CONF_JSON_STRING)
- {
- return NXT_ERROR;
- }
-
- nxt_conf_json_value_get_string(v, &ptr->str);
-
- break;
-
-
- case NXT_CONF_JSON_MAP_PTR:
-
- ptr->v = v;
-
- break;
- }
- }
-
- return NXT_OK;
-}
-
-
-nxt_conf_json_value_t *
-nxt_conf_json_object_next_member(nxt_conf_json_value_t *value, nxt_str_t *name,
- uint32_t *next)
-{
- uint32_t n;
- nxt_conf_json_object_t *object;
- nxt_conf_json_obj_member_t *member;
-
- if (value->type != NXT_CONF_JSON_OBJECT) {
- return NULL;
- }
-
- n = *next;
- object = value->u.object;
-
- if (n >= object->count) {
- return NULL;
- }
-
- member = &object->members[n];
- *next = n + 1;
-
- nxt_conf_json_value_get_string(&member->name, name);
-
- return &member->value;
-}
-
-
-nxt_int_t
-nxt_conf_json_op_compile(nxt_mp_t *mp, nxt_conf_json_op_t **ops,
- nxt_conf_json_value_t *root, nxt_str_t *path,
- nxt_conf_json_value_t *value)
-{
- nxt_str_t token;
- nxt_conf_json_op_t *op, **parent;
- nxt_conf_path_parse_t parse;
- nxt_conf_json_obj_member_t *member;
-
- parse.start = path->start;
- parse.end = path->start + path->length;
- parse.last = 0;
-
- parent = ops;
-
- for ( ;; ) {
- op = nxt_mp_zget(mp, sizeof(nxt_conf_json_op_t));
- if (nxt_slow_path(op == NULL)) {
- return NXT_ERROR;
- }
-
- *parent = op;
- parent = (nxt_conf_json_op_t **) &op->ctx;
-
- nxt_conf_json_path_next_token(&parse, &token);
-
- root = nxt_conf_json_object_get_member(root, &token, &op->index);
-
- if (parse.last) {
- break;
- }
-
- if (root == NULL) {
- return NXT_DECLINED;
- }
-
- op->action = NXT_CONF_JSON_OP_PASS;
- }
-
- if (value == NULL) {
-
- if (root == NULL) {
- return NXT_DECLINED;
- }
-
- op->action = NXT_CONF_JSON_OP_DELETE;
-
- return NXT_OK;
- }
-
- if (root == NULL) {
-
- member = nxt_mp_zget(mp, sizeof(nxt_conf_json_obj_member_t));
- if (nxt_slow_path(member == NULL)) {
- return NXT_ERROR;
- }
-
- if (token.length > NXT_CONF_JSON_STR_SIZE) {
-
- member->name.u.string = nxt_mp_get(mp, sizeof(nxt_str_t));
- if (nxt_slow_path(member->name.u.string == NULL)) {
- return NXT_ERROR;
- }
-
- *member->name.u.string = token;
- member->name.type = NXT_CONF_JSON_STRING;
-
- } else {
- member->name.u.str[0] = token.length;
- nxt_memcpy(&member->name.u.str[1], token.start, token.length);
-
- member->name.type = NXT_CONF_JSON_SHORT_STRING;
- }
-
- member->value = *value;
-
- op->action = NXT_CONF_JSON_OP_CREATE;
- op->ctx = member;
-
- } else {
- op->action = NXT_CONF_JSON_OP_REPLACE;
- op->ctx = value;
- }
-
- return NXT_OK;
-}
-
-
-nxt_conf_json_value_t *
-nxt_conf_json_clone_value(nxt_mp_t *mp, nxt_conf_json_op_t *op,
- nxt_conf_json_value_t *value)
-{
- nxt_int_t rc;
- nxt_conf_json_value_t *copy;
-
- copy = nxt_mp_get(mp, sizeof(nxt_conf_json_value_t));
- if (nxt_slow_path(copy == NULL)) {
- return NULL;
- }
-
- rc = nxt_conf_json_copy_value(mp, op, copy, value);
-
- if (nxt_slow_path(rc != NXT_OK)) {
- return NULL;
- }
-
- return copy;
-}
-
-
-static nxt_int_t
-nxt_conf_json_copy_value(nxt_mp_t *mp, nxt_conf_json_op_t *op,
- nxt_conf_json_value_t *dst, nxt_conf_json_value_t *src)
-{
- size_t size;
- nxt_int_t rc;
- nxt_uint_t n;
-
- if (op != NULL && src->type != NXT_CONF_JSON_OBJECT) {
- return NXT_ERROR;
- }
-
- dst->type = src->type;
-
- switch (src->type) {
-
- case NXT_CONF_JSON_STRING:
-
- dst->u.string = nxt_str_dup(mp, NULL, src->u.string);
-
- if (nxt_slow_path(dst->u.string == NULL)) {
- return NXT_ERROR;
- }
-
- break;
-
- case NXT_CONF_JSON_ARRAY:
-
- size = sizeof(nxt_conf_json_array_t)
- + src->u.array->count * sizeof(nxt_conf_json_value_t);
-
- dst->u.array = nxt_mp_get(mp, size);
- if (nxt_slow_path(dst->u.array == NULL)) {
- return NXT_ERROR;
- }
-
- dst->u.array->count = src->u.array->count;
-
- for (n = 0; n < src->u.array->count; n++) {
- rc = nxt_conf_json_copy_value(mp, NULL,
- &dst->u.array->elements[n],
- &src->u.array->elements[n]);
-
- if (nxt_slow_path(rc != NXT_OK)) {
- return NXT_ERROR;
- }
- }
-
- break;
-
- case NXT_CONF_JSON_OBJECT:
- return nxt_conf_json_copy_object(mp, op, dst, src);
-
- default:
- dst->u = src->u;
- }
-
- return NXT_OK;
-}
-
-
-static nxt_int_t
-nxt_conf_json_copy_object(nxt_mp_t *mp, nxt_conf_json_op_t *op,
- nxt_conf_json_value_t *dst, nxt_conf_json_value_t *src)
-{
- size_t size;
- nxt_int_t rc;
- nxt_uint_t s, d, count, index;
- nxt_conf_json_op_t *pass_op;
- nxt_conf_json_value_t *value;
- nxt_conf_json_obj_member_t *member;
-
- count = src->u.object->count;
-
- if (op != NULL) {
- if (op->action == NXT_CONF_JSON_OP_CREATE) {
- count++;
-
- } else if (op->action == NXT_CONF_JSON_OP_DELETE) {
- count--;
- }
- }
-
- size = sizeof(nxt_conf_json_object_t)
- + count * sizeof(nxt_conf_json_obj_member_t);
-
- dst->u.object = nxt_mp_get(mp, size);
- if (nxt_slow_path(dst->u.object == NULL)) {
- return NXT_ERROR;
- }
-
- dst->u.object->count = count;
-
- s = 0;
- d = 0;
-
- pass_op = NULL;
-
- /*
- * This initialization is needed only to
- * suppress a warning on GCC 4.8 and older.
- */
- index = 0;
-
- do {
- if (pass_op == NULL) {
- index = (op == NULL || op->action == NXT_CONF_JSON_OP_CREATE)
- ? src->u.object->count : op->index;
- }
-
- while (s != index) {
- rc = nxt_conf_json_copy_value(mp, NULL,
- &dst->u.object->members[d].name,
- &src->u.object->members[s].name);
-
- if (nxt_slow_path(rc != NXT_OK)) {
- return NXT_ERROR;
- }
-
- rc = nxt_conf_json_copy_value(mp, pass_op,
- &dst->u.object->members[d].value,
- &src->u.object->members[s].value);
-
- if (nxt_slow_path(rc != NXT_OK)) {
- return NXT_ERROR;
- }
-
- s++;
- d++;
- }
-
- if (pass_op != NULL) {
- pass_op = NULL;
- continue;
- }
-
- if (op != NULL) {
- switch (op->action) {
- case NXT_CONF_JSON_OP_PASS:
- pass_op = op->ctx;
- index++;
- break;
-
- case NXT_CONF_JSON_OP_CREATE:
- member = op->ctx;
-
- rc = nxt_conf_json_copy_value(mp, NULL,
- &dst->u.object->members[d].name,
- &member->name);
-
- if (nxt_slow_path(rc != NXT_OK)) {
- return NXT_ERROR;
- }
-
- dst->u.object->members[d].value = member->value;
-
- d++;
- break;
-
- case NXT_CONF_JSON_OP_REPLACE:
- rc = nxt_conf_json_copy_value(mp, NULL,
- &dst->u.object->members[d].name,
- &src->u.object->members[s].name);
-
- if (nxt_slow_path(rc != NXT_OK)) {
- return NXT_ERROR;
- }
-
- value = op->ctx;
-
- dst->u.object->members[d].value = *value;
-
- s++;
- d++;
- break;
-
- case NXT_CONF_JSON_OP_DELETE:
- s++;
- break;
- }
-
- op = op->next;
- }
-
- } while (d != count);
-
- dst->type = src->type;
-
- return NXT_OK;
-}
-
-
-nxt_conf_json_value_t *
-nxt_conf_json_parse(nxt_mp_t *mp, u_char *start, u_char *end)
-{
- u_char *p;
- nxt_conf_json_value_t *value;
-
- value = nxt_mp_get(mp, sizeof(nxt_conf_json_value_t));
- if (nxt_slow_path(value == NULL)) {
- return NULL;
- }
-
- p = nxt_conf_json_skip_space(start, end);
-
- if (nxt_slow_path(p == end)) {
- return NULL;
- }
-
- p = nxt_conf_json_parse_value(mp, value, p, end);
-
- if (nxt_slow_path(p == NULL)) {
- return NULL;
- }
-
- p = nxt_conf_json_skip_space(p, end);
-
- if (nxt_slow_path(p != end)) {
- return NULL;
- }
-
- return value;
-}
-
-
-static u_char *
-nxt_conf_json_skip_space(u_char *start, u_char *end)
-{
- u_char *p;
-
- for (p = start; nxt_fast_path(p != end); p++) {
-
- switch (*p) {
- case ' ':
- case '\t':
- case '\r':
- case '\n':
- continue;
- }
-
- break;
- }
-
- return p;
-}
-
-
-static u_char *
-nxt_conf_json_parse_value(nxt_mp_t *mp, nxt_conf_json_value_t *value,
- u_char *start, u_char *end)
-{
- u_char ch;
-
- ch = *start;
-
- switch (ch) {
- case '{':
- return nxt_conf_json_parse_object(mp, value, start, end);
-
- case '[':
- return nxt_conf_json_parse_array(mp, value, start, end);
-
- case '"':
- return nxt_conf_json_parse_string(mp, value, start, end);
-
- case 't':
- if (nxt_fast_path(end - start >= 4
- && nxt_memcmp(start, "true", 4) == 0))
- {
- value->u.boolean = 1;
- value->type = NXT_CONF_JSON_BOOLEAN;
-
- return start + 4;
- }
-
- return NULL;
-
- case 'f':
- if (nxt_fast_path(end - start >= 5
- && nxt_memcmp(start, "false", 5) == 0))
- {
- value->u.boolean = 0;
- value->type = NXT_CONF_JSON_BOOLEAN;
-
- return start + 5;
- }
-
- return NULL;
-
- case 'n':
- if (nxt_fast_path(end - start >= 4
- && nxt_memcmp(start, "null", 4) == 0))
- {
- value->type = NXT_CONF_JSON_NULL;
- return start + 4;
- }
-
- return NULL;
- }
-
- if (nxt_fast_path(ch == '-' || (ch - '0') <= 9)) {
- return nxt_conf_json_parse_number(mp, value, start, end);
- }
-
- return NULL;
-}
-
-
-static const nxt_lvlhsh_proto_t nxt_conf_json_object_hash_proto
- nxt_aligned(64) =
-{
- NXT_LVLHSH_DEFAULT,
- nxt_conf_json_object_hash_test,
- nxt_conf_json_object_hash_alloc,
- nxt_conf_json_object_hash_free,
-};
-
-
-static u_char *
-nxt_conf_json_parse_object(nxt_mp_t *mp, nxt_conf_json_value_t *value,
- u_char *start, u_char *end)
-{
- u_char *p;
- nxt_mp_t *mp_temp;
- nxt_int_t rc;
- nxt_uint_t count;
- nxt_lvlhsh_t hash;
- nxt_lvlhsh_each_t lhe;
- nxt_conf_json_object_t *object;
- nxt_conf_json_obj_member_t *member, *element;
-
- p = nxt_conf_json_skip_space(start + 1, end);
-
- if (nxt_slow_path(p == end)) {
- return NULL;
- }
-
- mp_temp = nxt_mp_create(1024, 128, 256, 32);
- if (nxt_slow_path(mp_temp == NULL)) {
- return NULL;
- }
-
- nxt_lvlhsh_init(&hash);
-
- count = 0;
-
- if (*p != '}') {
-
- for ( ;; ) {
- count++;
-
- if (*p != '"') {
- goto error;
- }
-
- member = nxt_mp_get(mp_temp, sizeof(nxt_conf_json_obj_member_t));
- if (nxt_slow_path(member == NULL)) {
- goto error;
- }
-
- p = nxt_conf_json_parse_string(mp, &member->name, p, end);
-
- if (nxt_slow_path(p == NULL)) {
- goto error;
- }
-
- rc = nxt_conf_json_object_hash_add(mp_temp, &hash, member);
-
- if (nxt_slow_path(rc != NXT_OK)) {
- goto error;
- }
-
- p = nxt_conf_json_skip_space(p, end);
-
- if (nxt_slow_path(p == end || *p != ':')) {
- goto error;
- }
-
- p = nxt_conf_json_skip_space(p + 1, end);
-
- if (nxt_slow_path(p == end)) {
- goto error;
- }
-
- p = nxt_conf_json_parse_value(mp, &member->value, p, end);
-
- if (nxt_slow_path(p == NULL)) {
- goto error;
- }
-
- p = nxt_conf_json_skip_space(p, end);
-
- if (nxt_slow_path(p == end)) {
- goto error;
- }
-
- if (*p == '}') {
- break;
- }
-
- if (nxt_slow_path(*p != ',')) {
- goto error;
- }
-
- p = nxt_conf_json_skip_space(p + 1, end);
-
- if (nxt_slow_path(p == end)) {
- goto error;
- }
- }
- }
-
- object = nxt_mp_get(mp, sizeof(nxt_conf_json_object_t)
- + count * sizeof(nxt_conf_json_obj_member_t));
- if (nxt_slow_path(object == NULL)) {
- goto error;
- }
-
- value->u.object = object;
- value->type = NXT_CONF_JSON_OBJECT;
-
- object->count = count;
- member = object->members;
-
- nxt_memzero(&lhe, sizeof(nxt_lvlhsh_each_t));
- lhe.proto = &nxt_conf_json_object_hash_proto;
-
- for ( ;; ) {
- element = nxt_lvlhsh_each(&hash, &lhe);
-
- if (element == NULL) {
- break;
- }
-
- *member++ = *element;
- }
-
- nxt_mp_destroy(mp_temp);
-
- return p + 1;
-
-error:
-
- nxt_mp_destroy(mp_temp);
- return NULL;
-}
-
-
-static nxt_int_t
-nxt_conf_json_object_hash_add(nxt_mp_t *mp, nxt_lvlhsh_t *lvlhsh,
- nxt_conf_json_obj_member_t *member)
-{
- nxt_lvlhsh_query_t lhq;
-
- nxt_conf_json_value_get_string(&member->name, &lhq.key);
-
- lhq.key_hash = nxt_djb_hash(lhq.key.start, lhq.key.length);
- lhq.replace = 0;
- lhq.value = member;
- lhq.proto = &nxt_conf_json_object_hash_proto;
- lhq.pool = mp;
-
- return nxt_lvlhsh_insert(lvlhsh, &lhq);
-}
-
-
-static nxt_int_t
-nxt_conf_json_object_hash_test(nxt_lvlhsh_query_t *lhq, void *data)
-{
- nxt_str_t str;
- nxt_conf_json_obj_member_t *member;
-
- member = data;
-
- nxt_conf_json_value_get_string(&member->name, &str);
-
- if (nxt_strstr_eq(&lhq->key, &str)) {
- return NXT_OK;
- }
-
- return NXT_DECLINED;
-}
-
-
-static void *
-nxt_conf_json_object_hash_alloc(void *data, size_t size)
-{
- return nxt_mp_align(data, size, size);
-}
-
-
-static void
-nxt_conf_json_object_hash_free(void *data, void *p)
-{
- nxt_mp_free(data, p);
-}
-
-
-static u_char *
-nxt_conf_json_parse_array(nxt_mp_t *mp, nxt_conf_json_value_t *value,
- u_char *start, u_char *end)
-{
- u_char *p;
- nxt_mp_t *mp_temp;
- nxt_uint_t count;
- nxt_list_t *list;
- nxt_conf_json_array_t *array;
- nxt_conf_json_value_t *element;
-
- p = nxt_conf_json_skip_space(start + 1, end);
-
- if (nxt_slow_path(p == end)) {
- return NULL;
- }
-
- mp_temp = nxt_mp_create(1024, 128, 256, 32);
- if (nxt_slow_path(mp_temp == NULL)) {
- return NULL;
- }
-
- list = nxt_list_create(mp_temp, 8, sizeof(nxt_conf_json_value_t));
- if (nxt_slow_path(list == NULL)) {
- goto error;
- }
-
- count = 0;
-
- if (*p != ']') {
-
- for ( ;; ) {
- count++;
-
- element = nxt_list_add(list);
- if (nxt_slow_path(element == NULL)) {
- goto error;
- }
-
- p = nxt_conf_json_parse_value(mp, element, p, end);
-
- if (nxt_slow_path(p == NULL)) {
- goto error;
- }
-
- p = nxt_conf_json_skip_space(p, end);
-
- if (nxt_slow_path(p == end)) {
- goto error;
- }
-
- if (*p == ']') {
- break;
- }
-
- if (nxt_slow_path(*p != ',')) {
- goto error;
- }
-
- p = nxt_conf_json_skip_space(p + 1, end);
-
- if (nxt_slow_path(p == end)) {
- goto error;
- }
- }
- }
-
- array = nxt_mp_get(mp, sizeof(nxt_conf_json_array_t)
- + count * sizeof(nxt_conf_json_value_t));
- if (nxt_slow_path(array == NULL)) {
- goto error;
- }
-
- value->u.array = array;
- value->type = NXT_CONF_JSON_ARRAY;
-
- array->count = count;
- element = array->elements;
-
- nxt_list_each(value, list) {
- *element++ = *value;
- } nxt_list_loop;
-
- nxt_mp_destroy(mp_temp);
-
- return p + 1;
-
-error:
-
- nxt_mp_destroy(mp_temp);
- return NULL;
-}
-
-
-static u_char *
-nxt_conf_json_parse_string(nxt_mp_t *mp, nxt_conf_json_value_t *value,
- u_char *start, u_char *end)
-{
- u_char *p, ch, *last, *s;
- size_t size, surplus;
- uint32_t utf, utf_high;
- nxt_uint_t i;
- enum {
- sw_usual = 0,
- sw_escape,
- sw_encoded1,
- sw_encoded2,
- sw_encoded3,
- sw_encoded4,
- } state;
-
- start++;
-
- state = 0;
- surplus = 0;
-
- for (p = start; nxt_fast_path(p != end); p++) {
- ch = *p;
-
- switch (state) {
-
- case sw_usual:
-
- if (ch == '"') {
- break;
- }
-
- if (ch == '\\') {
- state = sw_escape;
- continue;
- }
-
- if (nxt_fast_path(ch >= ' ')) {
- continue;
- }
-
- return NULL;
-
- case sw_escape:
-
- switch (ch) {
- case '"':
- case '\\':
- case '/':
- case 'n':
- case 'r':
- case 't':
- case 'b':
- case 'f':
- surplus++;
- state = sw_usual;
- continue;
-
- case 'u':
- /*
- * Basic unicode 6 bytes "\uXXXX" in JSON
- * and up to 3 bytes in UTF-8.
- *
- * Surrogate pair: 12 bytes "\uXXXX\uXXXX" in JSON
- * and 3 or 4 bytes in UTF-8.
- */
- surplus += 3;
- state = sw_encoded1;
- continue;
- }
-
- return NULL;
-
- case sw_encoded1:
- case sw_encoded2:
- case sw_encoded3:
- case sw_encoded4:
-
- if (nxt_fast_path((ch >= '0' && ch <= '9')
- || (ch >= 'A' && ch <= 'F')))
- {
- state = (state == sw_encoded4) ? sw_usual : state + 1;
- continue;
- }
-
- return NULL;
- }
-
- break;
- }
-
- if (nxt_slow_path(p == end)) {
- return NULL;
- }
-
- /* Points to the ending quote mark. */
- last = p;
-
- size = last - start - surplus;
-
- if (size > NXT_CONF_JSON_STR_SIZE) {
- value->type = NXT_CONF_JSON_STRING;
- value->u.string = nxt_str_alloc(mp, size);
-
- if (nxt_slow_path(value->u.string == NULL)) {
- return NULL;
- }
-
- s = value->u.string->start;
-
- } else {
- value->type = NXT_CONF_JSON_SHORT_STRING;
- value->u.str[0] = size;
-
- s = &value->u.str[1];
- }
-
- if (surplus == 0) {
- nxt_memcpy(s, start, size);
- return last + 1;
- }
-
- p = start;
-
- do {
- ch = *p++;
-
- if (ch != '\\') {
- *s++ = ch;
- continue;
- }
-
- ch = *p++;
-
- switch (ch) {
- case '"':
- case '\\':
- case '/':
- *s++ = ch;
- continue;
-
- case 'n':
- *s++ = '\n';
- continue;
-
- case 'r':
- *s++ = '\r';
- continue;
-
- case 't':
- *s++ = '\t';
- continue;
-
- case 'b':
- *s++ = '\b';
- continue;
-
- case 'f':
- *s++ = '\f';
- continue;
- }
-
- utf = 0;
- utf_high = 0;
-
- for ( ;; ) {
- for (i = 0; i < 4; i++) {
- utf = (utf << 4) + (p[i] - (p[i] >= 'A' ? 'A' : '0'));
- }
-
- p += 4;
-
- if (utf < 0xd800 || utf > 0xdbff || utf_high) {
- break;
- }
-
- utf_high = utf;
- utf = 0;
-
- if (p[0] != '\\' || p[1] != 'u') {
- break;
- }
-
- p += 2;
- }
-
- if (utf_high != 0) {
- if (nxt_slow_path(utf_high < 0xd800
- || utf_high > 0xdbff
- || utf < 0xdc00
- || utf > 0xdfff))
- {
- /* Invalid surrogate pair. */
- return NULL;
- }
-
- utf = ((utf_high - 0xd800) << 10) + (utf - 0xdc00);
- }
-
- s = nxt_utf8_encode(s, utf);
-
- } while (p != last);
-
- if (size > NXT_CONF_JSON_STR_SIZE) {
- value->u.string->length = s - value->u.string->start;
-
- } else {
- value->u.str[0] = s - &value->u.str[1];
- }
-
- return last + 1;
-}
-
-
-static u_char *
-nxt_conf_json_parse_number(nxt_mp_t *mp, nxt_conf_json_value_t *value,
- u_char *start, u_char *end)
-{
- u_char *p, ch;
- uint64_t integer;
- nxt_int_t sign;
-#if 0
- uint64_t frac, power
- nxt_int_t e, negative;
-#endif
-
- static const uint64_t cutoff = NXT_INT64_T_MAX / 10;
- static const uint64_t cutlim = NXT_INT64_T_MAX % 10;
-
- ch = *start;
-
- if (ch == '-') {
- sign = -1;
- start++;
-
- } else {
- sign = 1;
- }
-
- integer = 0;
-
- for (p = start; nxt_fast_path(p != end); p++) {
- ch = *p;
-
- /* Values below '0' become >= 208. */
- ch = ch - '0';
-
- if (ch > 9) {
- break;
- }
-
- if (nxt_slow_path(integer >= cutoff
- && (integer > cutoff || ch > cutlim)))
- {
- return NULL;
- }
-
- integer = integer * 10 + ch;
- }
-
- if (nxt_slow_path(p == start || (p - start > 1 && *start == '0'))) {
- return NULL;
- }
-
- if (ch != '.') {
- value->type = NXT_CONF_JSON_INTEGER;
- value->u.integer = sign * integer;
- return p;
- }
-
-#if 0
- start = p + 1;
-
- frac = 0;
- power = 1;
-
- for (p = start; nxt_fast_path(p != end); p++) {
- ch = *p;
-
- /* Values below '0' become >= 208. */
- ch = ch - '0';
-
- if (ch > 9) {
- break;
- }
-
- if (nxt_slow_path((frac >= cutoff && (frac > cutoff || ch > cutlim))
- || power > cutoff))
- {
- return NULL;
- }
-
- frac = frac * 10 + ch;
- power *= 10;
- }
-
- if (nxt_slow_path(p == start)) {
- return NULL;
- }
-
- value->type = NXT_CONF_JSON_NUMBER;
- value->u.number = integer + (double) frac / power;
-
- value->u.number = copysign(value->u.number, sign);
-
- if (ch == 'e' || ch == 'E') {
- start = p + 1;
-
- ch = *start;
-
- if (ch == '-' || ch == '+') {
- start++;
- }
-
- negative = (ch == '-') ? 1 : 0;
- e = 0;
-
- for (p = start; nxt_fast_path(p != end); p++) {
- ch = *p;
-
- /* Values below '0' become >= 208. */
- ch = ch - '0';
-
- if (ch > 9) {
- break;
- }
-
- e = e * 10 + ch;
-
- if (nxt_slow_path(e > DBL_MAX_10_EXP)) {
- return NULL;
- }
- }
-
- if (nxt_slow_path(p == start)) {
- return NULL;
- }
-
- if (negative) {
- value->u.number /= exp10(e);
-
- } else {
- value->u.number *= exp10(e);
- }
- }
-
- if (nxt_fast_path(isfinite(value->u.number))) {
- return p;
- }
-#endif
-
- return NULL;
-}
-
-
-size_t
-nxt_conf_json_value_length(nxt_conf_json_value_t *value,
- nxt_conf_json_pretty_t *pretty)
-{
- switch (value->type) {
-
- case NXT_CONF_JSON_NULL:
- return sizeof("null") - 1;
-
- case NXT_CONF_JSON_BOOLEAN:
- return value->u.boolean ? sizeof("true") - 1 : sizeof("false") - 1;
-
- case NXT_CONF_JSON_INTEGER:
- return nxt_conf_json_integer_length(value);
-
- case NXT_CONF_JSON_NUMBER:
- /* TODO */
- return 0;
-
- case NXT_CONF_JSON_SHORT_STRING:
- case NXT_CONF_JSON_STRING:
- return nxt_conf_json_string_length(value);
-
- case NXT_CONF_JSON_ARRAY:
- return nxt_conf_json_array_length(value, pretty);
-
- case NXT_CONF_JSON_OBJECT:
- return nxt_conf_json_object_length(value, pretty);
- }
-
- nxt_unreachable();
-
- return 0;
-}
-
-
-u_char *
-nxt_conf_json_value_print(u_char *p, nxt_conf_json_value_t *value,
- nxt_conf_json_pretty_t *pretty)
-{
- switch (value->type) {
-
- case NXT_CONF_JSON_NULL:
- return nxt_cpymem(p, "null", 4);
-
- case NXT_CONF_JSON_BOOLEAN:
- return value->u.boolean ? nxt_cpymem(p, "true", 4)
- : nxt_cpymem(p, "false", 5);
-
- case NXT_CONF_JSON_INTEGER:
- return nxt_conf_json_integer_print(p, value);
-
- case NXT_CONF_JSON_NUMBER:
- /* TODO */
- return p;
-
- case NXT_CONF_JSON_SHORT_STRING:
- case NXT_CONF_JSON_STRING:
- return nxt_conf_json_string_print(p, value);
-
- case NXT_CONF_JSON_ARRAY:
- return nxt_conf_json_array_print(p, value, pretty);
-
- case NXT_CONF_JSON_OBJECT:
- return nxt_conf_json_object_print(p, value, pretty);
- }
-
- nxt_unreachable();
-
- return p;
-}
-
-
-static size_t
-nxt_conf_json_integer_length(nxt_conf_json_value_t *value)
-{
- int64_t num;
-
- num = llabs(value->u.integer);
-
- if (num <= 9999) {
- return sizeof("-9999") - 1;
- }
-
- if (num <= 99999999999) {
- return sizeof("-99999999999") - 1;
- }
-
- return NXT_INT64_T_LEN;
-}
-
-
-static u_char *
-nxt_conf_json_integer_print(u_char *p, nxt_conf_json_value_t *value)
-{
- return nxt_sprintf(p, p + NXT_INT64_T_LEN, "%L", value->u.integer);
-}
-
-
-static size_t
-nxt_conf_json_string_length(nxt_conf_json_value_t *value)
-{
- nxt_str_t str;
-
- nxt_conf_json_value_get_string(value, &str);
-
- return 2 + nxt_conf_json_escape_length(str.start, str.length);
-}
-
-
-static u_char *
-nxt_conf_json_string_print(u_char *p, nxt_conf_json_value_t *value)
-{
- nxt_str_t str;
-
- nxt_conf_json_value_get_string(value, &str);
-
- *p++ = '"';
-
- p = nxt_conf_json_escape(p, str.start, str.length);
-
- *p++ = '"';
-
- return p;
-}
-
-
-static size_t
-nxt_conf_json_array_length(nxt_conf_json_value_t *value,
- nxt_conf_json_pretty_t *pretty)
-{
- size_t len;
- nxt_uint_t n;
- nxt_conf_json_array_t *array;
-
- array = value->u.array;
-
- /* [] */
- len = 2;
-
- if (pretty != NULL) {
- pretty->level++;
- }
-
- value = array->elements;
-
- for (n = 0; n < array->count; n++) {
- len += nxt_conf_json_value_length(&value[n], pretty);
-
- if (pretty != NULL) {
- /* Indentation and new line. */
- len += pretty->level + 2;
- }
- }
-
- if (pretty != NULL) {
- pretty->level--;
-
- if (n != 0) {
- /* Indentation and new line. */
- len += pretty->level + 2;
- }
- }
-
- /* Reserve space for "n" commas. */
- return len + n;
-}
-
-
-static u_char *
-nxt_conf_json_array_print(u_char *p, nxt_conf_json_value_t *value,
- nxt_conf_json_pretty_t *pretty)
-{
- nxt_uint_t n;
- nxt_conf_json_array_t *array;
-
- array = value->u.array;
-
- *p++ = '[';
-
- if (array->count != 0) {
- value = array->elements;
-
- if (pretty != NULL) {
- p = nxt_conf_json_newline(p);
-
- pretty->level++;
- p = nxt_conf_json_indentation(p, pretty->level);
- }
-
- p = nxt_conf_json_value_print(p, &value[0], pretty);
-
- for (n = 1; n < array->count; n++) {
- *p++ = ',';
-
- if (pretty != NULL) {
- p = nxt_conf_json_newline(p);
- p = nxt_conf_json_indentation(p, pretty->level);
-
- pretty->more_space = 0;
- }
-
- p = nxt_conf_json_value_print(p, &value[n], pretty);
- }
-
- if (pretty != NULL) {
- p = nxt_conf_json_newline(p);
-
- pretty->level--;
- p = nxt_conf_json_indentation(p, pretty->level);
-
- pretty->more_space = 1;
- }
- }
-
- *p++ = ']';
-
- return p;
-}
-
-
-static size_t
-nxt_conf_json_object_length(nxt_conf_json_value_t *value,
- nxt_conf_json_pretty_t *pretty)
-{
- size_t len;
- nxt_uint_t n;
- nxt_conf_json_object_t *object;
- nxt_conf_json_obj_member_t *member;
-
- object = value->u.object;
-
- /* {} */
- len = 2;
-
- if (pretty != NULL) {
- pretty->level++;
- }
-
- member = object->members;
-
- for (n = 0; n < object->count; n++) {
- len += nxt_conf_json_string_length(&member[n].name) + 1
- + nxt_conf_json_value_length(&member[n].value, pretty) + 1;
-
- if (pretty != NULL) {
- /*
- * Indentation, space after ":", new line, and possible
- * additional empty line between non-empty objects.
- */
- len += pretty->level + 1 + 2 + 2;
- }
- }
-
- if (pretty != NULL) {
- pretty->level--;
-
- /* Indentation and new line. */
- len += pretty->level + 2;
- }
-
- return len;
-}
-
-
-static u_char *
-nxt_conf_json_object_print(u_char *p, nxt_conf_json_value_t *value,
- nxt_conf_json_pretty_t *pretty)
-{
- nxt_uint_t n;
- nxt_conf_json_object_t *object;
- nxt_conf_json_obj_member_t *member;
-
- object = value->u.object;
-
- *p++ = '{';
-
- if (object->count != 0) {
-
- if (pretty != NULL) {
- p = nxt_conf_json_newline(p);
- pretty->level++;
- }
-
- member = object->members;
-
- n = 0;
-
- for ( ;; ) {
- if (pretty != NULL) {
- p = nxt_conf_json_indentation(p, pretty->level);
- }
-
- p = nxt_conf_json_string_print(p, &member[n].name);
-
- *p++ = ':';
-
- if (pretty != NULL) {
- *p++ = ' ';
- }
-
- p = nxt_conf_json_value_print(p, &member[n].value, pretty);
-
- n++;
-
- if (n == object->count) {
- break;
- }
-
- *p++ = ',';
-
- if (pretty != NULL) {
- p = nxt_conf_json_newline(p);
-
- if (pretty->more_space) {
- pretty->more_space = 0;
- p = nxt_conf_json_newline(p);
- }
- }
- }
-
- if (pretty != NULL) {
- p = nxt_conf_json_newline(p);
-
- pretty->level--;
- p = nxt_conf_json_indentation(p, pretty->level);
-
- pretty->more_space = 1;
- }
- }
-
- *p++ = '}';
-
- return p;
-}
-
-
-static size_t
-nxt_conf_json_escape_length(u_char *p, size_t size)
-{
- u_char ch;
- size_t len;
-
- len = size;
-
- while (size) {
- ch = *p++;
-
- if (ch == '\\' || ch == '"') {
- len++;
-
- } else if (ch <= 0x1f) {
-
- switch (ch) {
- case '\n':
- case '\r':
- case '\t':
- case '\b':
- case '\f':
- len++;
- break;
-
- default:
- len += sizeof("\\u001F") - 2;
- }
- }
-
- size--;
- }
-
- return len;
-}
-
-
-static u_char *
-nxt_conf_json_escape(u_char *dst, u_char *src, size_t size)
-{
- u_char ch;
-
- while (size) {
- ch = *src++;
-
- if (ch > 0x1f) {
-
- if (ch == '\\' || ch == '"') {
- *dst++ = '\\';
- }
-
- *dst++ = ch;
-
- } else {
- *dst++ = '\\';
-
- switch (ch) {
- case '\n':
- *dst++ = 'n';
- break;
-
- case '\r':
- *dst++ = 'r';
- break;
-
- case '\t':
- *dst++ = 't';
- break;
-
- case '\b':
- *dst++ = 'b';
- break;
-
- case '\f':
- *dst++ = 'f';
- break;
-
- default:
- *dst++ = 'u'; *dst++ = '0'; *dst++ = '0';
- *dst++ = '0' + (ch >> 4);
-
- ch &= 0xf;
-
- *dst++ = (ch < 10) ? ('0' + ch) : ('A' + ch - 10);
- }
- }
-
- size--;
- }
-
- return dst;
-}