summaryrefslogtreecommitdiffhomepage
path: root/src/nxt_conf_json.c
diff options
context:
space:
mode:
Diffstat (limited to '')
-rw-r--r--src/nxt_conf_json.c772
1 files changed, 772 insertions, 0 deletions
diff --git a/src/nxt_conf_json.c b/src/nxt_conf_json.c
new file mode 100644
index 00000000..5f181611
--- /dev/null
+++ b/src/nxt_conf_json.c
@@ -0,0 +1,772 @@
+
+/*
+ * 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
+
+
+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;
+
+
+struct nxt_conf_json_value_s {
+ union {
+ uint32_t boolean; /* 1 bit. */
+ int64_t integer;
+ /* double number; */
+ u_char str[15];
+ nxt_str_t *string;
+ nxt_lvlhsh_t *object;
+ nxt_array_t *array;
+ } u;
+
+ nxt_conf_json_type_t type:8; /* 3 bits. */
+};
+
+
+typedef struct {
+ nxt_conf_json_value_t name;
+ nxt_conf_json_value_t value;
+} nxt_conf_json_obj_member_t;
+
+
+static nxt_int_t nxt_conf_json_object_hash_test(nxt_lvlhsh_query_t *lhq,
+ void *data);
+static nxt_int_t nxt_conf_json_object_member_add(nxt_lvlhsh_t *lvlhsh,
+ nxt_conf_json_obj_member_t *member, nxt_mem_pool_t *pool);
+#if 0
+static nxt_conf_json_value_t *nxt_conf_json_object_member_get(
+ nxt_lvlhsh_t *lvlhsh, u_char *name, size_t length);
+#endif
+
+
+static u_char *nxt_conf_json_skip_space(u_char *pos, u_char *end);
+static u_char *nxt_conf_json_parse_value(u_char *pos, u_char *end,
+ nxt_conf_json_value_t *value, nxt_mem_pool_t *pool);
+static u_char *nxt_conf_json_parse_object(u_char *pos, u_char *end,
+ nxt_conf_json_value_t *value, nxt_mem_pool_t *pool);
+static u_char *nxt_conf_json_parse_array(u_char *pos, u_char *end,
+ nxt_conf_json_value_t *value, nxt_mem_pool_t *pool);
+static u_char *nxt_conf_json_parse_string(u_char *pos, u_char *end,
+ nxt_conf_json_value_t *value, nxt_mem_pool_t *pool);
+static u_char *nxt_conf_json_parse_number(u_char *pos, u_char *end,
+ nxt_conf_json_value_t *value, nxt_mem_pool_t *pool);
+
+
+static const nxt_lvlhsh_proto_t nxt_conf_json_object_hash_proto
+ nxt_aligned(64) =
+{
+ NXT_LVLHSH_DEFAULT,
+ 0,
+ nxt_conf_json_object_hash_test,
+ nxt_lvlhsh_pool_alloc,
+ nxt_lvlhsh_pool_free,
+};
+
+
+static nxt_int_t
+nxt_conf_json_object_hash_test(nxt_lvlhsh_query_t *lhq, void *data)
+{
+ nxt_conf_json_value_t *name;
+
+ name = &((nxt_conf_json_obj_member_t *) data)->name;
+
+ if (name->type == NXT_CONF_JSON_SHORT_STRING) {
+
+ if (nxt_str_eq(&lhq->key, &name->u.str[1], name->u.str[0])) {
+ return NXT_OK;
+ }
+
+ } else {
+
+ if (nxt_strstr_eq(&lhq->key, name->u.string)) {
+ return NXT_OK;
+ }
+ }
+
+ return NXT_DECLINED;
+}
+
+
+static nxt_int_t
+nxt_conf_json_object_member_add(nxt_lvlhsh_t *lvlhsh,
+ nxt_conf_json_obj_member_t *member, nxt_mem_pool_t *pool)
+{
+ nxt_lvlhsh_query_t lhq;
+ nxt_conf_json_value_t *name;
+
+ name = &member->name;
+
+ if (name->type == NXT_CONF_JSON_SHORT_STRING) {
+ lhq.key.length = name->u.str[0];
+ lhq.key.start = &name->u.str[1];
+
+ } else {
+ lhq.key = *name->u.string;
+ }
+
+ 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 = pool;
+
+ return nxt_lvlhsh_insert(lvlhsh, &lhq);
+}
+
+
+#if 0
+static nxt_conf_json_value_t *
+nxt_conf_json_object_member_get(nxt_lvlhsh_t *lvlhsh, u_char *name,
+ size_t length)
+{
+ nxt_lvlhsh_query_t lhq;
+ nxt_conf_json_obj_member_t *member;
+
+ lhq.key_hash = nxt_djb_hash(name, length);
+ lhq.key.length = length;
+ lhq.key.start = name;
+ lhq.proto = &nxt_conf_json_object_hash_proto;
+
+ if (nxt_fast_path(nxt_lvlhsh_find(lvlhsh, &lhq) == NXT_OK)) {
+ member = lhq.value;
+ return &member->value;
+ }
+
+ return NULL;
+}
+#endif
+
+
+nxt_conf_json_value_t *
+nxt_conf_json_parse(nxt_buf_mem_t *b, nxt_mem_pool_t *pool)
+{
+ u_char *pos, *end;
+ nxt_conf_json_value_t *value;
+
+ value = nxt_mem_alloc(pool, sizeof(nxt_conf_json_value_t));
+ if (nxt_slow_path(value == NULL)) {
+ return NULL;
+ }
+
+ pos = b->pos;
+ end = b->free;
+
+ pos = nxt_conf_json_skip_space(pos, end);
+
+ if (nxt_slow_path(pos == end)) {
+ return NULL;
+ }
+
+ pos = nxt_conf_json_parse_value(pos, end, value, pool);
+
+ if (nxt_slow_path(pos == NULL)) {
+ return NULL;
+ }
+
+ pos = nxt_conf_json_skip_space(pos, end);
+
+ if (nxt_slow_path(pos != end)) {
+ return NULL;
+ }
+
+ return value;
+}
+
+
+static u_char *
+nxt_conf_json_skip_space(u_char *pos, u_char *end)
+{
+ for ( /* void */ ; nxt_fast_path(pos != end); pos++) {
+
+ switch (*pos) {
+ case ' ':
+ case '\t':
+ case '\r':
+ case '\n':
+ continue;
+ }
+
+ break;
+ }
+
+ return pos;
+}
+
+
+static u_char *
+nxt_conf_json_parse_value(u_char *pos, u_char *end,
+ nxt_conf_json_value_t *value, nxt_mem_pool_t *pool)
+{
+ u_char ch;
+
+ ch = *pos;
+
+ switch (ch) {
+ case '{':
+ return nxt_conf_json_parse_object(pos, end, value, pool);
+
+ case '[':
+ return nxt_conf_json_parse_array(pos, end, value, pool);
+
+ case '"':
+ return nxt_conf_json_parse_string(pos, end, value, pool);
+
+ case 't':
+ if (nxt_fast_path(end - pos >= 4
+ || nxt_memcmp(pos, (u_char *) "true", 4) == 0))
+ {
+ value->u.boolean = 1;
+ value->type = NXT_CONF_JSON_BOOLEAN;
+
+ return pos + 4;
+ }
+
+ return NULL;
+
+ case 'f':
+ if (nxt_fast_path(end - pos >= 5
+ || nxt_memcmp(pos, (u_char *) "false", 5) == 0))
+ {
+ value->u.boolean = 0;
+ value->type = NXT_CONF_JSON_BOOLEAN;
+
+ return pos + 5;
+ }
+
+ return NULL;
+
+ case 'n':
+ if (nxt_fast_path(end - pos >= 4
+ || nxt_memcmp(pos, (u_char *) "null", 4) == 0))
+ {
+ value->type = NXT_CONF_JSON_NULL;
+ return pos + 4;
+ }
+
+ return NULL;
+ }
+
+ if (nxt_fast_path(ch == '-' || (ch - '0') <= 9)) {
+ return nxt_conf_json_parse_number(pos, end, value, pool);
+ }
+
+ return NULL;
+}
+
+
+static u_char *
+nxt_conf_json_parse_object(u_char *pos, u_char *end,
+ nxt_conf_json_value_t *value, nxt_mem_pool_t *pool)
+{
+ nxt_int_t rc;
+ nxt_lvlhsh_t *object;
+ nxt_conf_json_obj_member_t *member;
+
+ object = nxt_mem_zalloc(pool, sizeof(nxt_lvlhsh_t));
+ if (nxt_slow_path(object == NULL)) {
+ return NULL;
+ }
+
+ value->type = NXT_CONF_JSON_OBJECT;
+ value->u.object = object;
+
+ pos = nxt_conf_json_skip_space(pos + 1, end);
+
+ if (nxt_slow_path(pos == end)) {
+ return NULL;
+ }
+
+ if (*pos != '}') {
+
+ for ( ;; ) {
+ if (*pos != '"') {
+ return NULL;
+ }
+
+ member = nxt_mem_alloc(pool, sizeof(nxt_conf_json_obj_member_t));
+ if (nxt_slow_path(member == NULL)) {
+ return NULL;
+ }
+
+ pos = nxt_conf_json_parse_string(pos, end, &member->name, pool);
+
+ if (nxt_slow_path(pos == NULL)) {
+ return NULL;
+ }
+
+ pos = nxt_conf_json_skip_space(pos, end);
+
+ if (nxt_slow_path(pos == end || *pos != ':')) {
+ return NULL;
+ }
+
+ pos = nxt_conf_json_skip_space(pos + 1, end);
+
+ if (nxt_slow_path(pos == end)) {
+ return NULL;
+ }
+
+ pos = nxt_conf_json_parse_value(pos, end, &member->value, pool);
+
+ if (nxt_slow_path(pos == NULL)) {
+ return NULL;
+ }
+
+ rc = nxt_conf_json_object_member_add(object, member, pool);
+
+ if (nxt_slow_path(rc != NXT_OK)) {
+ return NULL;
+ }
+
+ pos = nxt_conf_json_skip_space(pos, end);
+
+ if (nxt_slow_path(pos == end)) {
+ return NULL;
+ }
+
+ if (*pos == '}') {
+ break;
+ }
+
+ if (nxt_slow_path(*pos != ',')) {
+ return NULL;
+ }
+
+ pos = nxt_conf_json_skip_space(pos + 1, end);
+
+ if (nxt_slow_path(pos == end)) {
+ return NULL;
+ }
+ }
+ }
+
+ return pos + 1;
+}
+
+
+static u_char *
+nxt_conf_json_parse_array(u_char *pos, u_char *end,
+ nxt_conf_json_value_t *value, nxt_mem_pool_t *pool)
+{
+ nxt_array_t *array;
+
+ array = nxt_array_create(pool, 8, sizeof(nxt_conf_json_value_t));
+ if (nxt_slow_path(array == NULL)) {
+ return NULL;
+ }
+
+ value->type = NXT_CONF_JSON_ARRAY;
+ value->u.array = array;
+
+ pos = nxt_conf_json_skip_space(pos + 1, end);
+
+ if (nxt_slow_path(pos == end)) {
+ return NULL;
+ }
+
+ if (*pos != ']') {
+
+ for ( ;; ) {
+ value = nxt_array_add(array);
+ if (nxt_slow_path(value == NULL)) {
+ return NULL;
+ }
+
+ pos = nxt_conf_json_parse_value(pos, end, value, pool);
+
+ if (nxt_slow_path(pos == NULL)) {
+ return NULL;
+ }
+
+ pos = nxt_conf_json_skip_space(pos, end);
+
+ if (nxt_slow_path(pos == end)) {
+ return NULL;
+ }
+
+ if (*pos == ']') {
+ break;
+ }
+
+ if (nxt_slow_path(*pos != ',')) {
+ return NULL;
+ }
+
+ pos = nxt_conf_json_skip_space(pos + 1, end);
+
+ if (nxt_slow_path(pos == end)) {
+ return NULL;
+ }
+ }
+ }
+
+ return pos + 1;
+}
+
+
+static u_char *
+nxt_conf_json_parse_string(u_char *pos, u_char *end,
+ nxt_conf_json_value_t *value, nxt_mem_pool_t *pool)
+{
+ u_char 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;
+
+ pos++;
+
+ state = 0;
+ surplus = 0;
+
+ for (last = pos; last != end; last++) {
+ ch = *last;
+
+ 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 'b':
+ case 'f':
+ case 'n':
+ case 'r':
+ case 't':
+ 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(last == end)) {
+ return NULL;
+ }
+
+ size = last - pos - surplus;
+
+ if (size > 14) {
+ value->type = NXT_CONF_JSON_STRING;
+ value->u.string = nxt_str_alloc(pool, size);
+
+ if (nxt_slow_path(value->u.string == NULL)) {
+ return NULL;
+ }
+
+ s = value->u.string->start;
+
+ } else {
+ value->type = NXT_CONF_JSON_SHORT_STRING;
+ s = &value->u.str[1];
+ }
+
+ if (surplus == 0) {
+ nxt_memcpy(s, pos, size);
+ return last + 1;
+ }
+
+ state = 0;
+
+ do {
+ ch = *pos++;
+
+ if (ch != '\\') {
+ *s++ = ch;
+ continue;
+ }
+
+ ch = *pos++;
+
+ switch (ch) {
+ case '"':
+ case '\\':
+ case '/':
+ *s++ = ch;
+ continue;
+
+ case 'b':
+ *s++ = '\b';
+ continue;
+
+ case 'f':
+ *s++ = '\f';
+ continue;
+
+ case 'n':
+ *s++ = '\n';
+ continue;
+
+ case 'r':
+ *s++ = '\r';
+ continue;
+
+ case 't':
+ *s++ = '\t';
+ continue;
+ }
+
+ utf = 0;
+ utf_high = 0;
+
+ for ( ;; ) {
+ for (i = 0; i < 4; i++) {
+ utf = (utf << 4) + (pos[i] - (pos[i] >= 'A' ? 'A' : '0'));
+ }
+
+ pos += 4;
+
+ if (utf < 0xd800 || utf > 0xdbff || utf_high) {
+ break;
+ }
+
+ utf_high = utf;
+ utf = 0;
+
+ if (pos[0] != '\\' || pos[1] != 'u') {
+ break;
+ }
+
+ pos += 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 (pos != last);
+
+ if (size > 14) {
+ value->u.string->length = s - value->u.string->start;
+
+ } else {
+ value->u.str[0] = s - &value->u.str[1];
+ }
+
+ return pos + 1;
+}
+
+
+static u_char *
+nxt_conf_json_parse_number(u_char *pos, u_char *end,
+ nxt_conf_json_value_t *value, nxt_mem_pool_t *pool)
+{
+ u_char ch, *p;
+ 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 = *pos;
+
+ if (ch == '-') {
+ sign = -1;
+ pos++;
+
+ } else {
+ sign = 1;
+ }
+
+ integer = 0;
+
+ for (p = pos; 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 == pos || (p > pos + 1 && *pos == '0'))) {
+ return NULL;
+ }
+
+ if (ch != '.') {
+ value->type = NXT_CONF_JSON_INTEGER;
+ value->u.integer = sign * integer;
+ return p;
+ }
+
+#if 0
+ pos = p + 1;
+
+ frac = 0;
+ power = 1;
+
+ for (p = pos; 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 == pos)) {
+ 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') {
+ pos = p + 1;
+
+ ch = *pos;
+
+ if (ch == '-' || ch == '+') {
+ pos++;
+ }
+
+ negative = (ch == '-') ? 1 : 0;
+ e = 0;
+
+ for (p = pos; 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 == pos)) {
+ 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;
+}