#include "JSONNode.h" #ifdef JSON_ITERATORS #ifdef JSON_REF_COUNT #define JSON_ASSERT_UNIQUE(x) JSON_ASSERT(internal -> refcount == 1, json_string(JSON_TEXT(x)) + JSON_TEXT(" in non single reference")) #else #define JSON_ASSERT_UNIQUE(x) (void)0 #endif #ifdef JSON_MUTEX_CALLBACKS #define JSON_MUTEX_COPY2 ,internal -> mylock #else #define JSON_MUTEX_COPY2 #endif JSONNode::json_iterator JSONNode::find(const json_string & name_t) json_nothrow { JSON_CHECK_INTERNAL(); JSON_ASSERT(type() == JSON_NODE, json_global(ERROR_NON_ITERATABLE) + JSON_TEXT("find")); makeUniqueInternal(); if (JSONNode ** res = internal -> at(name_t)){ return ptr_to_json_iterator(res); } return end(); } #ifdef JSON_CASE_INSENSITIVE_FUNCTIONS JSONNode::json_iterator JSONNode::find_nocase(const json_string & name_t) json_nothrow { JSON_CHECK_INTERNAL(); JSON_ASSERT(type() == JSON_NODE, json_global(ERROR_NON_ITERATABLE) + JSON_TEXT("find_nocase")); makeUniqueInternal(); if (JSONNode ** res = internal -> at_nocase(name_t)){ return ptr_to_json_iterator(res); } return end(); } #endif JSONNode::json_iterator JSONNode::erase(json_iterator pos) json_nothrow { JSON_CHECK_INTERNAL(); JSON_ASSERT(type() == JSON_NODE || type() == JSON_ARRAY, json_global(ERROR_NON_ITERATABLE) + JSON_TEXT("erase")); JSON_ASSERT_UNIQUE("erase 1"); JSON_ASSERT_SAFE(pos < end(), JSON_TEXT("erase out of range"), return end();); JSON_ASSERT_SAFE(pos >= begin(), JSON_TEXT("erase out of range"), return begin();); deleteJSONNode(*(json_iterator_ptr(pos))); internal -> CHILDREN -> erase(json_iterator_ptr(pos)); return (empty()) ? end() : pos; } JSONNode::json_iterator JSONNode::erase(json_iterator _start, const json_iterator & _end) json_nothrow { if (_start == _end) return _start; JSON_CHECK_INTERNAL(); JSON_ASSERT(type() == JSON_NODE || type() == JSON_ARRAY, json_global(ERROR_NON_ITERATABLE) + JSON_TEXT("erase")); JSON_ASSERT_UNIQUE("erase 3"); JSON_ASSERT_SAFE(_start <= end(), JSON_TEXT("erase out of lo range"), return end();); JSON_ASSERT_SAFE(_end <= end(), JSON_TEXT("erase out of hi range"), return end();); JSON_ASSERT_SAFE(_start >= begin(), JSON_TEXT("erase out of lo range"), return begin();); JSON_ASSERT_SAFE(_end >= begin(), JSON_TEXT("erase out of hi range"), return begin();); for (JSONNode ** pos = json_iterator_ptr(_start); pos < json_iterator_ptr(_end); ++pos){ deleteJSONNode(*pos); } internal -> CHILDREN -> erase(json_iterator_ptr(_start), (json_index_t)(json_iterator_ptr(_end) - json_iterator_ptr(_start))); return (empty()) ? end() : _start; } #ifdef JSON_LIBRARY JSONNode::json_iterator JSONNode::insert(json_iterator pos, JSONNode * x) json_nothrow { #else JSONNode::json_iterator JSONNode::insert(json_iterator pos, const JSONNode & x) json_nothrow { #endif JSON_CHECK_INTERNAL(); JSON_ASSERT(type() == JSON_NODE || type() == JSON_ARRAY, json_global(ERROR_NON_ITERATABLE) + JSON_TEXT("insert")); JSON_ASSERT_UNIQUE("insert 1"); if (json_iterator_ptr(pos) >= internal -> CHILDREN -> end()){ internal -> push_back(x); return end() - 1; } JSON_ASSERT_SAFE(pos >= begin(), JSON_TEXT("insert out of lo range"), return begin();); #ifdef JSON_LIBRARY internal -> CHILDREN -> insert(json_iterator_ptr(pos), x); #else internal -> CHILDREN -> insert(json_iterator_ptr(pos), newJSONNode(x)); #endif return pos; } JSONNode::json_iterator JSONNode::insertFFF(json_iterator pos, JSONNode ** const _start, JSONNode ** const _end) json_nothrow { JSON_CHECK_INTERNAL(); JSON_ASSERT(type() == JSON_NODE || type() == JSON_ARRAY, json_global(ERROR_NON_ITERATABLE) + JSON_TEXT("insertFFF")); JSON_ASSERT_UNIQUE("insertFFF"); JSON_ASSERT_SAFE(pos <= end(), JSON_TEXT("insert out of high range"), return end();); JSON_ASSERT_SAFE(pos >= begin(), JSON_TEXT("insert out of low range"), return begin();); const json_index_t num = (json_index_t)(_end - _start); json_auto mem(num); JSONNode ** runner = mem.ptr; for (JSONNode ** po = _start; po < _end; ++po){ *runner++ = newJSONNode(*(*po) JSON_MUTEX_COPY2); } internal -> CHILDREN -> insert(json_iterator_ptr(pos), mem.ptr, num); return pos; } #ifndef JSON_LIBRARY JSONNode::const_iterator JSONNode::find(const json_string & name_t) const json_nothrow { JSON_CHECK_INTERNAL(); JSON_ASSERT(type() == JSON_NODE, json_global(ERROR_NON_ITERATABLE) + JSON_TEXT("find")); if (JSONNode ** res = internal -> at(name_t)){ return JSONNode::const_iterator(res); } return JSONNode::const_iterator(internal -> end()); } #ifdef JSON_CASE_INSENSITIVE_FUNCTIONS JSONNode::const_iterator JSONNode::find_nocase(const json_string & name_t) const json_nothrow { JSON_CHECK_INTERNAL(); JSON_ASSERT(type() == JSON_NODE, json_global(ERROR_NON_ITERATABLE) + JSON_TEXT("find_nocase")); if (JSONNode ** res = internal -> at_nocase(name_t)){ return JSONNode::const_iterator(res); } return JSONNode::const_iterator(internal -> end()); } #endif JSONNode::reverse_iterator JSONNode::erase(reverse_iterator pos) json_nothrow { JSON_CHECK_INTERNAL(); JSON_ASSERT(type() == JSON_NODE || type() == JSON_ARRAY, json_global(ERROR_NON_ITERATABLE) + JSON_TEXT("erase")); JSON_ASSERT_UNIQUE("erase 2"); JSON_ASSERT_SAFE(pos < rend(), JSON_TEXT("erase out of range"), return rend();); JSON_ASSERT_SAFE(pos >= rbegin(), JSON_TEXT("erase out of range"), return rbegin();); deleteJSONNode(*(pos.it)); internal -> CHILDREN -> erase(pos.it); return (empty()) ? rend() : pos + 1; } JSONNode::reverse_iterator JSONNode::erase(reverse_iterator _start, const reverse_iterator & _end) json_nothrow { if (_start == _end) return _start; JSON_CHECK_INTERNAL(); JSON_ASSERT(type() == JSON_NODE || type() == JSON_ARRAY, json_global(ERROR_NON_ITERATABLE) + JSON_TEXT("erase")); JSON_ASSERT_UNIQUE("erase 4"); JSON_ASSERT_SAFE(_start <= rend(), JSON_TEXT("erase out of lo range"), return rend();); JSON_ASSERT_SAFE(_end <= rend(), JSON_TEXT("erase out of hi range"), return rend();); JSON_ASSERT_SAFE(_start >= rbegin(), JSON_TEXT("erase out of lo range"), return rbegin();); JSON_ASSERT_SAFE(_end >= rbegin(), JSON_TEXT("erase out of hi range"), return rbegin();); for (JSONNode ** pos = _start.it; pos > _end.it; --pos){ deleteJSONNode(*pos); } const json_index_t num = (json_index_t)(_start.it - _end.it); internal -> CHILDREN -> erase(_end.it + 1, num, _start.it); return (empty()) ? rend() : _start + num; } JSONNode::reverse_iterator JSONNode::insert(reverse_iterator pos, const JSONNode & x) json_nothrow { JSON_CHECK_INTERNAL(); JSON_ASSERT(type() == JSON_NODE || type() == JSON_ARRAY, json_global(ERROR_NON_ITERATABLE) + JSON_TEXT("insert")); JSON_ASSERT_UNIQUE("insert 1"); if (pos.it < internal -> CHILDREN -> begin()){ internal -> push_front(x); return rend() - 1; } JSON_ASSERT_SAFE(pos >= rbegin(), JSON_TEXT("insert out of range"), return rbegin();); internal -> CHILDREN -> insert(++pos.it, newJSONNode(x), true); return pos; } JSONNode::reverse_iterator JSONNode::insertRFF(reverse_iterator pos, JSONNode ** const _start, JSONNode ** const _end) json_nothrow { JSON_CHECK_INTERNAL(); JSON_ASSERT(type() == JSON_NODE || type() == JSON_ARRAY, json_global(ERROR_NON_ITERATABLE) + JSON_TEXT("insertRFF")); JSON_ASSERT_UNIQUE("insert RFF"); JSON_ASSERT_SAFE(pos <= rend(), JSON_TEXT("insert out of range"), return rend();); JSON_ASSERT_SAFE(pos >= rbegin(), JSON_TEXT("insert out of range"), return rbegin();); const json_index_t num = (json_index_t)(_end - _start); json_auto mem(num); JSONNode ** runner = mem.ptr + num; for (JSONNode ** po = _start; po < _end; ++po){ //fill it backwards *(--runner) = newJSONNode(*(*po) JSON_MUTEX_COPY2); } internal -> CHILDREN -> insert(++pos.it, mem.ptr, num); return pos - num + 1; } JSONNode::iterator JSONNode::insertFRR(json_iterator pos, JSONNode ** const _start, JSONNode ** const _end) json_nothrow { JSON_CHECK_INTERNAL(); JSON_ASSERT(type() == JSON_NODE || type() == JSON_ARRAY, json_global(ERROR_NON_ITERATABLE) + JSON_TEXT("insertFRR")); JSON_ASSERT_UNIQUE("insert FRR"); JSON_ASSERT_SAFE(pos <= end(), JSON_TEXT("insert out of range"), return end();); JSON_ASSERT_SAFE(pos >= begin(), JSON_TEXT("insert out of range"), return begin();); const json_index_t num = (json_index_t)(_start - _end); json_auto mem(num); JSONNode ** runner = mem.ptr; for (JSONNode ** po = _start; po > _end; --po){ *runner++ = newJSONNode(*(*po) JSON_MUTEX_COPY2); } internal -> CHILDREN -> insert(pos.it, mem.ptr, num); return pos; } JSONNode::reverse_iterator JSONNode::insertRRR(reverse_iterator pos, JSONNode ** const _start, JSONNode ** const _end) json_nothrow { JSON_CHECK_INTERNAL(); JSON_ASSERT(type() == JSON_NODE || type() == JSON_ARRAY, json_global(ERROR_NON_ITERATABLE) + JSON_TEXT("insertRRR")); JSON_ASSERT_UNIQUE("insert RRR"); JSON_ASSERT_SAFE(pos <= rend(), JSON_TEXT("insert out of range"), return rend();); JSON_ASSERT_SAFE(pos >= rbegin(), JSON_TEXT("insert out of range"), return rbegin();); const json_index_t num = (json_index_t)(_start - _end); json_auto mem(num); JSONNode ** runner = mem.ptr; for (JSONNode ** po = _start; po > _end; --po){ *runner++ = newJSONNode(*(*po) JSON_MUTEX_COPY2); } internal -> CHILDREN -> insert(++pos.it, mem.ptr, num); return pos - num + 1; } #endif #endif