summaryrefslogtreecommitdiff
path: root/src/json.c
diff options
context:
space:
mode:
Diffstat (limited to 'src/json.c')
-rw-r--r--src/json.c509
1 files changed, 509 insertions, 0 deletions
diff --git a/src/json.c b/src/json.c
new file mode 100644
index 000000000..7b67994dd
--- /dev/null
+++ b/src/json.c
@@ -0,0 +1,509 @@
+/* vi:set ts=8 sts=4 sw=4:
+ *
+ * VIM - Vi IMproved by Bram Moolenaar
+ *
+ * Do ":help uganda" in Vim to read copying and usage conditions.
+ * Do ":help credits" in Vim to see a list of people who contributed.
+ * See README.txt for an overview of the Vim source code.
+ */
+
+/*
+ * json.c: Encoding and decoding JSON.
+ *
+ * Follows this standard: http://www.ietf.org/rfc/rfc4627.txt
+ */
+
+#include "vim.h"
+
+#if defined(FEAT_EVAL) || defined(PROTO)
+static void json_decode_item(js_read_T *reader, typval_T *res);
+
+/*
+ * Encode "val" into a JSON format string.
+ */
+ char_u *
+json_encode(typval_T *val)
+{
+ garray_T ga;
+
+ /* Store bytes in the growarray. */
+ ga_init2(&ga, 1, 4000);
+ json_encode_item(&ga, val, get_copyID());
+ return ga.ga_data;
+}
+
+ static void
+write_string(garray_T *gap, char_u *str)
+{
+ char_u *res = str;
+ char_u numbuf[NUMBUFLEN];
+
+ if (res == NULL)
+ ga_concat(gap, (char_u *)"null");
+ else
+ {
+ ga_append(gap, '"');
+ while (*res != NUL)
+ {
+ int c = PTR2CHAR(res);
+
+ switch (c)
+ {
+ case 0x08:
+ ga_append(gap, '\\'); ga_append(gap, 'b'); break;
+ case 0x09:
+ ga_append(gap, '\\'); ga_append(gap, 't'); break;
+ case 0x0a:
+ ga_append(gap, '\\'); ga_append(gap, 'n'); break;
+ case 0x0c:
+ ga_append(gap, '\\'); ga_append(gap, 'f'); break;
+ case 0x0d:
+ ga_append(gap, '\\'); ga_append(gap, 'r'); break;
+ case 0x22: /* " */
+ case 0x5c: /* \ */
+ ga_append(gap, '\\');
+ ga_append(gap, c);
+ break;
+ default:
+ if (c >= 0x20)
+ {
+ numbuf[mb_char2bytes(c, numbuf)] = NUL;
+ ga_concat(gap, numbuf);
+ }
+ else
+ {
+ vim_snprintf((char *)numbuf, NUMBUFLEN,
+ "\\u%04lx", (long)c);
+ ga_concat(gap, numbuf);
+ }
+ }
+ mb_cptr_adv(res);
+ }
+ ga_append(gap, '"');
+ }
+}
+
+ void
+json_encode_item(garray_T *gap, typval_T *val, int copyID)
+{
+ char_u numbuf[NUMBUFLEN];
+ char_u *res;
+ list_T *l;
+ dict_T *d;
+
+ switch (val->v_type)
+ {
+ case VAR_SPECIAL:
+ switch(val->vval.v_number)
+ {
+ case VVAL_FALSE: ga_concat(gap, (char_u *)"false"); break;
+ case VVAL_TRUE: ga_concat(gap, (char_u *)"true"); break;
+ case VVAL_NONE: break;
+ case VVAL_NULL: ga_concat(gap, (char_u *)"null"); break;
+ }
+ break;
+
+ case VAR_NUMBER:
+ vim_snprintf((char *)numbuf, NUMBUFLEN, "%ld",
+ (long)val->vval.v_number);
+ ga_concat(gap, numbuf);
+ break;
+
+ case VAR_STRING:
+ res = val->vval.v_string;
+ write_string(gap, res);
+ break;
+
+ case VAR_FUNC:
+ /* no JSON equivalent, skip */
+ break;
+
+ case VAR_LIST:
+ l = val->vval.v_list;
+ if (l == NULL)
+ ga_concat(gap, (char_u *)"null");
+ else
+ {
+ if (l->lv_copyID == copyID)
+ ga_concat(gap, (char_u *)"[]");
+ else
+ {
+ listitem_T *li;
+
+ l->lv_copyID = copyID;
+ ga_append(gap, '[');
+ for (li = l->lv_first; li != NULL && !got_int; )
+ {
+ json_encode_item(gap, &li->li_tv, copyID);
+ li = li->li_next;
+ if (li != NULL)
+ ga_append(gap, ',');
+ }
+ ga_append(gap, ']');
+ }
+ }
+ break;
+
+ case VAR_DICT:
+ d = val->vval.v_dict;
+ if (d == NULL)
+ ga_concat(gap, (char_u *)"null");
+ else
+ {
+ if (d->dv_copyID == copyID)
+ ga_concat(gap, (char_u *)"{}");
+ else
+ {
+ int first = TRUE;
+ int todo = (int)d->dv_hashtab.ht_used;
+ hashitem_T *hi;
+
+ d->dv_copyID = copyID;
+ ga_append(gap, '{');
+
+ for (hi = d->dv_hashtab.ht_array; todo > 0 && !got_int;
+ ++hi)
+ if (!HASHITEM_EMPTY(hi))
+ {
+ --todo;
+ if (first)
+ first = FALSE;
+ else
+ ga_append(gap, ',');
+ write_string(gap, hi->hi_key);
+ ga_append(gap, ':');
+ json_encode_item(gap, &dict_lookup(hi)->di_tv,
+ copyID);
+ }
+ ga_append(gap, '}');
+ }
+ }
+ break;
+
+#ifdef FEAT_FLOAT
+ case VAR_FLOAT:
+ vim_snprintf((char *)numbuf, NUMBUFLEN, "%g", val->vval.v_float);
+ ga_concat(gap, numbuf);
+ break;
+#endif
+ default: EMSG2(_(e_intern2), "json_encode_item()"); break;
+ }
+}
+
+/*
+ * Skip white space in "reader".
+ */
+ static void
+json_skip_white(js_read_T *reader)
+{
+ int c;
+
+ while ((c = reader->js_buf[reader->js_used]) == ' '
+ || c == TAB || c == NL || c == CAR)
+ ++reader->js_used;
+}
+
+/*
+ * Make sure there are at least enough characters buffered to read a number.
+ */
+ static void
+json_fill_buffer(js_read_T *reader UNUSED)
+{
+ /* TODO */
+}
+
+ static void
+json_decode_array(js_read_T *reader, typval_T *res)
+{
+ char_u *p;
+ typval_T item;
+ listitem_T *li;
+
+ if (rettv_list_alloc(res) == FAIL)
+ goto fail;
+ ++reader->js_used; /* consume the '[' */
+
+ while (TRUE)
+ {
+ json_skip_white(reader);
+ p = reader->js_buf + reader->js_used;
+ if (*p == NUL)
+ goto fail;
+ if (*p == ']')
+ {
+ ++reader->js_used; /* consume the ']' */
+ return;
+ }
+
+ if (!reader->js_eof && (int)(reader->js_end - p) < NUMBUFLEN)
+ json_fill_buffer(reader);
+
+ json_decode_item(reader, &item);
+ li = listitem_alloc();
+ if (li == NULL)
+ return;
+ li->li_tv = item;
+ list_append(res->vval.v_list, li);
+
+ json_skip_white(reader);
+ p = reader->js_buf + reader->js_used;
+ if (*p == ',')
+ ++reader->js_used;
+ else if (*p != ']')
+ goto fail;
+ }
+fail:
+ res->v_type = VAR_SPECIAL;
+ res->vval.v_number = VVAL_NONE;
+}
+
+ static void
+json_decode_object(js_read_T *reader, typval_T *res)
+{
+ char_u *p;
+ typval_T tvkey;
+ typval_T item;
+ dictitem_T *di;
+ char_u buf[NUMBUFLEN];
+ char_u *key;
+
+ if (rettv_dict_alloc(res) == FAIL)
+ goto fail;
+ ++reader->js_used; /* consume the '{' */
+
+ while (TRUE)
+ {
+ json_skip_white(reader);
+ p = reader->js_buf + reader->js_used;
+ if (*p == NUL)
+ goto fail;
+ if (*p == '}')
+ {
+ ++reader->js_used; /* consume the '}' */
+ return;
+ }
+
+ if (!reader->js_eof && (int)(reader->js_end - p) < NUMBUFLEN)
+ json_fill_buffer(reader);
+ json_decode_item(reader, &tvkey);
+ key = get_tv_string_buf_chk(&tvkey, buf);
+ if (key == NULL || *key == NUL)
+ {
+ /* "key" is NULL when get_tv_string_buf_chk() gave an errmsg */
+ if (key != NULL)
+ EMSG(_(e_emptykey));
+ clear_tv(&tvkey);
+ goto fail;
+ }
+
+ json_skip_white(reader);
+ p = reader->js_buf + reader->js_used;
+ if (*p != ':')
+ {
+ clear_tv(&tvkey);
+ goto fail;
+ }
+ ++reader->js_used;
+ json_skip_white(reader);
+
+ if (!reader->js_eof && (int)(reader->js_end - p) < NUMBUFLEN)
+ json_fill_buffer(reader);
+ json_decode_item(reader, &item);
+
+ di = dictitem_alloc(key);
+ clear_tv(&tvkey);
+ if (di == NULL)
+ {
+ clear_tv(&item);
+ goto fail;
+ }
+ di->di_tv = item;
+ dict_add(res->vval.v_dict, di);
+
+ json_skip_white(reader);
+ p = reader->js_buf + reader->js_used;
+ if (*p == ',')
+ ++reader->js_used;
+ else if (*p != '}')
+ goto fail;
+ }
+fail:
+ res->v_type = VAR_SPECIAL;
+ res->vval.v_number = VVAL_NONE;
+}
+
+ static void
+json_decode_string(js_read_T *reader, typval_T *res)
+{
+ garray_T ga;
+ int len;
+ char_u *p = reader->js_buf + reader->js_used + 1;
+ int c;
+ long nr;
+ char_u buf[NUMBUFLEN];
+
+ ga_init2(&ga, 1, 200);
+
+ /* TODO: fill buffer when needed. */
+ while (*p != NUL && *p != '"')
+ {
+ if (*p == '\\')
+ {
+ c = -1;
+ switch (p[1])
+ {
+ case 'b': c = BS; break;
+ case 't': c = TAB; break;
+ case 'n': c = NL; break;
+ case 'f': c = FF; break;
+ case 'r': c = CAR; break;
+ case 'u':
+ vim_str2nr(p + 2, NULL, &len,
+ STR2NR_HEX + STR2NR_FORCE, &nr, NULL, 4);
+ p += len + 2;
+#ifdef FEAT_MBYTE
+ buf[(*mb_char2bytes)((int)nr, buf)] = NUL;
+ ga_concat(&ga, buf);
+#else
+ ga_append(&ga, nr);
+#endif
+ break;
+ default: c = p[1]; break;
+ }
+ if (c > 0)
+ {
+ p += 2;
+ ga_append(&ga, c);
+ }
+ }
+ else
+ {
+ len = MB_PTR2LEN(p);
+ if (ga_grow(&ga, len) == OK)
+ {
+ mch_memmove((char *)ga.ga_data + ga.ga_len, p, (size_t)len);
+ ga.ga_len += len;
+ }
+ p += len;
+ }
+ if (!reader->js_eof && (int)(reader->js_end - p) < NUMBUFLEN)
+ {
+ reader->js_used = (int)(p - reader->js_buf);
+ json_fill_buffer(reader);
+ p = reader->js_buf + reader->js_used;
+ }
+ }
+ reader->js_used = (int)(p - reader->js_buf);
+ if (*p == '"')
+ {
+ ++reader->js_used;
+ res->v_type = VAR_STRING;
+ res->vval.v_string = vim_strsave(ga.ga_data);
+ }
+ else
+ {
+ res->v_type = VAR_SPECIAL;
+ res->vval.v_number = VVAL_NONE;
+ }
+ ga_clear(&ga);
+}
+
+/*
+ * Decode one item and put it in "result".
+ * Must already have skipped white space.
+ */
+ static void
+json_decode_item(js_read_T *reader, typval_T *res)
+{
+ char_u *p = reader->js_buf + reader->js_used;
+
+ switch (*p)
+ {
+ case '[': /* array */
+ json_decode_array(reader, res);
+ return;
+
+ case '{': /* object */
+ json_decode_object(reader, res);
+ return;
+
+ case '"': /* string */
+ json_decode_string(reader, res);
+ return;
+
+ case ',': /* comma: empty item */
+ case NUL: /* empty */
+ res->v_type = VAR_SPECIAL;
+ res->vval.v_number = VVAL_NONE;
+ return;
+
+ default:
+ if (VIM_ISDIGIT(*p) || *p == '-')
+ {
+ int len;
+ char_u *sp = p;
+#ifdef FEAT_FLOAT
+ if (*sp == '-')
+ ++sp;
+ sp = skipdigits(sp);
+ if (*sp == '.' || *sp == 'e' || *sp == 'E')
+ {
+ res->v_type = VAR_FLOAT;
+ len = string2float(p, &res->vval.v_float);
+ }
+ else
+#endif
+ {
+ long nr;
+
+ res->v_type = VAR_NUMBER;
+ vim_str2nr(reader->js_buf + reader->js_used,
+ NULL, &len, 0, /* what */
+ &nr, NULL, 0);
+ res->vval.v_number = nr;
+ }
+ reader->js_used += len;
+ return;
+ }
+ if (STRNICMP((char *)p, "false", 5) == 0)
+ {
+ reader->js_used += 5;
+ res->v_type = VAR_SPECIAL;
+ res->vval.v_number = VVAL_FALSE;
+ return;
+ }
+ if (STRNICMP((char *)p, "true", 4) == 0)
+ {
+ reader->js_used += 4;
+ res->v_type = VAR_SPECIAL;
+ res->vval.v_number = VVAL_TRUE;
+ return;
+ }
+ if (STRNICMP((char *)p, "null", 4) == 0)
+ {
+ reader->js_used += 4;
+ res->v_type = VAR_SPECIAL;
+ res->vval.v_number = VVAL_NULL;
+ return;
+ }
+ break;
+ }
+
+ EMSG(_(e_invarg));
+ res->v_type = VAR_SPECIAL;
+ res->vval.v_number = VVAL_NONE;
+}
+
+/*
+ * Decode the JSON from "reader" and store the result in "res".
+ */
+ void
+json_decode(js_read_T *reader, typval_T *res)
+{
+ json_skip_white(reader);
+ json_decode_item(reader, res);
+ json_skip_white(reader);
+ if (reader->js_buf[reader->js_used] != NUL)
+ EMSG(_(e_invarg));
+}
+#endif