| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983 |
- /*
- * Copyright (c) 2009-2011 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.
- */
- #define _GNU_SOURCE
- #include <stddef.h>
- #include <stdlib.h>
- #include <string.h>
- #include <jansson.h>
- #include "hashtable.h"
- #include "jansson_private.h"
- #include "utf.h"
- static JSON_INLINE void json_init(json_t *json, json_type type)
- {
- json->type = type;
- json->refcount = 1;
- }
- /*** object ***/
- /* From http://www.cse.yorku.ca/~oz/hash.html */
- size_t jsonp_hash_str(const void *ptr)
- {
- const char *str = (const char *)ptr;
- size_t hash = 5381;
- size_t c;
- while((c = (size_t)*str))
- {
- hash = ((hash << 5) + hash) + c;
- str++;
- }
- return hash;
- }
- int jsonp_str_equal(const void *ptr1, const void *ptr2)
- {
- return strcmp((const char *)ptr1, (const char *)ptr2) == 0;
- }
- /* 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 size_t hash_key(const void *ptr)
- {
- return jsonp_hash_str(((const object_key_t *)ptr)->key);
- }
- static int key_equal(const void *ptr1, const void *ptr2)
- {
- return jsonp_str_equal(((const object_key_t *)ptr1)->key,
- ((const object_key_t *)ptr2)->key);
- }
- static void value_decref(void *value)
- {
- json_decref((json_t *)value);
- }
- json_t *json_object(void)
- {
- json_object_t *object = jsonp_malloc(sizeof(json_object_t));
- if(!object)
- return NULL;
- json_init(&object->json, JSON_OBJECT);
- if(hashtable_init(&object->hashtable,
- hash_key, key_equal,
- jsonp_free, value_decref))
- {
- jsonp_free(object);
- return NULL;
- }
- object->serial = 0;
- object->visited = 0;
- return &object->json;
- }
- static void json_delete_object(json_object_t *object)
- {
- hashtable_close(&object->hashtable);
- jsonp_free(object);
- }
- size_t json_object_size(const json_t *json)
- {
- json_object_t *object;
- if(!json_is_object(json))
- return 0;
- object = json_to_object(json);
- return object->hashtable.size;
- }
- json_t *json_object_get(const json_t *json, const char *key)
- {
- json_object_t *object;
- if(!json_is_object(json))
- return NULL;
- object = json_to_object(json);
- 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;
- if(!json_is_object(json) || json == value)
- {
- json_decref(value);
- return -1;
- }
- object = json_to_object(json);
- /* offsetof(...) returns the size of object_key_t without the
- last, flexible member. This way, the correct amount is
- allocated. */
- k = jsonp_malloc(offsetof(object_key_t, key) + strlen(key) + 1);
- if(!k)
- {
- json_decref(value);
- return -1;
- }
- k->serial = object->serial++;
- strcpy(k->key, key);
- if(hashtable_set(&object->hashtable, k, value))
- {
- json_decref(value);
- return -1;
- }
- return 0;
- }
- int json_object_set_new(json_t *json, const char *key, json_t *value)
- {
- if(!key || !utf8_check_string(key, -1))
- {
- json_decref(value);
- return -1;
- }
- return json_object_set_new_nocheck(json, key, value);
- }
- int json_object_del(json_t *json, const char *key)
- {
- json_object_t *object;
- if(!json_is_object(json))
- return -1;
- object = json_to_object(json);
- return hashtable_del(&object->hashtable, string_to_key(key));
- }
- int json_object_clear(json_t *json)
- {
- json_object_t *object;
- if(!json_is_object(json))
- return -1;
- object = json_to_object(json);
- hashtable_clear(&object->hashtable);
- return 0;
- }
- int json_object_update(json_t *object, json_t *other)
- {
- void *iter;
- if(!json_is_object(object) || !json_is_object(other))
- return -1;
- iter = json_object_iter(other);
- while(iter) {
- const char *key;
- json_t *value;
- key = json_object_iter_key(iter);
- value = json_object_iter_value(iter);
- if(json_object_set_nocheck(object, key, value))
- return -1;
- iter = json_object_iter_next(other, iter);
- }
- return 0;
- }
- void *json_object_iter(json_t *json)
- {
- json_object_t *object;
- if(!json_is_object(json))
- return NULL;
- object = json_to_object(json);
- return hashtable_iter(&object->hashtable);
- }
- void *json_object_iter_at(json_t *json, const char *key)
- {
- json_object_t *object;
- if(!key || !json_is_object(json))
- return NULL;
- object = json_to_object(json);
- return hashtable_iter_at(&object->hashtable, string_to_key(key));
- }
- void *json_object_iter_next(json_t *json, void *iter)
- {
- json_object_t *object;
- if(!json_is_object(json) || iter == NULL)
- return NULL;
- object = json_to_object(json);
- 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 jsonp_object_iter_fullkey(iter)->key;
- }
- json_t *json_object_iter_value(void *iter)
- {
- if(!iter)
- return NULL;
- return (json_t *)hashtable_iter_value(iter);
- }
- int json_object_iter_set_new(json_t *json, void *iter, json_t *value)
- {
- json_object_t *object;
- if(!json_is_object(json) || !iter || !value)
- return -1;
- object = json_to_object(json);
- hashtable_iter_set(&object->hashtable, iter, value);
- return 0;
- }
- static int json_object_equal(json_t *object1, json_t *object2)
- {
- void *iter;
- if(json_object_size(object1) != json_object_size(object2))
- return 0;
- iter = json_object_iter(object1);
- while(iter)
- {
- const char *key;
- json_t *value1, *value2;
- key = json_object_iter_key(iter);
- value1 = json_object_iter_value(iter);
- value2 = json_object_get(object2, key);
- if(!json_equal(value1, value2))
- return 0;
- iter = json_object_iter_next(object1, iter);
- }
- return 1;
- }
- static json_t *json_object_copy(json_t *object)
- {
- json_t *result;
- void *iter;
- result = json_object();
- if(!result)
- return NULL;
- iter = json_object_iter(object);
- while(iter)
- {
- const char *key;
- json_t *value;
- key = json_object_iter_key(iter);
- value = json_object_iter_value(iter);
- json_object_set_nocheck(result, key, value);
- iter = json_object_iter_next(object, iter);
- }
- return result;
- }
- static json_t *json_object_deep_copy(json_t *object)
- {
- json_t *result;
- void *iter;
- result = json_object();
- if(!result)
- return NULL;
- iter = json_object_iter(object);
- while(iter)
- {
- const char *key;
- json_t *value;
- key = json_object_iter_key(iter);
- value = json_object_iter_value(iter);
- json_object_set_new_nocheck(result, key, json_deep_copy(value));
- iter = json_object_iter_next(object, iter);
- }
- return result;
- }
- /*** array ***/
- json_t *json_array(void)
- {
- json_array_t *array = jsonp_malloc(sizeof(json_array_t));
- if(!array)
- return NULL;
- json_init(&array->json, JSON_ARRAY);
- array->entries = 0;
- array->size = 8;
- array->table = jsonp_malloc(array->size * sizeof(json_t *));
- if(!array->table) {
- jsonp_free(array);
- return NULL;
- }
- array->visited = 0;
- return &array->json;
- }
- static void json_delete_array(json_array_t *array)
- {
- size_t i;
- for(i = 0; i < array->entries; i++)
- json_decref(array->table[i]);
- jsonp_free(array->table);
- jsonp_free(array);
- }
- size_t json_array_size(const json_t *json)
- {
- if(!json_is_array(json))
- return 0;
- return json_to_array(json)->entries;
- }
- json_t *json_array_get(const json_t *json, size_t index)
- {
- json_array_t *array;
- if(!json_is_array(json))
- return NULL;
- array = json_to_array(json);
- if(index >= array->entries)
- return NULL;
- return array->table[index];
- }
- int json_array_set_new(json_t *json, size_t index, json_t *value)
- {
- json_array_t *array;
- if(!value)
- return -1;
- if(!json_is_array(json) || json == value)
- {
- json_decref(value);
- return -1;
- }
- array = json_to_array(json);
- if(index >= array->entries)
- {
- json_decref(value);
- return -1;
- }
- json_decref(array->table[index]);
- array->table[index] = value;
- return 0;
- }
- static void array_move(json_array_t *array, size_t dest,
- size_t src, size_t count)
- {
- memmove(&array->table[dest], &array->table[src], count * sizeof(json_t *));
- }
- static void array_copy(json_t **dest, size_t dpos,
- json_t **src, size_t spos,
- size_t count)
- {
- memcpy(&dest[dpos], &src[spos], count * sizeof(json_t *));
- }
- static json_t **json_array_grow(json_array_t *array,
- size_t amount,
- int copy)
- {
- size_t new_size;
- json_t **old_table, **new_table;
- if(array->entries + amount <= array->size)
- return array->table;
- old_table = array->table;
- new_size = max(array->size + amount, array->size * 2);
- new_table = jsonp_malloc(new_size * sizeof(json_t *));
- if(!new_table)
- return NULL;
- array->size = new_size;
- array->table = new_table;
- if(copy) {
- array_copy(array->table, 0, old_table, 0, array->entries);
- jsonp_free(old_table);
- return array->table;
- }
- return old_table;
- }
- int json_array_append_new(json_t *json, json_t *value)
- {
- json_array_t *array;
- if(!value)
- return -1;
- if(!json_is_array(json) || json == value)
- {
- json_decref(value);
- return -1;
- }
- array = json_to_array(json);
- if(!json_array_grow(array, 1, 1)) {
- json_decref(value);
- return -1;
- }
- array->table[array->entries] = value;
- array->entries++;
- return 0;
- }
- int json_array_insert_new(json_t *json, size_t index, json_t *value)
- {
- json_array_t *array;
- json_t **old_table;
- if(!value)
- return -1;
- if(!json_is_array(json) || json == value) {
- json_decref(value);
- return -1;
- }
- array = json_to_array(json);
- if(index > array->entries) {
- json_decref(value);
- return -1;
- }
- old_table = json_array_grow(array, 1, 0);
- if(!old_table) {
- json_decref(value);
- return -1;
- }
- if(old_table != array->table) {
- array_copy(array->table, 0, old_table, 0, index);
- array_copy(array->table, index + 1, old_table, index,
- array->entries - index);
- jsonp_free(old_table);
- }
- else
- array_move(array, index + 1, index, array->entries - index);
- array->table[index] = value;
- array->entries++;
- return 0;
- }
- int json_array_remove(json_t *json, size_t index)
- {
- json_array_t *array;
- if(!json_is_array(json))
- return -1;
- array = json_to_array(json);
- if(index >= array->entries)
- return -1;
- json_decref(array->table[index]);
- array_move(array, index, index + 1, array->entries - index);
- array->entries--;
- return 0;
- }
- int json_array_clear(json_t *json)
- {
- json_array_t *array;
- size_t i;
- if(!json_is_array(json))
- return -1;
- array = json_to_array(json);
- for(i = 0; i < array->entries; i++)
- json_decref(array->table[i]);
- array->entries = 0;
- return 0;
- }
- int json_array_extend(json_t *json, json_t *other_json)
- {
- json_array_t *array, *other;
- size_t i;
- if(!json_is_array(json) || !json_is_array(other_json))
- return -1;
- array = json_to_array(json);
- other = json_to_array(other_json);
- if(!json_array_grow(array, other->entries, 1))
- return -1;
- for(i = 0; i < other->entries; i++)
- json_incref(other->table[i]);
- array_copy(array->table, array->entries, other->table, 0, other->entries);
- array->entries += other->entries;
- return 0;
- }
- static int json_array_equal(json_t *array1, json_t *array2)
- {
- size_t i, size;
- size = json_array_size(array1);
- if(size != json_array_size(array2))
- return 0;
- for(i = 0; i < size; i++)
- {
- json_t *value1, *value2;
- value1 = json_array_get(array1, i);
- value2 = json_array_get(array2, i);
- if(!json_equal(value1, value2))
- return 0;
- }
- return 1;
- }
- static json_t *json_array_copy(json_t *array)
- {
- json_t *result;
- size_t i;
- result = json_array();
- if(!result)
- return NULL;
- for(i = 0; i < json_array_size(array); i++)
- json_array_append(result, json_array_get(array, i));
- return result;
- }
- static json_t *json_array_deep_copy(json_t *array)
- {
- json_t *result;
- size_t i;
- result = json_array();
- if(!result)
- return NULL;
- for(i = 0; i < json_array_size(array); i++)
- json_array_append_new(result, json_deep_copy(json_array_get(array, i)));
- return result;
- }
- /*** string ***/
- json_t *json_string_nocheck(const char *value)
- {
- json_string_t *string;
- if(!value)
- return NULL;
- string = jsonp_malloc(sizeof(json_string_t));
- if(!string)
- return NULL;
- json_init(&string->json, JSON_STRING);
- string->value = jsonp_strdup(value);
- if(!string->value) {
- jsonp_free(string);
- return NULL;
- }
- return &string->json;
- }
- json_t *json_string(const char *value)
- {
- if(!value || !utf8_check_string(value, -1))
- return NULL;
- return json_string_nocheck(value);
- }
- const char *json_string_value(const json_t *json)
- {
- if(!json_is_string(json))
- return NULL;
- return json_to_string(json)->value;
- }
- int json_string_set_nocheck(json_t *json, const char *value)
- {
- char *dup;
- json_string_t *string;
- dup = jsonp_strdup(value);
- if(!dup)
- return -1;
- string = json_to_string(json);
- jsonp_free(string->value);
- string->value = dup;
- return 0;
- }
- int json_string_set(json_t *json, const char *value)
- {
- if(!value || !utf8_check_string(value, -1))
- return -1;
- return json_string_set_nocheck(json, value);
- }
- static void json_delete_string(json_string_t *string)
- {
- jsonp_free(string->value);
- jsonp_free(string);
- }
- static int json_string_equal(json_t *string1, json_t *string2)
- {
- return strcmp(json_string_value(string1), json_string_value(string2)) == 0;
- }
- static json_t *json_string_copy(json_t *string)
- {
- return json_string_nocheck(json_string_value(string));
- }
- /*** integer ***/
- json_t *json_integer(json_int_t value)
- {
- json_integer_t *integer = jsonp_malloc(sizeof(json_integer_t));
- if(!integer)
- return NULL;
- json_init(&integer->json, JSON_INTEGER);
- integer->value = value;
- return &integer->json;
- }
- json_int_t json_integer_value(const json_t *json)
- {
- if(!json_is_integer(json))
- return 0;
- return json_to_integer(json)->value;
- }
- int json_integer_set(json_t *json, json_int_t value)
- {
- if(!json_is_integer(json))
- return -1;
- json_to_integer(json)->value = value;
- return 0;
- }
- static void json_delete_integer(json_integer_t *integer)
- {
- jsonp_free(integer);
- }
- static int json_integer_equal(json_t *integer1, json_t *integer2)
- {
- return json_integer_value(integer1) == json_integer_value(integer2);
- }
- static json_t *json_integer_copy(json_t *integer)
- {
- return json_integer(json_integer_value(integer));
- }
- /*** real ***/
- json_t *json_real(double value)
- {
- json_real_t *real = jsonp_malloc(sizeof(json_real_t));
- if(!real)
- return NULL;
- json_init(&real->json, JSON_REAL);
- real->value = value;
- return &real->json;
- }
- double json_real_value(const json_t *json)
- {
- if(!json_is_real(json))
- return 0;
- return json_to_real(json)->value;
- }
- int json_real_set(json_t *json, double value)
- {
- if(!json_is_real(json))
- return 0;
- json_to_real(json)->value = value;
- return 0;
- }
- static void json_delete_real(json_real_t *real)
- {
- jsonp_free(real);
- }
- static int json_real_equal(json_t *real1, json_t *real2)
- {
- return json_real_value(real1) == json_real_value(real2);
- }
- static json_t *json_real_copy(json_t *real)
- {
- return json_real(json_real_value(real));
- }
- /*** number ***/
- double json_number_value(const json_t *json)
- {
- if(json_is_integer(json))
- return json_integer_value(json);
- else if(json_is_real(json))
- return json_real_value(json);
- else
- return 0.0;
- }
- /*** simple values ***/
- json_t *json_true(void)
- {
- static json_t the_true = {JSON_TRUE, (size_t)-1};
- return &the_true;
- }
- json_t *json_false(void)
- {
- static json_t the_false = {JSON_FALSE, (size_t)-1};
- return &the_false;
- }
- json_t *json_null(void)
- {
- static json_t the_null = {JSON_NULL, (size_t)-1};
- return &the_null;
- }
- /*** deletion ***/
- void json_delete(json_t *json)
- {
- if(json_is_object(json))
- json_delete_object(json_to_object(json));
- else if(json_is_array(json))
- json_delete_array(json_to_array(json));
- else if(json_is_string(json))
- json_delete_string(json_to_string(json));
- else if(json_is_integer(json))
- json_delete_integer(json_to_integer(json));
- else if(json_is_real(json))
- json_delete_real(json_to_real(json));
- /* json_delete is not called for true, false or null */
- }
- /*** equality ***/
- int json_equal(json_t *json1, json_t *json2)
- {
- if(!json1 || !json2)
- return 0;
- if(json_typeof(json1) != json_typeof(json2))
- return 0;
- /* this covers true, false and null as they are singletons */
- if(json1 == json2)
- return 1;
- if(json_is_object(json1))
- return json_object_equal(json1, json2);
- if(json_is_array(json1))
- return json_array_equal(json1, json2);
- if(json_is_string(json1))
- return json_string_equal(json1, json2);
- if(json_is_integer(json1))
- return json_integer_equal(json1, json2);
- if(json_is_real(json1))
- return json_real_equal(json1, json2);
- return 0;
- }
- /*** copying ***/
- json_t *json_copy(json_t *json)
- {
- if(!json)
- return NULL;
- if(json_is_object(json))
- return json_object_copy(json);
- if(json_is_array(json))
- return json_array_copy(json);
- if(json_is_string(json))
- return json_string_copy(json);
- if(json_is_integer(json))
- return json_integer_copy(json);
- if(json_is_real(json))
- return json_real_copy(json);
- if(json_is_true(json) || json_is_false(json) || json_is_null(json))
- return json;
- return NULL;
- }
- json_t *json_deep_copy(json_t *json)
- {
- if(!json)
- return NULL;
- if(json_is_object(json))
- return json_object_deep_copy(json);
- if(json_is_array(json))
- return json_array_deep_copy(json);
- /* for the rest of the types, deep copying doesn't differ from
- shallow copying */
- if(json_is_string(json))
- return json_string_copy(json);
- if(json_is_integer(json))
- return json_integer_copy(json);
- if(json_is_real(json))
- return json_real_copy(json);
- if(json_is_true(json) || json_is_false(json) || json_is_null(json))
- return json;
- return NULL;
- }
|