2012-08-27 10:11:55 +01:00
|
|
|
#ifndef JV_H
|
|
|
|
#define JV_H
|
|
|
|
|
2012-08-28 18:09:43 +01:00
|
|
|
#include <stdint.h>
|
|
|
|
#include <assert.h>
|
2012-09-01 19:16:43 +01:00
|
|
|
#include <stddef.h>
|
2012-08-27 10:11:55 +01:00
|
|
|
|
2012-08-28 18:09:43 +01:00
|
|
|
|
|
|
|
|
|
|
|
typedef enum {
|
2012-09-03 16:16:14 +01:00
|
|
|
JV_KIND_INVALID,
|
2012-08-28 18:09:43 +01:00
|
|
|
JV_KIND_NULL,
|
|
|
|
JV_KIND_FALSE,
|
|
|
|
JV_KIND_TRUE,
|
|
|
|
JV_KIND_NUMBER,
|
|
|
|
JV_KIND_STRING,
|
2012-09-10 15:04:19 +01:00
|
|
|
JV_KIND_ARRAY,
|
|
|
|
JV_KIND_OBJECT
|
2012-08-28 18:09:43 +01:00
|
|
|
} jv_kind;
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
size_t count;
|
|
|
|
} jv_refcnt;
|
|
|
|
|
|
|
|
typedef struct{
|
|
|
|
jv_refcnt* ptr;
|
|
|
|
int i[2];
|
|
|
|
} jv_complex;
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
jv_kind kind;
|
|
|
|
union {
|
|
|
|
double number;
|
|
|
|
jv_complex complex;
|
|
|
|
} val;
|
|
|
|
} jv;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* All jv_* functions consume (decref) input and produce (incref) output
|
|
|
|
* Except jv_copy
|
|
|
|
*/
|
|
|
|
|
|
|
|
jv_kind jv_get_kind(jv);
|
2012-09-10 16:07:03 +01:00
|
|
|
const char* jv_kind_name(jv_kind);
|
2012-09-03 16:16:14 +01:00
|
|
|
static int jv_is_valid(jv x) { return jv_get_kind(x) != JV_KIND_INVALID; }
|
2012-08-28 18:09:43 +01:00
|
|
|
|
|
|
|
jv jv_copy(jv);
|
|
|
|
void jv_free(jv);
|
|
|
|
|
|
|
|
int jv_equal(jv, jv);
|
|
|
|
|
2012-09-03 16:16:14 +01:00
|
|
|
jv jv_invalid();
|
2012-09-10 15:04:19 +01:00
|
|
|
jv jv_invalid_with_msg(jv);
|
|
|
|
jv jv_invalid_get_msg(jv);
|
|
|
|
|
2012-08-28 18:09:43 +01:00
|
|
|
jv jv_null();
|
|
|
|
jv jv_true();
|
|
|
|
jv jv_false();
|
2012-09-09 18:09:05 +01:00
|
|
|
jv jv_bool(int);
|
2012-08-28 18:09:43 +01:00
|
|
|
|
|
|
|
jv jv_number(double);
|
|
|
|
double jv_number_value(jv);
|
|
|
|
|
|
|
|
jv jv_array();
|
|
|
|
jv jv_array_sized(int);
|
|
|
|
int jv_array_length(jv);
|
|
|
|
jv jv_array_get(jv, int);
|
|
|
|
jv jv_array_set(jv, int, jv);
|
|
|
|
jv jv_array_append(jv, jv);
|
|
|
|
jv jv_array_concat(jv, jv);
|
|
|
|
jv jv_array_slice(jv, int, int);
|
|
|
|
|
|
|
|
|
2012-09-02 16:31:59 +01:00
|
|
|
jv jv_string(const char*);
|
|
|
|
jv jv_string_sized(const char*, int);
|
2012-08-28 18:09:43 +01:00
|
|
|
int jv_string_length(jv);
|
|
|
|
uint32_t jv_string_hash(jv);
|
|
|
|
const char* jv_string_value(jv);
|
2012-09-10 16:16:39 +01:00
|
|
|
jv jv_string_concat(jv, jv);
|
2012-09-10 16:01:56 +01:00
|
|
|
jv jv_string_fmt(const char*, ...);
|
2012-08-28 18:09:43 +01:00
|
|
|
|
|
|
|
jv jv_object();
|
|
|
|
jv jv_object_get(jv object, jv key);
|
|
|
|
jv jv_object_set(jv object, jv key, jv value);
|
|
|
|
jv jv_object_delete(jv object, jv key);
|
2012-09-02 00:24:23 +01:00
|
|
|
int jv_object_length(jv object);
|
2012-09-09 19:17:07 +01:00
|
|
|
jv jv_object_merge(jv, jv);
|
2012-08-28 18:09:43 +01:00
|
|
|
|
|
|
|
int jv_object_iter(jv);
|
|
|
|
int jv_object_iter_next(jv, int);
|
|
|
|
int jv_object_iter_valid(jv, int);
|
|
|
|
jv jv_object_iter_key(jv, int);
|
|
|
|
jv jv_object_iter_value(jv, int);
|
2012-09-09 19:17:07 +01:00
|
|
|
#define jv_object_foreach(i,t) \
|
|
|
|
for (int i = jv_object_iter(t); \
|
|
|
|
jv_object_iter_valid(t, i); \
|
|
|
|
i = jv_object_iter_next(t, i)) \
|
|
|
|
|
2012-08-28 18:09:43 +01:00
|
|
|
|
|
|
|
|
2012-09-02 21:45:27 +01:00
|
|
|
int jv_get_refcnt(jv);
|
|
|
|
|
2012-09-01 19:16:43 +01:00
|
|
|
void jv_dump(jv);
|
2012-09-02 16:31:59 +01:00
|
|
|
jv jv_parse(const char* string);
|
2012-09-03 13:36:12 +01:00
|
|
|
jv jv_parse_sized(const char* string, int length);
|
2012-09-02 16:31:59 +01:00
|
|
|
|
|
|
|
|
|
|
|
static jv jv_lookup(jv t, jv k) {
|
|
|
|
jv v;
|
|
|
|
if (jv_get_kind(t) == JV_KIND_OBJECT && jv_get_kind(k) == JV_KIND_STRING) {
|
|
|
|
v = jv_object_get(t, k);
|
2012-09-03 16:16:14 +01:00
|
|
|
if (!jv_is_valid(v)) {
|
2012-09-10 15:04:19 +01:00
|
|
|
jv_free(v);
|
2012-09-03 16:16:14 +01:00
|
|
|
v = jv_null();
|
|
|
|
}
|
2012-09-02 16:31:59 +01:00
|
|
|
} else if (jv_get_kind(t) == JV_KIND_ARRAY && jv_get_kind(k) == JV_KIND_NUMBER) {
|
|
|
|
// FIXME: don't do lookup for noninteger index
|
|
|
|
v = jv_array_get(t, (int)jv_number_value(k));
|
2012-09-03 16:50:38 +01:00
|
|
|
if (!jv_is_valid(v)) {
|
2012-09-10 15:04:19 +01:00
|
|
|
jv_free(v);
|
2012-09-03 16:50:38 +01:00
|
|
|
v = jv_null();
|
|
|
|
}
|
|
|
|
} else if (jv_get_kind(t) == JV_KIND_NULL &&
|
|
|
|
(jv_get_kind(k) == JV_KIND_STRING || jv_get_kind(k) == JV_KIND_NUMBER)) {
|
|
|
|
jv_free(t);
|
|
|
|
jv_free(k);
|
|
|
|
v = jv_null();
|
2012-09-02 16:31:59 +01:00
|
|
|
} else {
|
2012-09-11 10:37:44 +01:00
|
|
|
v = jv_invalid_with_msg(jv_string_fmt("Cannot index %s with %s",
|
|
|
|
jv_kind_name(jv_get_kind(t)),
|
|
|
|
jv_kind_name(jv_get_kind(k))));
|
|
|
|
jv_free(t);
|
|
|
|
jv_free(k);
|
2012-09-02 16:31:59 +01:00
|
|
|
}
|
|
|
|
return v;
|
|
|
|
}
|
|
|
|
|
|
|
|
static jv jv_modify(jv t, jv k, jv v) {
|
2012-09-11 10:37:44 +01:00
|
|
|
int isnull = jv_get_kind(t) == JV_KIND_NULL;
|
|
|
|
if (jv_get_kind(k) == JV_KIND_STRING &&
|
|
|
|
(jv_get_kind(t) == JV_KIND_OBJECT || isnull)) {
|
|
|
|
if (isnull) t = jv_object();
|
|
|
|
t = jv_object_set(t, k, v);
|
|
|
|
} else if (jv_get_kind(k) == JV_KIND_NUMBER &&
|
|
|
|
(jv_get_kind(t) == JV_KIND_ARRAY || isnull)) {
|
|
|
|
if (isnull) t = jv_array();
|
|
|
|
t = jv_array_set(t, (int)jv_number_value(k), v);
|
2012-09-02 16:31:59 +01:00
|
|
|
} else {
|
2012-09-11 10:37:44 +01:00
|
|
|
jv err = jv_invalid_with_msg(jv_string_fmt("Cannot update field at %s index of %s",
|
|
|
|
jv_kind_name(jv_get_kind(t)),
|
|
|
|
jv_kind_name(jv_get_kind(v))));
|
|
|
|
jv_free(t);
|
|
|
|
jv_free(k);
|
|
|
|
jv_free(v);
|
|
|
|
t = err;
|
2012-09-02 16:31:59 +01:00
|
|
|
}
|
|
|
|
return t;
|
|
|
|
}
|
|
|
|
|
|
|
|
static jv jv_insert(jv root, jv value, jv* path, int pathlen) {
|
|
|
|
if (pathlen == 0) {
|
|
|
|
jv_free(root);
|
|
|
|
return value;
|
|
|
|
}
|
2012-09-03 15:33:59 +01:00
|
|
|
return jv_modify(root, jv_copy(*path),
|
2012-09-02 16:31:59 +01:00
|
|
|
jv_insert(jv_lookup(jv_copy(root), jv_copy(*path)), value, path+1, pathlen-1));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2012-09-01 19:16:43 +01:00
|
|
|
|
|
|
|
|
|
|
|
|
2012-08-27 10:11:55 +01:00
|
|
|
#endif
|
2012-08-28 18:09:43 +01:00
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
|
|
|
true/false/null:
|
|
|
|
check kind
|
|
|
|
|
|
|
|
number:
|
|
|
|
introduce/eliminate jv
|
|
|
|
to integer
|
|
|
|
|
|
|
|
array:
|
|
|
|
copy
|
|
|
|
free
|
|
|
|
slice
|
|
|
|
index
|
|
|
|
update
|
|
|
|
|
|
|
|
updateslice?
|
|
|
|
|
|
|
|
|
|
|
|
*/
|