aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorPetri Lehtinen <petri@digip.org>2020-11-19 17:21:12 +0200
committerGitHub <noreply@github.com>2020-11-19 17:21:12 +0200
commit9a0fc069bfa2a7111ae3ca2803baff778f0626f7 (patch)
tree3b0b6bd5642c7a6b11d3955acb8327612103b597
parentcc318fc042f552c94bdacbe82135bafbbbf4c8d2 (diff)
parent0758caaac02826fbda6179ce072cd37d343dc1cf (diff)
downloadjansson-9a0fc069bfa2a7111ae3ca2803baff778f0626f7.zip
jansson-9a0fc069bfa2a7111ae3ca2803baff778f0626f7.tar.gz
jansson-9a0fc069bfa2a7111ae3ca2803baff778f0626f7.tar.bz2
Merge pull request #520 from Mephistophiles/getn
Add support getn, setn functions
-rw-r--r--CMakeLists.txt5
-rw-r--r--doc/apiref.rst160
-rw-r--r--doc/conformance.rst3
-rw-r--r--src/dump.c53
-rw-r--r--src/hashtable.c88
-rw-r--r--src/hashtable.h20
-rw-r--r--src/jansson.def5
-rw-r--r--src/jansson.h33
-rw-r--r--src/jansson_private.h4
-rw-r--r--src/load.c4
-rw-r--r--src/pack_unpack.c8
-rw-r--r--src/value.c92
-rw-r--r--test/suites/api/Makefile.am4
-rw-r--r--test/suites/api/test_fixed_size.c205
14 files changed, 594 insertions, 90 deletions
diff --git a/CMakeLists.txt b/CMakeLists.txt
index 02b9785..7956fc3 100644
--- a/CMakeLists.txt
+++ b/CMakeLists.txt
@@ -508,14 +508,15 @@ if (NOT JANSSON_WITHOUT_TESTS)
set(api_tests
test_array
- test_copy
test_chaos
+ test_copy
test_dump
test_dump_callback
test_equal
+ test_fixed_size
test_load
- test_loadb
test_load_callback
+ test_loadb
test_number
test_object
test_pack
diff --git a/doc/apiref.rst b/doc/apiref.rst
index bd622e3..8ef190e 100644
--- a/doc/apiref.rst
+++ b/doc/apiref.rst
@@ -648,6 +648,15 @@ allowed in object keys.
Get a value corresponding to *key* from *object*. Returns *NULL* if
*key* is not found and on error.
+.. function:: json_t *json_object_getn(const json_t *object, const char *key, size_t key_len)
+
+ .. refcounting:: borrow
+
+ Like :func:`json_object_get`, but give the fixed-length *key* with length *key_len*.
+ See :ref:`fixed_length_keys` for details.
+
+ .. versionadded:: 2.14
+
.. function:: int json_object_set(json_t *object, const char *key, json_t *value)
Set the value of *key* to *value* in *object*. *key* must be a
@@ -655,6 +664,13 @@ allowed in object keys.
already is a value for *key*, it is replaced by the new value.
Returns 0 on success and -1 on error.
+.. function:: int json_object_setn(json_t *object, const char *key, size_t key_len, json_t *value)
+
+ Like :func:`json_object_set`, but give the fixed-length *key* with length *key_len*.
+ See :ref:`fixed_length_keys` for details.
+
+ .. versionadded:: 2.14
+
.. function:: int json_object_set_nocheck(json_t *object, const char *key, json_t *value)
Like :func:`json_object_set`, but doesn't check that *key* is
@@ -662,12 +678,26 @@ allowed in object keys.
really is the case (e.g. you have already checked it by other
means).
+.. function:: int json_object_setn_nocheck(json_t *object, const char *key, size_t key_len, json_t *value)
+
+ Like :func:`json_object_set_nocheck`, but give the fixed-length *key* with length *key_len*.
+ See :ref:`fixed_length_keys` for details.
+
+ .. versionadded:: 2.14
+
.. function:: int json_object_set_new(json_t *object, const char *key, json_t *value)
Like :func:`json_object_set()` but steals the reference to
*value*. This is useful when *value* is newly created and not used
after the call.
+.. function:: int json_object_setn_new(json_t *object, const char *key, size_t key_len, json_t *value)
+
+ Like :func:`json_object_set_new`, but give the fixed-length *key* with length *key_len*.
+ See :ref:`fixed_length_keys` for details.
+
+ .. versionadded:: 2.14
+
.. function:: int json_object_set_new_nocheck(json_t *object, const char *key, json_t *value)
Like :func:`json_object_set_new`, but doesn't check that *key* is
@@ -675,12 +705,26 @@ allowed in object keys.
really is the case (e.g. you have already checked it by other
means).
+.. function:: int json_object_setn_new_nocheck(json_t *object, const char *key, size_t key_len, json_t *value)
+
+ Like :func:`json_object_set_new_nocheck`, but give the fixed-length *key* with length *key_len*.
+ See :ref:`fixed_length_keys` for details.
+
+ .. versionadded:: 2.14
+
.. function:: int json_object_del(json_t *object, const char *key)
Delete *key* from *object* if it exists. Returns 0 on success, or
-1 if *key* was not found. The reference count of the removed value
is decremented.
+.. function:: int json_object_deln(json_t *object, const char *key, size_t key_len)
+
+ Like :func:`json_object_del`, but give the fixed-length *key* with length *key_len*.
+ See :ref:`fixed_length_keys` for details.
+
+ .. versionadded:: 2.14
+
.. function:: int json_object_clear(json_t *object)
Remove all elements from *object*. Returns 0 on success and -1 if
@@ -750,7 +794,7 @@ allowed in object keys.
The items are returned in the order they were inserted to the
object.
- **Note:** It's not safe to call ``json_object_del(object, key)``
+ **Note:** It's not safe to call ``json_object_del(object, key)`` or ``json_object_deln(object, key, key_len)``
during iteration. If you need to, use
:func:`json_object_foreach_safe` instead.
@@ -767,11 +811,39 @@ allowed in object keys.
.. function:: void json_object_foreach_safe(object, tmp, key, value)
Like :func:`json_object_foreach()`, but it's safe to call
- ``json_object_del(object, key)`` during iteration. You need to pass
- an extra ``void *`` parameter ``tmp`` that is used for temporary storage.
+ ``json_object_del(object, key)`` or ``json_object_deln(object, key, key_len)`` during iteration.
+ You need to pass an extra ``void *`` parameter ``tmp`` that is used for temporary storage.
.. versionadded:: 2.8
+.. function:: void json_object_keylen_foreach(object, key, key_len, value)
+
+ Like :c:func:`json_object_foreach`, but in *key_len* stored length of the *key*.
+ Example::
+
+ /* obj is a JSON object */
+ const char *key;
+ json_t *value;
+ size_t len;
+
+ json_object_keylen_foreach(obj, key, len, value) {
+ printf("got key %s with length %zu\n", key, len);
+ }
+
+ **Note:** It's not safe to call ``json_object_deln(object, key, key_len)``
+ during iteration. If you need to, use
+ :func:`json_object_keylen_foreach_safe` instead.
+
+ .. versionadded:: 2.14
+
+
+.. function:: void json_object_keylen_foreach_safe(object, tmp, key, key_len, value)
+
+ Like :func:`json_object_keylen_foreach()`, but it's safe to call
+ ``json_object_deln(object, key, key_len)`` during iteration.
+ You need to pass an extra ``void *`` parameter ``tmp`` that is used for temporary storage.
+
+ .. versionadded:: 2.14
The following functions can be used to iterate through all key-value
pairs in an object. The items are returned in the order they were
@@ -800,6 +872,12 @@ inserted to the object.
Extract the associated key from *iter*.
+.. function:: size_t json_object_iter_key_len(void *iter)
+
+ Extract the associated key length from *iter*.
+
+ .. versionadded:: 2.14
+
.. function:: json_t *json_object_iter_value(void *iter)
.. refcounting:: borrow
@@ -1909,3 +1987,79 @@ memory, see
http://www.dwheeler.com/secure-programs/Secure-Programs-HOWTO/protect-secrets.html.
The page also explains the :func:`guaranteed_memset()` function used
in the example and gives a sample implementation for it.
+
+.. _fixed_length_keys:
+
+Fixed-Length keys
+=================
+
+The Jansson API allows work with fixed-length keys. This can be useful in the following cases:
+
+* The key is contained inside a buffer and is not null-terminated. In this case creating a new temporary buffer is not needed.
+* The key contains U+0000 inside it.
+
+List of API for fixed-length keys:
+
+* :c:func:`json_object_getn`
+* :c:func:`json_object_setn`
+* :c:func:`json_object_setn_nocheck`
+* :c:func:`json_object_setn_new`
+* :c:func:`json_object_setn_new_nocheck`
+* :c:func:`json_object_deln`
+* :c:func:`json_object_iter_key_len`
+* :c:func:`json_object_keylen_foreach`
+* :c:func:`json_object_keylen_foreach_safe`
+
+**Examples:**
+
+Try to write a new function to get :c:struct:`json_t` by path separated by ``.``
+
+This requires:
+
+* string iterator (no need to modify the input for better performance)
+* API for working with fixed-size keys
+
+The iterator::
+
+ struct string {
+ const char *string;
+ size_t length;
+ };
+
+ size_t string_try_next(struct string *str, const char *delimiter) {
+ str->string += strspn(str->string, delimiter);
+ str->length = strcspn(str->string, delimiter);
+ return str->length;
+ }
+
+ #define string_foreach(_string, _delimiter) \
+ for (; string_try_next(&(_string), _delimiter); (_string).string += (_string).length)
+
+
+The function::
+
+ json_t *json_object_get_by_path(json_t *object, const char *path) {
+ struct string str;
+ json_t *out = object;
+
+ str.string = path;
+
+ string_foreach(str, ".") {
+ out = json_object_getn(out, str.string, str.length);
+ if (out == NULL)
+ return NULL;
+ }
+
+ return out;
+ }
+
+And usage::
+
+ int main(void) {
+ json_t *obj = json_pack("{s:{s:{s:b}}}", "a", "b", "c", 1);
+
+ json_t *c = json_object_get_by_path(obj, "a.b.c");
+ assert(json_is_true(c));
+
+ json_decref(obj);
+ }
diff --git a/doc/conformance.rst b/doc/conformance.rst
index c1766bc..5556a6b 100644
--- a/doc/conformance.rst
+++ b/doc/conformance.rst
@@ -22,8 +22,7 @@ JSON strings are mapped to C-style null-terminated character arrays,
and UTF-8 encoding is used internally.
All Unicode codepoints U+0000 through U+10FFFF are allowed in string
-values. However, U+0000 is not allowed in object keys because of API
-restrictions.
+values. However, U+0000 is allowed in object keys only for length-aware functions.
Unicode normalization or any other transformation is never performed
on any strings (string values or object keys). When checking for
diff --git a/src/dump.c b/src/dump.c
index 1baa248..c0fb06e 100644
--- a/src/dump.c
+++ b/src/dump.c
@@ -195,8 +195,21 @@ static int dump_string(const char *str, size_t len, json_dump_callback_t dump, v
return dump("\"", 1, data);
}
+struct key_len {
+ const char *key;
+ int len;
+};
+
static int compare_keys(const void *key1, const void *key2) {
- return strcmp(*(const char **)key1, *(const char **)key2);
+ const struct key_len *k1 = key1;
+ const struct key_len *k2 = key2;
+ const size_t min_size = k1->len < k2->len ? k1->len : k2->len;
+ int res = memcmp(k1->key, k2->key, min_size);
+
+ if (res)
+ return res;
+
+ return k1->len - k2->len;
}
static int do_dump(const json_t *json, size_t flags, int depth, hashtable_t *parents,
@@ -253,9 +266,10 @@ static int do_dump(const json_t *json, size_t flags, int depth, hashtable_t *par
/* Space for "0x", double the sizeof a pointer for the hex and a
* terminator. */
char key[2 + (sizeof(json) * 2) + 1];
+ size_t key_len;
/* detect circular references */
- if (jsonp_loop_check(parents, json, key, sizeof(key)))
+ if (jsonp_loop_check(parents, json, key, sizeof(key), &key_len))
return -1;
n = json_array_size(json);
@@ -263,7 +277,7 @@ static int do_dump(const json_t *json, size_t flags, int depth, hashtable_t *par
if (!embed && dump("[", 1, data))
return -1;
if (n == 0) {
- hashtable_del(parents, key);
+ hashtable_del(parents, key, key_len);
return embed ? 0 : dump("]", 1, data);
}
if (dump_indent(flags, depth + 1, 0, dump, data))
@@ -284,7 +298,7 @@ static int do_dump(const json_t *json, size_t flags, int depth, hashtable_t *par
}
}
- hashtable_del(parents, key);
+ hashtable_del(parents, key, key_len);
return embed ? 0 : dump("]", 1, data);
}
@@ -293,6 +307,7 @@ static int do_dump(const json_t *json, size_t flags, int depth, hashtable_t *par
const char *separator;
int separator_length;
char loop_key[LOOP_KEY_LEN];
+ size_t loop_key_len;
if (flags & JSON_COMPACT) {
separator = ":";
@@ -303,7 +318,8 @@ static int do_dump(const json_t *json, size_t flags, int depth, hashtable_t *par
}
/* detect circular references */
- if (jsonp_loop_check(parents, json, loop_key, sizeof(loop_key)))
+ if (jsonp_loop_check(parents, json, loop_key, sizeof(loop_key),
+ &loop_key_len))
return -1;
iter = json_object_iter((json_t *)json);
@@ -311,40 +327,44 @@ static int do_dump(const json_t *json, size_t flags, int depth, hashtable_t *par
if (!embed && dump("{", 1, data))
return -1;
if (!iter) {
- hashtable_del(parents, loop_key);
+ hashtable_del(parents, loop_key, loop_key_len);
return embed ? 0 : dump("}", 1, data);
}
if (dump_indent(flags, depth + 1, 0, dump, data))
return -1;
if (flags & JSON_SORT_KEYS) {
- const char **keys;
+ struct key_len *keys;
size_t size, i;
size = json_object_size(json);
- keys = jsonp_malloc(size * sizeof(const char *));
+ keys = jsonp_malloc(size * sizeof(struct key_len));
if (!keys)
return -1;
i = 0;
while (iter) {
- keys[i] = json_object_iter_key(iter);
+ struct key_len *keylen = &keys[i];
+
+ keylen->key = json_object_iter_key(iter);
+ keylen->len = json_object_iter_key_len(iter);
+
iter = json_object_iter_next((json_t *)json, iter);
i++;
}
assert(i == size);
- qsort(keys, size, sizeof(const char *), compare_keys);
+ qsort(keys, size, sizeof(struct key_len), compare_keys);
for (i = 0; i < size; i++) {
- const char *key;
+ const struct key_len *key;
json_t *value;
- key = keys[i];
- value = json_object_get(json, key);
+ key = &keys[i];
+ value = json_object_getn(json, key->key, key->len);
assert(value);
- dump_string(key, strlen(key), dump, data, flags);
+ dump_string(key->key, key->len, dump, data, flags);
if (dump(separator, separator_length, data) ||
do_dump(value, flags, depth + 1, parents, dump, data)) {
jsonp_free(keys);
@@ -372,8 +392,9 @@ static int do_dump(const json_t *json, size_t flags, int depth, hashtable_t *par
while (iter) {
void *next = json_object_iter_next((json_t *)json, iter);
const char *key = json_object_iter_key(iter);
+ const size_t key_len = json_object_iter_key_len(iter);
- dump_string(key, strlen(key), dump, data, flags);
+ dump_string(key, key_len, dump, data, flags);
if (dump(separator, separator_length, data) ||
do_dump(json_object_iter_value(iter), flags, depth + 1, parents,
dump, data))
@@ -392,7 +413,7 @@ static int do_dump(const json_t *json, size_t flags, int depth, hashtable_t *par
}
}
- hashtable_del(parents, loop_key);
+ hashtable_del(parents, loop_key, loop_key_len);
return embed ? 0 : dump("}", 1, data);
}
diff --git a/src/hashtable.c b/src/hashtable.c
index fd1e706..1508d74 100644
--- a/src/hashtable.c
+++ b/src/hashtable.c
@@ -35,7 +35,7 @@ extern volatile uint32_t hashtable_seed;
#define list_to_pair(list_) container_of(list_, pair_t, list)
#define ordered_list_to_pair(list_) container_of(list_, pair_t, ordered_list)
-#define hash_str(key) ((size_t)hashlittle((key), strlen(key), hashtable_seed))
+#define hash_str(key, len) ((size_t)hashlittle((key), len, hashtable_seed))
static JSON_INLINE void list_init(list_t *list) {
list->next = list;
@@ -69,7 +69,7 @@ static void insert_to_bucket(hashtable_t *hashtable, bucket_t *bucket, list_t *l
}
static pair_t *hashtable_find_pair(hashtable_t *hashtable, bucket_t *bucket,
- const char *key, size_t hash) {
+ const char *key, size_t key_len, size_t hash) {
list_t *list;
pair_t *pair;
@@ -79,7 +79,8 @@ static pair_t *hashtable_find_pair(hashtable_t *hashtable, bucket_t *bucket,
list = bucket->first;
while (1) {
pair = list_to_pair(list);
- if (pair->hash == hash && strcmp(pair->key, key) == 0)
+ if (pair->hash == hash && pair->key_len == key_len &&
+ memcmp(pair->key, key, key_len) == 0)
return pair;
if (list == bucket->last)
@@ -92,7 +93,8 @@ static pair_t *hashtable_find_pair(hashtable_t *hashtable, bucket_t *bucket,
}
/* returns 0 on success, -1 if key was not found */
-static int hashtable_do_del(hashtable_t *hashtable, const char *key, size_t hash) {
+static int hashtable_do_del(hashtable_t *hashtable, const char *key, size_t key_len,
+ size_t hash) {
pair_t *pair;
bucket_t *bucket;
size_t index;
@@ -100,7 +102,7 @@ static int hashtable_do_del(hashtable_t *hashtable, const char *key, size_t hash
index = hash & hashmask(hashtable->order);
bucket = &hashtable->buckets[index];
- pair = hashtable_find_pair(hashtable, bucket, key, hash);
+ pair = hashtable_find_pair(hashtable, bucket, key, key_len, hash);
if (!pair)
return -1;
@@ -193,7 +195,37 @@ void hashtable_close(hashtable_t *hashtable) {
jsonp_free(hashtable->buckets);
}
-int hashtable_set(hashtable_t *hashtable, const char *key, json_t *value) {
+static pair_t *init_pair(json_t *value, const char *key, size_t key_len, size_t hash) {
+ pair_t *pair;
+
+ /* offsetof(...) returns the size of pair_t without the last,
+ flexible member. This way, the correct amount is
+ allocated. */
+
+ if (key_len >= (size_t)-1 - offsetof(pair_t, key)) {
+ /* Avoid an overflow if the key is very long */
+ return NULL;
+ }
+
+ pair = jsonp_malloc(offsetof(pair_t, key) + key_len + 1);
+
+ if (!pair)
+ return NULL;
+
+ pair->hash = hash;
+ memcpy(pair->key, key, key_len);
+ pair->key[key_len] = '\0';
+ pair->key_len = key_len;
+ pair->value = value;
+
+ list_init(&pair->list);
+ list_init(&pair->ordered_list);
+
+ return pair;
+}
+
+int hashtable_set(hashtable_t *hashtable, const char *key, size_t key_len,
+ json_t *value) {
pair_t *pair;
bucket_t *bucket;
size_t hash, index;
@@ -203,35 +235,20 @@ int hashtable_set(hashtable_t *hashtable, const char *key, json_t *value) {
if (hashtable_do_rehash(hashtable))
return -1;
- hash = hash_str(key);
+ hash = hash_str(key, key_len);
index = hash & hashmask(hashtable->order);
bucket = &hashtable->buckets[index];
- pair = hashtable_find_pair(hashtable, bucket, key, hash);
+ pair = hashtable_find_pair(hashtable, bucket, key, key_len, hash);
if (pair) {
json_decref(pair->value);
pair->value = value;
} else {
- /* offsetof(...) returns the size of pair_t without the last,
- flexible member. This way, the correct amount is
- allocated. */
-
- size_t len = strlen(key);
- if (len >= (size_t)-1 - offsetof(pair_t, key)) {
- /* Avoid an overflow if the key is very long */
- return -1;
- }
+ pair = init_pair(value, key, key_len, hash);
- pair = jsonp_malloc(offsetof(pair_t, key) + len + 1);
if (!pair)
return -1;
- pair->hash = hash;
- strncpy(pair->key, key, len + 1);
- pair->value = value;
- list_init(&pair->list);
- list_init(&pair->ordered_list);
-
insert_to_bucket(hashtable, bucket, &pair->list);
list_insert(&hashtable->ordered_list, &pair->ordered_list);
@@ -240,24 +257,24 @@ int hashtable_set(hashtable_t *hashtable, const char *key, json_t *value) {
return 0;
}
-void *hashtable_get(hashtable_t *hashtable, const char *key) {
+void *hashtable_get(hashtable_t *hashtable, const char *key, size_t key_len) {
pair_t *pair;
size_t hash;
bucket_t *bucket;
- hash = hash_str(key);
+ hash = hash_str(key, key_len);
bucket = &hashtable->buckets[hash & hashmask(hashtable->order)];
- pair = hashtable_find_pair(hashtable, bucket, key, hash);
+ pair = hashtable_find_pair(hashtable, bucket, key, key_len, hash);
if (!pair)
return NULL;
return pair->value;
}
-int hashtable_del(hashtable_t *hashtable, const char *key) {
- size_t hash = hash_str(key);
- return hashtable_do_del(hashtable, key, hash);
+int hashtable_del(hashtable_t *hashtable, const char *key, size_t key_len) {
+ size_t hash = hash_str(key, key_len);
+ return hashtable_do_del(hashtable, key, key_len, hash);
}
void hashtable_clear(hashtable_t *hashtable) {
@@ -278,15 +295,15 @@ void *hashtable_iter(hashtable_t *hashtable) {
return hashtable_iter_next(hashtable, &hashtable->ordered_list);
}
-void *hashtable_iter_at(hashtable_t *hashtable, const char *key) {
+void *hashtable_iter_at(hashtable_t *hashtable, const char *key, size_t key_len) {
pair_t *pair;
size_t hash;
bucket_t *bucket;
- hash = hash_str(key);
+ hash = hash_str(key, key_len);
bucket = &hashtable->buckets[hash & hashmask(hashtable->order)];
- pair = hashtable_find_pair(hashtable, bucket, key, hash);
+ pair = hashtable_find_pair(hashtable, bucket, key, key_len, hash);
if (!pair)
return NULL;
@@ -305,6 +322,11 @@ void *hashtable_iter_key(void *iter) {
return pair->key;
}
+size_t hashtable_iter_key_len(void *iter) {
+ pair_t *pair = ordered_list_to_pair((list_t *)iter);
+ return pair->key_len;
+}
+
void *hashtable_iter_value(void *iter) {
pair_t *pair = ordered_list_to_pair((list_t *)iter);
return pair->value;
diff --git a/src/hashtable.h b/src/hashtable.h
index 6c4845d..03a1f5a 100644
--- a/src/hashtable.h
+++ b/src/hashtable.h
@@ -24,6 +24,7 @@ struct hashtable_pair {
struct hashtable_list ordered_list;
size_t hash;
json_t *value;
+ size_t key_len;
char key[1];
};
@@ -69,6 +70,7 @@ void hashtable_close(hashtable_t *hashtable);
*
* @hashtable: The hashtable object
* @key: The key
+ * @key: The length of key
* @serial: For addition order of keys
* @value: The value
*
@@ -79,27 +81,29 @@ void hashtable_close(hashtable_t *hashtable);
*
* Returns 0 on success, -1 on failure (out of memory).
*/
-int hashtable_set(hashtable_t *hashtable, const char *key, json_t *value);
+int hashtable_set(hashtable_t *hashtable, const char *key, size_t key_len, json_t *value);
/**
* hashtable_get - Get a value associated with a key
*
* @hashtable: The hashtable object
* @key: The key
+ * @key: The length of key
*
* Returns value if it is found, or NULL otherwise.
*/
-void *hashtable_get(hashtable_t *hashtable, const char *key);
+void *hashtable_get(hashtable_t *hashtable, const char *key, size_t key_len);
/**
* hashtable_del - Remove a value from the hashtable
*
* @hashtable: The hashtable object
* @key: The key
+ * @key: The length of key
*
* Returns 0 on success, or -1 if the key was not found.
*/
-int hashtable_del(hashtable_t *hashtable, const char *key);
+int hashtable_del(hashtable_t *hashtable, const char *key, size_t key_len);
/**
* hashtable_clear - Clear hashtable
@@ -132,11 +136,12 @@ void *hashtable_iter(hashtable_t *hashtable);
*
* @hashtable: The hashtable object
* @key: The key that the iterator should point to
+ * @key: The length of key
*
* Like hashtable_iter() but returns an iterator pointing to a
* specific key.
*/
-void *hashtable_iter_at(hashtable_t *hashtable, const char *key);
+void *hashtable_iter_at(hashtable_t *hashtable, const char *key, size_t key_len);
/**
* hashtable_iter_next - Advance an iterator
@@ -157,6 +162,13 @@ void *hashtable_iter_next(hashtable_t *hashtable, void *iter);
void *hashtable_iter_key(void *iter);
/**
+ * hashtable_iter_key_len - Retrieve the key length pointed by an iterator
+ *
+ * @iter: The iterator
+ */
+size_t hashtable_iter_key_len(void *iter);
+
+/**
* hashtable_iter_value - Retrieve the value pointed by an iterator
*
* @iter: The iterator
diff --git a/src/jansson.def b/src/jansson.def
index 55b39c8..5c76c2f 100644
--- a/src/jansson.def
+++ b/src/jansson.def
@@ -34,9 +34,13 @@ EXPORTS
json_object
json_object_size
json_object_get
+ json_object_getn
json_object_set_new
+ json_object_setn_new
json_object_set_new_nocheck
+ json_object_setn_new_nocheck
json_object_del
+ json_object_deln
json_object_clear
json_object_update
json_object_update_existing
@@ -46,6 +50,7 @@ EXPORTS
json_object_iter_at
json_object_iter_next
json_object_iter_key
+ json_object_iter_key_len
json_object_iter_value
json_object_iter_set_new
json_object_key_to_iter
diff --git a/src/jansson.h b/src/jansson.h
index fbc7381..b93a401 100644
--- a/src/jansson.h
+++ b/src/jansson.h
@@ -188,9 +188,15 @@ void json_object_seed(size_t seed);
size_t json_object_size(const json_t *object);
json_t *json_object_get(const json_t *object, const char *key)
JANSSON_ATTRS((warn_unused_result));
+json_t *json_object_getn(const json_t *object, const char *key, size_t key_len)
+ JANSSON_ATTRS((warn_unused_result));
int json_object_set_new(json_t *object, const char *key, json_t *value);
+int json_object_setn_new(json_t *object, const char *key, size_t key_len, json_t *value);
int json_object_set_new_nocheck(json_t *object, const char *key, json_t *value);
+int json_object_setn_new_nocheck(json_t *object, const char *key, size_t key_len,
+ json_t *value);
int json_object_del(json_t *object, const char *key);
+int json_object_deln(json_t *object, const char *key, size_t key_len);
int json_object_clear(json_t *object);
int json_object_update(json_t *object, json_t *other);
int json_object_update_existing(json_t *object, json_t *other);
@@ -201,6 +207,7 @@ void *json_object_iter_at(json_t *object, const char *key);
void *json_object_key_to_iter(const char *key);
void *json_object_iter_next(json_t *object, void *iter);
const char *json_object_iter_key(void *iter);
+size_t json_object_iter_key_len(void *iter);
json_t *json_object_iter_value(void *iter);
int json_object_iter_set_new(json_t *object, void *iter, json_t *value);
@@ -210,6 +217,14 @@ int json_object_iter_set_new(json_t *object, void *iter, json_t *value);
key = json_object_iter_key( \
json_object_iter_next(object, json_object_key_to_iter(key))))
+#define json_object_keylen_foreach(object, key, key_len, value) \
+ for (key = json_object_iter_key(json_object_iter(object)), \
+ key_len = json_object_iter_key_len(json_object_key_to_iter(key)); \
+ key && (value = json_object_iter_value(json_object_key_to_iter(key))); \
+ key = json_object_iter_key( \
+ json_object_iter_next(object, json_object_key_to_iter(key))), \
+ key_len = json_object_iter_key_len(json_object_key_to_iter(key)))
+
#define json_object_foreach_safe(object, n, key, value) \
for (key = json_object_iter_key(json_object_iter(object)), \
n = json_object_iter_next(object, json_object_key_to_iter(key)); \
@@ -217,6 +232,14 @@ int json_object_iter_set_new(json_t *object, void *iter, json_t *value);
key = json_object_iter_key(n), \
n = json_object_iter_next(object, json_object_key_to_iter(key)))
+#define json_object_keylen_foreach_safe(object, n, key, key_len, value) \
+ for (key = json_object_iter_key(json_object_iter(object)), \
+ n = json_object_iter_next(object, json_object_key_to_iter(key)), \
+ key_len = json_object_iter_key_len(json_object_key_to_iter(key)); \
+ key && (value = json_object_iter_value(json_object_key_to_iter(key))); \
+ key = json_object_iter_key(n), key_len = json_object_iter_key_len(n), \
+ n = json_object_iter_next(object, json_object_key_to_iter(key)))
+
#define json_array_foreach(array, index, value) \
for (index = 0; \
index < json_array_size(array) && (value = json_array_get(array, index)); \
@@ -226,11 +249,21 @@ static JSON_INLINE int json_object_set(json_t *object, const char *key, json_t *
return json_object_set_new(object, key, json_incref(value));
}
+static JSON_INLINE int json_object_setn(json_t *object, const char *key, size_t key_len,
+ json_t *value) {
+ return json_object_setn_new(object, key, key_len, json_incref(value));
+}
+
static JSON_INLINE int json_object_set_nocheck(json_t *object, const char *key,
json_t *value) {
return json_object_set_new_nocheck(object, key, json_incref(value));
}
+static JSON_INLINE int json_object_setn_nocheck(json_t *object, const char *key,
+ size_t key_len, json_t *value) {
+ return json_object_setn_new_nocheck(object, key, key_len, json_incref(value));
+}
+
static JSON_INLINE int json_object_iter_set(json_t *object, void *iter, json_t *value) {
return json_object_iter_set_new(object, iter, json_incref(value));
}
diff --git a/src/jansson_private.h b/src/jansson_private.h
index d3b2a64..ea2593c 100644
--- a/src/jansson_private.h
+++ b/src/jansson_private.h
@@ -91,8 +91,8 @@ char *jsonp_strndup(const char *str, size_t len) JANSSON_ATTRS((warn_unused_resu
/* Circular reference check*/
/* Space for "0x", double the sizeof a pointer for the hex and a terminator. */
#define LOOP_KEY_LEN (2 + (sizeof(json_t *) * 2) + 1)
-int jsonp_loop_check(hashtable_t *parents, const json_t *json, char *key,
- size_t key_size);
+int jsonp_loop_check(hashtable_t *parents, const json_t *json, char *key, size_t key_size,
+ size_t *key_len_out);
/* Windows compatibility */
#if defined(_WIN32) || defined(WIN32)
diff --git a/src/load.c b/src/load.c
index ea6692f..8ae7abd 100644
--- a/src/load.c
+++ b/src/load.c
@@ -689,7 +689,7 @@ static json_t *parse_object(lex_t *lex, size_t flags, json_error_t *error) {
}
if (flags & JSON_REJECT_DUPLICATES) {
- if (json_object_get(object, key)) {
+ if (json_object_getn(object, key, len)) {
jsonp_free(key);
error_set(error, lex, json_error_duplicate_key, "duplicate object key");
goto error;
@@ -710,7 +710,7 @@ static json_t *parse_object(lex_t *lex, size_t flags, json_error_t *error) {
goto error;
}
- if (json_object_set_new_nocheck(object, key, value)) {
+ if (json_object_setn_new_nocheck(object, key, len, value)) {
jsonp_free(key);
goto error;
}
diff --git a/src/pack_unpack.c b/src/pack_unpack.c
index a89be58..04c116e 100644
--- a/src/pack_unpack.c
+++ b/src/pack_unpack.c
@@ -544,7 +544,7 @@ static int unpack_object(scanner_t *s, json_t *root, va_list *ap) {
if (unpack(s, value, ap))
goto out;
- hashtable_set(&key_set, key, json_null());
+ hashtable_set(&key_set, key, strlen(key), json_null());
next_token(s);
}
@@ -554,6 +554,7 @@ static int unpack_object(scanner_t *s, json_t *root, va_list *ap) {
if (root && strict == 1) {
/* We need to check that all non optional items have been parsed */
const char *key;
+ size_t key_len;
/* keys_res is 1 for uninitialized, 0 for success, -1 for error. */
int keys_res = 1;
strbuffer_t unrecognized_keys;
@@ -562,7 +563,8 @@ static int unpack_object(scanner_t *s, json_t *root, va_list *ap) {
if (gotopt || json_object_size(root) != key_set.size) {
json_object_foreach(root, key, value) {
- if (!hashtable_get(&key_set, key)) {
+ key_len = strlen(key);
+ if (!hashtable_get(&key_set, key, key_len)) {
unpacked++;
/* Save unrecognized keys for the error message */
@@ -574,7 +576,7 @@ static int unpack_object(scanner_t *s, json_t *root, va_list *ap) {
if (!keys_res)
keys_res =
- strbuffer_append_bytes(&unrecognized_keys, key, strlen(key));
+ strbuffer_append_bytes(&unrecognized_keys, key, key_len);
}
}
}
diff --git a/src/value.c b/src/value.c
index c85a0b4..07af087 100644
--- a/src/value.c
+++ b/src/value.c
@@ -44,13 +44,17 @@ static JSON_INLINE void json_init(json_t *json, json_type type) {
json->refcount = 1;
}
-int jsonp_loop_check(hashtable_t *parents, const json_t *json, char *key,
- size_t key_size) {
- snprintf(key, key_size, "%p", json);
- if (hashtable_get(parents, key))
+int jsonp_loop_check(hashtable_t *parents, const json_t *json, char *key, size_t key_size,
+ size_t *key_len_out) {
+ size_t key_len = snprintf(key, key_size, "%p", json);
+
+ if (key_len_out)
+ *key_len_out = key_len;
+
+ if (hashtable_get(parents, key, key_len))
return -1;
- return hashtable_set(parents, key, json_null());
+ return hashtable_set(parents, key, key_len, json_null());
}
/*** object ***/
@@ -93,16 +97,32 @@ size_t json_object_size(const json_t *json) {
}
json_t *json_object_get(const json_t *json, const char *key) {
+ if (!key)
+ return NULL;
+
+ return json_object_getn(json, key, strlen(key));
+}
+
+json_t *json_object_getn(const json_t *json, const char *key, size_t key_len) {
json_object_t *object;
if (!key || !json_is_object(json))
return NULL;
object = json_to_object(json);
- return hashtable_get(&object->hashtable, key);
+ return hashtable_get(&object->hashtable, key, key_len);
}
int json_object_set_new_nocheck(json_t *json, const char *key, json_t *value) {
+ if (!key) {
+ json_decref(value);
+ return -1;
+ }
+ return json_object_setn_new_nocheck(json, key, strlen(key), value);
+}
+
+int json_object_setn_new_nocheck(json_t *json, const char *key, size_t key_len,
+ json_t *value) {
json_object_t *object;
if (!value)
@@ -114,7 +134,7 @@ 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, key, value)) {
+ if (hashtable_set(&object->hashtable, key, key_len, value)) {
json_decref(value);
return -1;
}
@@ -123,22 +143,38 @@ int json_object_set_new_nocheck(json_t *json, const char *key, json_t *value) {
}
int json_object_set_new(json_t *json, const char *key, json_t *value) {
- if (!key || !utf8_check_string(key, strlen(key))) {
+ if (!key) {
json_decref(value);
return -1;
}
- return json_object_set_new_nocheck(json, key, value);
+ return json_object_setn_new(json, key, strlen(key), value);
+}
+
+int json_object_setn_new(json_t *json, const char *key, size_t key_len, json_t *value) {
+ if (!key || !utf8_check_string(key, key_len)) {
+ json_decref(value);
+ return -1;
+ }
+
+ return json_object_setn_new_nocheck(json, key, key_len, value);
}
int json_object_del(json_t *json, const char *key) {
+ if (!key)
+ return -1;
+
+ return json_object_deln(json, key, strlen(key));
+}
+
+int json_object_deln(json_t *json, const char *key, size_t key_len) {
json_object_t *object;
if (!key || !json_is_object(json))
return -1;
object = json_to_object(json);
- return hashtable_del(&object->hashtable, key);
+ return hashtable_del(&object->hashtable, key, key_len);
}
int json_object_clear(json_t *json) {
@@ -170,14 +206,15 @@ int json_object_update(json_t *object, json_t *other) {
int json_object_update_existing(json_t *object, json_t *other) {
const char *key;
+ size_t key_len;
json_t *value;
if (!json_is_object(object) || !json_is_object(other))
return -1;
- json_object_foreach(other, key, value) {
- if (json_object_get(object, key))
- json_object_set_nocheck(object, key, value);
+ json_object_keylen_foreach(other, key, key_len, value) {
+ if (json_object_getn(object, key, key_len))
+ json_object_setn_nocheck(object, key, key_len, value);
}
return 0;
@@ -200,17 +237,19 @@ int json_object_update_missing(json_t *object, json_t *other) {
int do_object_update_recursive(json_t *object, json_t *other, hashtable_t *parents) {
const char *key;
+ size_t key_len;
json_t *value;
char loop_key[LOOP_KEY_LEN];
int res = 0;
+ size_t loop_key_len;
if (!json_is_object(object) || !json_is_object(other))
return -1;
- if (jsonp_loop_check(parents, other, loop_key, sizeof(loop_key)))
+ if (jsonp_loop_check(parents, other, loop_key, sizeof(loop_key), &loop_key_len))
return -1;
- json_object_foreach(other, key, value) {
+ json_object_keylen_foreach(other, key, key_len, value) {
json_t *v = json_object_get(object, key);
if (json_is_object(v) && json_is_object(value)) {
@@ -219,14 +258,14 @@ int do_object_update_recursive(json_t *object, json_t *other, hashtable_t *paren
break;
}
} else {
- if (json_object_set_nocheck(object, key, value)) {
+ if (json_object_setn_nocheck(object, key, key_len, value)) {
res = -1;
break;
}
}
}
- hashtable_del(parents, loop_key);
+ hashtable_del(parents, loop_key, loop_key_len);
return res;
}
@@ -260,7 +299,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, key, strlen(key));
}
void *json_object_iter_next(json_t *json, void *iter) {
@@ -280,6 +319,13 @@ const char *json_object_iter_key(void *iter) {
return hashtable_iter_key(iter);
}
+size_t json_object_iter_key_len(void *iter) {
+ if (!iter)
+ return 0;
+
+ return hashtable_iter_key_len(iter);
+}
+
json_t *json_object_iter_value(void *iter) {
if (!iter)
return NULL;
@@ -340,8 +386,9 @@ static json_t *json_object_deep_copy(const json_t *object, hashtable_t *parents)
json_t *result;
void *iter;
char loop_key[LOOP_KEY_LEN];
+ size_t loop_key_len;
- if (jsonp_loop_check(parents, object, loop_key, sizeof(loop_key)))
+ if (jsonp_loop_check(parents, object, loop_key, sizeof(loop_key), &loop_key_len))
return NULL;
result = json_object();
@@ -366,7 +413,7 @@ static json_t *json_object_deep_copy(const json_t *object, hashtable_t *parents)
}
out:
- hashtable_del(parents, loop_key);
+ hashtable_del(parents, loop_key, loop_key_len);
return result;
}
@@ -633,8 +680,9 @@ static json_t *json_array_deep_copy(const json_t *array, hashtable_t *parents) {
json_t *result;
size_t i;
char loop_key[LOOP_KEY_LEN];
+ size_t loop_key_len;
- if (jsonp_loop_check(parents, array, loop_key, sizeof(loop_key)))
+ if (jsonp_loop_check(parents, array, loop_key, sizeof(loop_key), &loop_key_len))
return NULL;
result = json_array();
@@ -651,7 +699,7 @@ static json_t *json_array_deep_copy(const json_t *array, hashtable_t *parents) {
}
out:
- hashtable_del(parents, loop_key);
+ hashtable_del(parents, loop_key, loop_key_len);
return result;
}
diff --git a/test/suites/api/Makefile.am b/test/suites/api/Makefile.am
index e109e56..2bc638b 100644
--- a/test/suites/api/Makefile.am
+++ b/test/suites/api/Makefile.am
@@ -7,9 +7,10 @@ check_PROGRAMS = \
test_dump \
test_dump_callback \
test_equal \
+ test_fixed_size \
test_load \
- test_loadb \
test_load_callback \
+ test_loadb \
test_memory_funcs \
test_number \
test_object \
@@ -24,6 +25,7 @@ test_chaos_SOURCES = test_chaos.c util.h
test_copy_SOURCES = test_copy.c util.h
test_dump_SOURCES = test_dump.c util.h
test_dump_callback_SOURCES = test_dump_callback.c util.h
+test_fixed_size_SOURCES = test_fixed_size.c util.h
test_load_SOURCES = test_load.c util.h
test_loadb_SOURCES = test_loadb.c util.h
test_memory_funcs_SOURCES = test_memory_funcs.c util.h
diff --git a/test/suites/api/test_fixed_size.c b/test/suites/api/test_fixed_size.c
new file mode 100644
index 0000000..e4495ee
--- /dev/null
+++ b/test/suites/api/test_fixed_size.c
@@ -0,0 +1,205 @@
+/*
+ * Copyright (c) 2020 Petri Lehtinen <petri@digip.org>
+ *
+ * Jansson is free software; you can redistribute it and/or modify
+ * it under the terms of the MIT license. See LICENSE for details.
+ */
+
+#include "util.h"
+#include <jansson.h>
+#include <string.h>
+
+static void test_keylen_iterator(json_t *object) {
+ const char key1[] = {'t', 'e', 's', 't', '1'};
+ const char key2[] = {'t', 'e', 's', 't'};
+ const char key3[] = {'t', 'e', 's', '\0', 't'};
+ const char key4[] = {'t', 'e', 's', 't', '\0'};
+ const char *reference_keys[] = {key1, key2, key3, key4};
+ const size_t reference_keys_len[] = {sizeof(key1), sizeof(key2), sizeof(key3),
+ sizeof(key4)};
+ size_t index = 0;
+ json_t *value;
+ const char *key;
+ size_t keylen;
+
+ json_object_keylen_foreach(object, key, keylen, value) {
+ if (keylen != reference_keys_len[index])
+ fail("invalid key len in iterator");
+ if (memcmp(key, reference_keys[index], reference_keys_len[index]) != 0)
+ fail("invalid key in iterator");
+
+ index++;
+ }
+}
+
+static void test_keylen(void) {
+ json_t *obj = json_object();
+ const char key[] = {'t', 'e', 's', 't', '1'};
+ const char key2[] = {'t', 'e', 's', 't'};
+ const char key3[] = {'t', 'e', 's', '\0', 't'};
+ const char key4[] = {'t', 'e', 's', 't', '\0'};
+
+ if (json_object_size(obj) != 0)
+ fail("incorrect json");
+
+ json_object_set_new_nocheck(obj, "test1", json_true());
+
+ if (json_object_size(obj) != 1)
+ fail("incorrect json");
+
+ if (json_object_getn(obj, key, sizeof(key)) != json_true())
+ fail("json_object_getn failed");
+
+ if (json_object_getn(obj, key2, sizeof(key2)) != NULL)
+ fail("false positive json_object_getn by key2");
+
+ if (json_object_setn_nocheck(obj, key2, sizeof(key2), json_false()))
+ fail("json_object_setn_nocheck for key2 failed");
+
+ if (json_object_size(obj) != 2)
+ fail("incorrect json");
+
+ if (json_object_get(obj, "test") != json_false())
+ fail("json_object_setn_nocheck for key2 failed");
+
+ if (json_object_getn(obj, key2, sizeof(key2)) != json_false())
+ fail("json_object_getn by key 2 failed");
+
+ if (json_object_getn(obj, key3, sizeof(key3)) != NULL)
+ fail("false positive json_object_getn by key3");
+
+ if (json_object_setn_nocheck(obj, key3, sizeof(key3), json_false()))
+ fail("json_object_setn_nocheck for key3 failed");
+
+ if (json_object_size(obj) != 3)
+ fail("incorrect json");
+
+ if (json_object_getn(obj, key3, sizeof(key3)) != json_false())
+ fail("json_object_getn by key 3 failed");
+
+ if (json_object_getn(obj, key4, sizeof(key4)) != NULL)
+ fail("false positive json_object_getn by key3");
+
+ if (json_object_setn_nocheck(obj, key4, sizeof(key4), json_false()))
+ fail("json_object_setn_nocheck for key3 failed");
+
+ if (json_object_size(obj) != 4)
+ fail("incorrect json");
+
+ test_keylen_iterator(obj);
+
+ if (json_object_getn(obj, key4, sizeof(key4)) != json_false())
+ fail("json_object_getn by key 3 failed");
+
+ if (json_object_size(obj) != 4)
+ fail("incorrect json");
+
+ if (json_object_deln(obj, key4, sizeof(key4)))
+ fail("json_object_deln failed");
+ if (json_object_getn(obj, key4, sizeof(key4)) != NULL)
+ fail("json_object_deln failed");
+ if (json_object_size(obj) != 3)
+ fail("incorrect json");
+
+ if (json_object_deln(obj, key3, sizeof(key3)))
+ fail("json_object_deln failed");
+ if (json_object_getn(obj, key3, sizeof(key3)) != NULL)
+ fail("json_object_deln failed");
+ if (json_object_size(obj) != 2)
+ fail("incorrect json");
+
+ if (json_object_deln(obj, key2, sizeof(key2)))
+ fail("json_object_deln failed");
+ if (json_object_getn(obj, key2, sizeof(key2)) != NULL)
+ fail("json_object_deln failed");
+ if (json_object_size(obj) != 1)
+ fail("incorrect json");
+
+ if (json_object_deln(obj, key, sizeof(key)))
+ fail("json_object_deln failed");
+ if (json_object_getn(obj, key, sizeof(key)) != NULL)
+ fail("json_object_deln failed");
+ if (json_object_size(obj) != 0)
+ fail("incorrect json");
+
+ json_decref(obj);
+}
+
+static void test_invalid_keylen(void) {
+ json_t *obj = json_object();
+ const char key[] = {'t', 'e', 's', 't', '1'};
+
+ json_object_set_new_nocheck(obj, "test1", json_true());
+
+ if (json_object_getn(NULL, key, sizeof(key)) != NULL)
+ fail("json_object_getn on NULL failed");
+
+ if (json_object_getn(obj, NULL, sizeof(key)) != NULL)
+ fail("json_object_getn on NULL failed");
+
+ if (json_object_getn(obj, key, 0) != NULL)
+ fail("json_object_getn on NULL failed");
+
+ if (!json_object_setn_new(obj, NULL, sizeof(key), json_true()))
+ fail("json_object_setn_new with NULL key failed");
+
+ if (!json_object_setn_new_nocheck(obj, NULL, sizeof(key), json_true()))
+ fail("json_object_setn_new_nocheck with NULL key failed");
+
+ if (!json_object_del(obj, NULL))
+ fail("json_object_del with NULL failed");
+
+ json_decref(obj);
+}
+
+static void test_binary_keys(void) {
+ json_t *obj = json_object();
+ int key1 = 0;
+ int key2 = 1;
+
+ json_object_setn_nocheck(obj, (const char *)&key1, sizeof(key1), json_true());
+ json_object_setn_nocheck(obj, (const char *)&key2, sizeof(key2), json_true());
+
+ if (!json_is_true(json_object_getn(obj, (const char *)&key1, sizeof(key1))))
+ fail("cannot get integer key1");
+
+ if (!json_is_true(json_object_getn(obj, (const char *)&key1, sizeof(key2))))
+ fail("cannot get integer key2");
+
+ json_decref(obj);
+}
+
+static void test_dump_order(void) {
+ json_t *obj = json_object();
+ char key1[] = {'k', '\0', '-', '2'};
+ char key2[] = {'k', '\0', '-', '1'};
+ const char expected_sorted_str[] =
+ "{\"k\\u0000-1\": \"first\", \"k\\u0000-2\": \"second\"}";
+ const char expected_nonsorted_str[] =
+ "{\"k\\u0000-2\": \"second\", \"k\\u0000-1\": \"first\"}";
+ char *out;
+
+ json_object_setn_new_nocheck(obj, key1, sizeof(key1), json_string("second"));
+ json_object_setn_new_nocheck(obj, key2, sizeof(key2), json_string("first"));
+
+ out = malloc(512);
+
+ json_dumpb(obj, out, 512, 0);
+
+ if (memcmp(expected_nonsorted_str, out, sizeof(expected_nonsorted_str) - 1) != 0)
+ fail("preserve order failed");
+
+ json_dumpb(obj, out, 512, JSON_SORT_KEYS);
+ if (memcmp(expected_sorted_str, out, sizeof(expected_sorted_str) - 1) != 0)
+ fail("utf-8 sort failed");
+
+ free(out);
+ json_decref(obj);
+}
+
+static void run_tests() {
+ test_keylen();
+ test_invalid_keylen();
+ test_binary_keys();
+ test_dump_order();
+}