mirror of
https://github.com/akheron/jansson.git
synced 2025-04-05 21:45:04 +00:00
Merge branch 'master' of git://github.com/akheron/jansson into c++-docs
This commit is contained in:
commit
02e93f1d3c
17 changed files with 205 additions and 46 deletions
9
CHANGES
9
CHANGES
|
@ -1,3 +1,12 @@
|
|||
Version 1.3 (in development)
|
||||
============================
|
||||
|
||||
* New encoding flags:
|
||||
|
||||
- ``JSON_PRESERVE_ORDER``: Preserve the insertion order of object
|
||||
keys.
|
||||
|
||||
|
||||
Version 1.2
|
||||
===========
|
||||
|
||||
|
|
|
@ -1,5 +1,5 @@
|
|||
AC_PREREQ([2.59])
|
||||
AC_INIT([jansson], [1.2], [petri@digip.org])
|
||||
AC_INIT([jansson], [1.2+], [petri@digip.org])
|
||||
|
||||
AM_INIT_AUTOMAKE([1.10 foreign])
|
||||
|
||||
|
|
|
@ -595,6 +595,14 @@ can be ORed together to obtain *flags*.
|
|||
|
||||
.. versionadded:: 1.2
|
||||
|
||||
``JSON_PRESERVE_ORDER``
|
||||
If this flag is used, object keys in the output are sorted into the
|
||||
same order in which they were first inserted to the object. For
|
||||
example, decoding a JSON text and then encoding with this flag
|
||||
preserves the order of object keys.
|
||||
|
||||
.. versionadded:: 1.3
|
||||
|
||||
The following functions perform the actual JSON encoding. The result
|
||||
is in UTF-8.
|
||||
|
||||
|
|
|
@ -52,7 +52,7 @@ copyright = u'2009, 2010 Petri Lehtinen'
|
|||
# The short X.Y version.
|
||||
version = '1.2'
|
||||
# The full version, including alpha/beta/rc tags.
|
||||
release = '1.2'
|
||||
release = '1.2+'
|
||||
|
||||
# The language for content autogenerated by Sphinx. Refer to documentation
|
||||
# for a list of supported languages.
|
||||
|
|
31
src/dump.c
31
src/dump.c
|
@ -154,9 +154,16 @@ static int dump_string(const char *str, int ascii, dump_func dump, void *data)
|
|||
return dump("\"", 1, data);
|
||||
}
|
||||
|
||||
static int object_key_cmp(const void *key1, const void *key2)
|
||||
static int object_key_compare_keys(const void *key1, const void *key2)
|
||||
{
|
||||
return strcmp(*(const char **)key1, *(const char **)key2);
|
||||
return strcmp((*(const object_key_t **)key1)->key,
|
||||
(*(const object_key_t **)key2)->key);
|
||||
}
|
||||
|
||||
static int object_key_compare_serials(const void *key1, const void *key2)
|
||||
{
|
||||
return (*(const object_key_t **)key1)->serial -
|
||||
(*(const object_key_t **)key2)->serial;
|
||||
}
|
||||
|
||||
static int do_dump(const json_t *json, unsigned long flags, int depth,
|
||||
|
@ -290,36 +297,40 @@ static int do_dump(const json_t *json, unsigned long flags, int depth,
|
|||
if(dump_indent(flags, depth + 1, 0, dump, data))
|
||||
return -1;
|
||||
|
||||
if(flags & JSON_SORT_KEYS)
|
||||
if(flags & JSON_SORT_KEYS || flags & JSON_PRESERVE_ORDER)
|
||||
{
|
||||
/* Sort keys */
|
||||
|
||||
const char **keys;
|
||||
const object_key_t **keys;
|
||||
unsigned int size;
|
||||
unsigned int i;
|
||||
int (*cmp_func)(const void *, const void *);
|
||||
|
||||
size = json_object_size(json);
|
||||
keys = malloc(size * sizeof(const char *));
|
||||
keys = malloc(size * sizeof(object_key_t *));
|
||||
if(!keys)
|
||||
return -1;
|
||||
|
||||
i = 0;
|
||||
while(iter)
|
||||
{
|
||||
keys[i] = json_object_iter_key(iter);
|
||||
keys[i] = jsonp_object_iter_fullkey(iter);
|
||||
iter = json_object_iter_next((json_t *)json, iter);
|
||||
i++;
|
||||
}
|
||||
assert(i == size);
|
||||
|
||||
qsort(keys, size, sizeof(const char *), object_key_cmp);
|
||||
if(flags & JSON_SORT_KEYS)
|
||||
cmp_func = object_key_compare_keys;
|
||||
else
|
||||
cmp_func = object_key_compare_serials;
|
||||
|
||||
qsort(keys, size, sizeof(object_key_t *), cmp_func);
|
||||
|
||||
for(i = 0; i < size; i++)
|
||||
{
|
||||
const char *key;
|
||||
json_t *value;
|
||||
|
||||
key = keys[i];
|
||||
key = keys[i]->key;
|
||||
value = json_object_get(json, key);
|
||||
assert(value);
|
||||
|
||||
|
|
|
@ -247,31 +247,39 @@ int hashtable_set(hashtable_t *hashtable, void *key, void *value)
|
|||
bucket_t *bucket;
|
||||
unsigned int hash, index;
|
||||
|
||||
hash = hashtable->hash_key(key);
|
||||
|
||||
/* if the key already exists, delete it */
|
||||
hashtable_do_del(hashtable, key, hash);
|
||||
|
||||
/* rehash if the load ratio exceeds 1 */
|
||||
if(hashtable->size >= num_buckets(hashtable))
|
||||
if(hashtable_do_rehash(hashtable))
|
||||
return -1;
|
||||
|
||||
pair = malloc(sizeof(pair_t));
|
||||
if(!pair)
|
||||
return -1;
|
||||
|
||||
pair->key = key;
|
||||
pair->value = value;
|
||||
pair->hash = hash;
|
||||
list_init(&pair->list);
|
||||
|
||||
hash = hashtable->hash_key(key);
|
||||
index = hash % num_buckets(hashtable);
|
||||
bucket = &hashtable->buckets[index];
|
||||
pair = hashtable_find_pair(hashtable, bucket, key, hash);
|
||||
|
||||
insert_to_bucket(hashtable, bucket, &pair->list);
|
||||
if(pair)
|
||||
{
|
||||
if(hashtable->free_key)
|
||||
hashtable->free_key(key);
|
||||
if(hashtable->free_value)
|
||||
hashtable->free_value(pair->value);
|
||||
pair->value = value;
|
||||
}
|
||||
else
|
||||
{
|
||||
pair = malloc(sizeof(pair_t));
|
||||
if(!pair)
|
||||
return -1;
|
||||
|
||||
hashtable->size++;
|
||||
pair->key = key;
|
||||
pair->value = value;
|
||||
pair->hash = hash;
|
||||
list_init(&pair->list);
|
||||
|
||||
insert_to_bucket(hashtable, bucket, &pair->list);
|
||||
|
||||
hashtable->size++;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
|
|
@ -161,7 +161,7 @@ void hashtable_clear(hashtable_t *hashtable);
|
|||
void *hashtable_iter(hashtable_t *hashtable);
|
||||
|
||||
/**
|
||||
* hashtable_iter - Return an iterator at a specific key
|
||||
* hashtable_iter_at - Return an iterator at a specific key
|
||||
*
|
||||
* @hashtable: The hashtable object
|
||||
* @key: The key that the iterator should point to
|
||||
|
|
|
@ -173,6 +173,7 @@ json_t *json_load_file(const char *path, json_error_t *error);
|
|||
#define JSON_COMPACT 0x100
|
||||
#define JSON_ENSURE_ASCII 0x200
|
||||
#define JSON_SORT_KEYS 0x400
|
||||
#define JSON_PRESERVE_ORDER 0x800
|
||||
|
||||
char *json_dumps(const json_t *json, unsigned long flags);
|
||||
int json_dumpf(const json_t *json, FILE *output, unsigned long flags);
|
||||
|
|
|
@ -166,8 +166,9 @@ namespace json {
|
|||
// proxies an array element
|
||||
class ElementProxy {
|
||||
public:
|
||||
// constructor
|
||||
ElementProxy(json_t* array, unsigned int index) : _array(array), _index(index) {}
|
||||
ElementProxy(json_t* array, unsigned int index);
|
||||
ElementProxy(const ElementProxy& other);
|
||||
~ElementProxy();
|
||||
|
||||
// assign to the proxied element
|
||||
inline ElementProxy& operator=(const Value& value);
|
||||
|
@ -186,8 +187,9 @@ namespace json {
|
|||
// proxies an object property
|
||||
class PropertyProxy {
|
||||
public:
|
||||
// constructor
|
||||
PropertyProxy(json_t* array, const char* key) : _object(array), _key(key) {}
|
||||
PropertyProxy(json_t* object, const char *key);
|
||||
PropertyProxy(const PropertyProxy& other);
|
||||
~PropertyProxy();
|
||||
|
||||
// assign to the proxied element
|
||||
inline PropertyProxy& operator=(const Value& value);
|
||||
|
@ -199,8 +201,11 @@ namespace json {
|
|||
// array object we wrap
|
||||
json_t* _object;
|
||||
|
||||
// iterator pointing to property
|
||||
void* _iter;
|
||||
|
||||
// key of property
|
||||
const char* _key;
|
||||
char* _key;
|
||||
};
|
||||
|
||||
} // namespace json::detail
|
||||
|
|
|
@ -8,6 +8,8 @@
|
|||
#error "jansson.ipp may only be included from jansson.hpp"
|
||||
#endif
|
||||
|
||||
#include <string.h>
|
||||
|
||||
namespace json {
|
||||
namespace detail {
|
||||
// assignment operator
|
||||
|
@ -310,6 +312,20 @@ namespace json {
|
|||
return v;
|
||||
}
|
||||
|
||||
ElementProxy::ElementProxy(json_t* array, unsigned int index)
|
||||
: _array(array), _index(index) {
|
||||
json_incref(_array);
|
||||
}
|
||||
|
||||
ElementProxy::ElementProxy(const ElementProxy& other)
|
||||
: _array(other._array), _index(other._index) {
|
||||
json_incref(_array);
|
||||
}
|
||||
|
||||
ElementProxy::~ElementProxy() {
|
||||
json_decref(_array);
|
||||
}
|
||||
|
||||
// assign value to proxied array element
|
||||
ElementProxy& ElementProxy::operator=(const Value& value) {
|
||||
json_array_set(_array, _index, value.as_json());
|
||||
|
@ -321,14 +337,40 @@ namespace json {
|
|||
return json_array_get(_array, _index);
|
||||
}
|
||||
|
||||
PropertyProxy::PropertyProxy(json_t* object, const char* key)
|
||||
: _object(object), _key(0) {
|
||||
_iter = json_object_iter_at(object, key);
|
||||
if(!_iter)
|
||||
_key = strdup(key);
|
||||
json_incref(_object);
|
||||
}
|
||||
|
||||
PropertyProxy::PropertyProxy(const PropertyProxy& other)
|
||||
: _object(other._object), _iter(other._iter), _key(0) {
|
||||
if(other._key)
|
||||
_key = strdup(other._key);
|
||||
json_incref(_object);
|
||||
}
|
||||
|
||||
PropertyProxy::~PropertyProxy() {
|
||||
free(_key);
|
||||
json_decref(_object);
|
||||
}
|
||||
|
||||
// assign value to proxied object property
|
||||
PropertyProxy& PropertyProxy::operator=(const Value& value) {
|
||||
json_object_set(_object, _key, value.as_json());
|
||||
if(_iter)
|
||||
json_object_iter_set(_object, _iter, value.as_json());
|
||||
else
|
||||
json_object_set(_object, _key, value.as_json());
|
||||
return *this;
|
||||
}
|
||||
|
||||
json_t* PropertyProxy::as_json() const {
|
||||
return json_object_get(_object, _key);
|
||||
if(_iter)
|
||||
return json_object_iter_value(_iter);
|
||||
else
|
||||
return json_object_get(_object, _key);
|
||||
}
|
||||
|
||||
} // namespace json::detail
|
||||
|
|
|
@ -17,6 +17,7 @@
|
|||
typedef struct {
|
||||
json_t json;
|
||||
hashtable_t hashtable;
|
||||
unsigned long serial;
|
||||
int visited;
|
||||
} json_object_t;
|
||||
|
||||
|
@ -49,4 +50,11 @@ typedef struct {
|
|||
#define json_to_real(json_) container_of(json_, json_real_t, json)
|
||||
#define json_to_integer(json_) container_of(json_, json_integer_t, json)
|
||||
|
||||
typedef struct {
|
||||
unsigned long serial;
|
||||
char key[];
|
||||
} object_key_t;
|
||||
|
||||
const object_key_t *jsonp_object_iter_fullkey(void *iter);
|
||||
|
||||
#endif
|
||||
|
|
45
src/value.c
45
src/value.c
|
@ -25,9 +25,16 @@ static inline void json_init(json_t *json, json_type type)
|
|||
|
||||
/*** object ***/
|
||||
|
||||
static unsigned int hash_string(const void *key)
|
||||
/* This macro just returns a pointer that's a few bytes backwards from
|
||||
string. This makes it possible to pass a pointer to object_key_t
|
||||
when only the string inside it is used, without actually creating
|
||||
an object_key_t instance. */
|
||||
#define string_to_key(string) container_of(string, object_key_t, key)
|
||||
|
||||
static unsigned int hash_key(const void *ptr)
|
||||
{
|
||||
const char *str = (const char *)key;
|
||||
const char *str = ((const object_key_t *)ptr)->key;
|
||||
|
||||
unsigned int hash = 5381;
|
||||
unsigned int c;
|
||||
|
||||
|
@ -40,9 +47,10 @@ static unsigned int hash_string(const void *key)
|
|||
return hash;
|
||||
}
|
||||
|
||||
static int string_equal(const void *key1, const void *key2)
|
||||
static int key_equal(const void *ptr1, const void *ptr2)
|
||||
{
|
||||
return strcmp((const char *)key1, (const char *)key2) == 0;
|
||||
return strcmp(((const object_key_t *)ptr1)->key,
|
||||
((const object_key_t *)ptr2)->key) == 0;
|
||||
}
|
||||
|
||||
static void value_decref(void *value)
|
||||
|
@ -57,13 +65,14 @@ json_t *json_object(void)
|
|||
return NULL;
|
||||
json_init(&object->json, JSON_OBJECT);
|
||||
|
||||
if(hashtable_init(&object->hashtable, hash_string, string_equal,
|
||||
if(hashtable_init(&object->hashtable, hash_key, key_equal,
|
||||
free, value_decref))
|
||||
{
|
||||
free(object);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
object->serial = 0;
|
||||
object->visited = 0;
|
||||
|
||||
return &object->json;
|
||||
|
@ -94,12 +103,13 @@ json_t *json_object_get(const json_t *json, const char *key)
|
|||
return NULL;
|
||||
|
||||
object = json_to_object(json);
|
||||
return hashtable_get(&object->hashtable, key);
|
||||
return hashtable_get(&object->hashtable, string_to_key(key));
|
||||
}
|
||||
|
||||
int json_object_set_new_nocheck(json_t *json, const char *key, json_t *value)
|
||||
{
|
||||
json_object_t *object;
|
||||
object_key_t *k;
|
||||
|
||||
if(!key || !value)
|
||||
return -1;
|
||||
|
@ -111,7 +121,14 @@ int json_object_set_new_nocheck(json_t *json, const char *key, json_t *value)
|
|||
}
|
||||
object = json_to_object(json);
|
||||
|
||||
if(hashtable_set(&object->hashtable, strdup(key), value))
|
||||
k = malloc(sizeof(object_key_t) + strlen(key) + 1);
|
||||
if(!k)
|
||||
return -1;
|
||||
|
||||
k->serial = object->serial++;
|
||||
strcpy(k->key, key);
|
||||
|
||||
if(hashtable_set(&object->hashtable, k, value))
|
||||
{
|
||||
json_decref(value);
|
||||
return -1;
|
||||
|
@ -139,7 +156,7 @@ int json_object_del(json_t *json, const char *key)
|
|||
return -1;
|
||||
|
||||
object = json_to_object(json);
|
||||
return hashtable_del(&object->hashtable, key);
|
||||
return hashtable_del(&object->hashtable, string_to_key(key));
|
||||
}
|
||||
|
||||
int json_object_clear(json_t *json)
|
||||
|
@ -198,7 +215,7 @@ void *json_object_iter_at(json_t *json, const char *key)
|
|||
return NULL;
|
||||
|
||||
object = json_to_object(json);
|
||||
return hashtable_iter_at(&object->hashtable, key);
|
||||
return hashtable_iter_at(&object->hashtable, string_to_key(key));
|
||||
}
|
||||
|
||||
void *json_object_iter_next(json_t *json, void *iter)
|
||||
|
@ -212,12 +229,20 @@ void *json_object_iter_next(json_t *json, void *iter)
|
|||
return hashtable_iter_next(&object->hashtable, iter);
|
||||
}
|
||||
|
||||
const object_key_t *jsonp_object_iter_fullkey(void *iter)
|
||||
{
|
||||
if(!iter)
|
||||
return NULL;
|
||||
|
||||
return hashtable_iter_key(iter);
|
||||
}
|
||||
|
||||
const char *json_object_iter_key(void *iter)
|
||||
{
|
||||
if(!iter)
|
||||
return NULL;
|
||||
|
||||
return (const char *)hashtable_iter_key(iter);
|
||||
return jsonp_object_iter_fullkey(iter)->key;
|
||||
}
|
||||
|
||||
json_t *json_object_iter_value(void *iter)
|
||||
|
|
|
@ -53,6 +53,9 @@ int main(int argc, char *argv[])
|
|||
if(getenv_int("JSON_ENSURE_ASCII"))
|
||||
flags |= JSON_ENSURE_ASCII;
|
||||
|
||||
if(getenv_int("JSON_PRESERVE_ORDER"))
|
||||
flags |= JSON_PRESERVE_ORDER;
|
||||
|
||||
if(getenv_int("JSON_SORT_KEYS"))
|
||||
flags |= JSON_SORT_KEYS;
|
||||
|
||||
|
|
|
@ -402,6 +402,41 @@ static void test_misc()
|
|||
json_decref(object);
|
||||
}
|
||||
|
||||
static void test_preserve_order()
|
||||
{
|
||||
json_t *object;
|
||||
char *result;
|
||||
|
||||
const char *expected = "{\"foobar\": 1, \"bazquux\": 6, \"lorem ipsum\": 3, \"sit amet\": 5, \"helicopter\": 7}";
|
||||
|
||||
object = json_object();
|
||||
|
||||
json_object_set_new(object, "foobar", json_integer(1));
|
||||
json_object_set_new(object, "bazquux", json_integer(2));
|
||||
json_object_set_new(object, "lorem ipsum", json_integer(3));
|
||||
json_object_set_new(object, "dolor", json_integer(4));
|
||||
json_object_set_new(object, "sit amet", json_integer(5));
|
||||
|
||||
/* changing a value should preserve the order */
|
||||
json_object_set_new(object, "bazquux", json_integer(6));
|
||||
|
||||
/* deletion shouldn't change the order of others */
|
||||
json_object_del(object, "dolor");
|
||||
|
||||
/* add a new item just to make sure */
|
||||
json_object_set_new(object, "helicopter", json_integer(7));
|
||||
|
||||
result = json_dumps(object, JSON_PRESERVE_ORDER);
|
||||
|
||||
if(strcmp(expected, result) != 0) {
|
||||
fprintf(stderr, "%s != %s", expected, result);
|
||||
fail("JSON_PRESERVE_ORDER doesn't work");
|
||||
}
|
||||
|
||||
free(result);
|
||||
json_decref(object);
|
||||
}
|
||||
|
||||
int main()
|
||||
{
|
||||
test_misc();
|
||||
|
@ -410,6 +445,7 @@ int main()
|
|||
test_circular();
|
||||
test_set_nocheck();
|
||||
test_iterators();
|
||||
test_preserve_order();
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
|
1
test/suites/encoding-flags/preserve-order/env
Normal file
1
test/suites/encoding-flags/preserve-order/env
Normal file
|
@ -0,0 +1 @@
|
|||
export JSON_PRESERVE_ORDER=1
|
1
test/suites/encoding-flags/preserve-order/input
Normal file
1
test/suites/encoding-flags/preserve-order/input
Normal file
|
@ -0,0 +1 @@
|
|||
{"foo": 1, "bar": 2, "asdf": 3, "deadbeef": 4, "badc0ffee": 5, "qwerty": 6}
|
1
test/suites/encoding-flags/preserve-order/output
Normal file
1
test/suites/encoding-flags/preserve-order/output
Normal file
|
@ -0,0 +1 @@
|
|||
{"foo": 1, "bar": 2, "asdf": 3, "deadbeef": 4, "badc0ffee": 5, "qwerty": 6}
|
Loading…
Add table
Reference in a new issue