|
@@ -0,0 +1,5448 @@
|
|
|
+/// Json-cpp amalgamated source (http://jsoncpp.sourceforge.net/).
|
|
|
+/// It is intended to be used with #include "json/json.h"
|
|
|
+
|
|
|
+// //////////////////////////////////////////////////////////////////////
|
|
|
+// Beginning of content of file: LICENSE
|
|
|
+// //////////////////////////////////////////////////////////////////////
|
|
|
+
|
|
|
+/*
|
|
|
+The JsonCpp library's source code, including accompanying documentation,
|
|
|
+tests and demonstration applications, are licensed under the following
|
|
|
+conditions...
|
|
|
+
|
|
|
+Baptiste Lepilleur and The JsonCpp Authors explicitly disclaim copyright in all
|
|
|
+jurisdictions which recognize such a disclaimer. In such jurisdictions,
|
|
|
+this software is released into the Public Domain.
|
|
|
+
|
|
|
+In jurisdictions which do not recognize Public Domain property (e.g. Germany as of
|
|
|
+2010), this software is Copyright (c) 2007-2010 by Baptiste Lepilleur and
|
|
|
+The JsonCpp Authors, and is released under the terms of the MIT License (see below).
|
|
|
+
|
|
|
+In jurisdictions which recognize Public Domain property, the user of this
|
|
|
+software may choose to accept it either as 1) Public Domain, 2) under the
|
|
|
+conditions of the MIT License (see below), or 3) under the terms of dual
|
|
|
+Public Domain/MIT License conditions described here, as they choose.
|
|
|
+
|
|
|
+The MIT License is about as close to Public Domain as a license can get, and is
|
|
|
+described in clear, concise terms at:
|
|
|
+
|
|
|
+ http://en.wikipedia.org/wiki/MIT_License
|
|
|
+
|
|
|
+The full text of the MIT License follows:
|
|
|
+
|
|
|
+========================================================================
|
|
|
+Copyright (c) 2007-2010 Baptiste Lepilleur and The JsonCpp Authors
|
|
|
+
|
|
|
+Permission is hereby granted, free of charge, to any person
|
|
|
+obtaining a copy of this software and associated documentation
|
|
|
+files (the "Software"), to deal in the Software without
|
|
|
+restriction, including without limitation the rights to use, copy,
|
|
|
+modify, merge, publish, distribute, sublicense, and/or sell copies
|
|
|
+of the Software, and to permit persons to whom the Software is
|
|
|
+furnished to do so, subject to the following conditions:
|
|
|
+
|
|
|
+The above copyright notice and this permission notice shall be
|
|
|
+included in all copies or substantial portions of the Software.
|
|
|
+
|
|
|
+THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
|
+EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
|
+MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
|
+NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
|
|
|
+BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
|
|
|
+ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
|
|
+CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
|
+SOFTWARE.
|
|
|
+========================================================================
|
|
|
+(END LICENSE TEXT)
|
|
|
+
|
|
|
+The MIT license is compatible with both the GPL and commercial
|
|
|
+software, affording one all of the rights of Public Domain with the
|
|
|
+minor nuisance of being required to keep the above copyright notice
|
|
|
+and license text in the source code. Note also that by accepting the
|
|
|
+Public Domain "license" you can re-license your copy using whatever
|
|
|
+license you like.
|
|
|
+
|
|
|
+*/
|
|
|
+
|
|
|
+// //////////////////////////////////////////////////////////////////////
|
|
|
+// End of content of file: LICENSE
|
|
|
+// //////////////////////////////////////////////////////////////////////
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+#include "json.h"
|
|
|
+
|
|
|
+#ifndef JSON_IS_AMALGAMATION
|
|
|
+#error "Compile with -I PATH_TO_JSON_DIRECTORY"
|
|
|
+#endif
|
|
|
+
|
|
|
+
|
|
|
+// //////////////////////////////////////////////////////////////////////
|
|
|
+// Beginning of content of file: src/lib_json/json_tool.h
|
|
|
+// //////////////////////////////////////////////////////////////////////
|
|
|
+
|
|
|
+// Copyright 2007-2010 Baptiste Lepilleur and The JsonCpp Authors
|
|
|
+// Distributed under MIT license, or public domain if desired and
|
|
|
+// recognized in your jurisdiction.
|
|
|
+// See file LICENSE for detail or copy at http://jsoncpp.sourceforge.net/LICENSE
|
|
|
+
|
|
|
+#ifndef LIB_JSONCPP_JSON_TOOL_H_INCLUDED
|
|
|
+#define LIB_JSONCPP_JSON_TOOL_H_INCLUDED
|
|
|
+
|
|
|
+#if !defined(JSON_IS_AMALGAMATION)
|
|
|
+#include <json/config.h>
|
|
|
+#endif
|
|
|
+
|
|
|
+// Also support old flag NO_LOCALE_SUPPORT
|
|
|
+#ifdef NO_LOCALE_SUPPORT
|
|
|
+#define JSONCPP_NO_LOCALE_SUPPORT
|
|
|
+#endif
|
|
|
+
|
|
|
+#ifndef JSONCPP_NO_LOCALE_SUPPORT
|
|
|
+
|
|
|
+#include <clocale>
|
|
|
+
|
|
|
+#endif
|
|
|
+
|
|
|
+/* This header provides common string manipulation support, such as UTF-8,
|
|
|
+ * portable conversion from/to string...
|
|
|
+ *
|
|
|
+ * It is an internal header that must not be exposed.
|
|
|
+ */
|
|
|
+
|
|
|
+namespace Json {
|
|
|
+ static inline char getDecimalPoint() {
|
|
|
+#ifdef JSONCPP_NO_LOCALE_SUPPORT
|
|
|
+ return '\0';
|
|
|
+#else
|
|
|
+ struct lconv *lc = localeconv();
|
|
|
+ return lc ? *(lc->decimal_point) : '\0';
|
|
|
+#endif
|
|
|
+ }
|
|
|
+
|
|
|
+/// Converts a unicode code-point to UTF-8.
|
|
|
+ static inline String codePointToUTF8(unsigned int cp) {
|
|
|
+ String result;
|
|
|
+
|
|
|
+ // based on description from http://en.wikipedia.org/wiki/UTF-8
|
|
|
+
|
|
|
+ if (cp <= 0x7f) {
|
|
|
+ result.resize(1);
|
|
|
+ result[0] = static_cast<char>(cp);
|
|
|
+ } else if (cp <= 0x7FF) {
|
|
|
+ result.resize(2);
|
|
|
+ result[1] = static_cast<char>(0x80 | (0x3f & cp));
|
|
|
+ result[0] = static_cast<char>(0xC0 | (0x1f & (cp >> 6)));
|
|
|
+ } else if (cp <= 0xFFFF) {
|
|
|
+ result.resize(3);
|
|
|
+ result[2] = static_cast<char>(0x80 | (0x3f & cp));
|
|
|
+ result[1] = static_cast<char>(0x80 | (0x3f & (cp >> 6)));
|
|
|
+ result[0] = static_cast<char>(0xE0 | (0xf & (cp >> 12)));
|
|
|
+ } else if (cp <= 0x10FFFF) {
|
|
|
+ result.resize(4);
|
|
|
+ result[3] = static_cast<char>(0x80 | (0x3f & cp));
|
|
|
+ result[2] = static_cast<char>(0x80 | (0x3f & (cp >> 6)));
|
|
|
+ result[1] = static_cast<char>(0x80 | (0x3f & (cp >> 12)));
|
|
|
+ result[0] = static_cast<char>(0xF0 | (0x7 & (cp >> 18)));
|
|
|
+ }
|
|
|
+
|
|
|
+ return result;
|
|
|
+ }
|
|
|
+
|
|
|
+ enum {
|
|
|
+ /// Constant that specify the size of the buffer that must be passed to
|
|
|
+ /// uintToString.
|
|
|
+ uintToStringBufferSize = 3 * sizeof(LargestUInt) + 1
|
|
|
+ };
|
|
|
+
|
|
|
+// Defines a char buffer for use with uintToString().
|
|
|
+ using UIntToStringBuffer = char[uintToStringBufferSize];
|
|
|
+
|
|
|
+/** Converts an unsigned integer to string.
|
|
|
+ * @param value Unsigned integer to convert to string
|
|
|
+ * @param current Input/Output string buffer.
|
|
|
+ * Must have at least uintToStringBufferSize chars free.
|
|
|
+ */
|
|
|
+ static inline void uintToString(LargestUInt value, char *¤t) {
|
|
|
+ *--current = 0;
|
|
|
+ do {
|
|
|
+ *--current = static_cast<char>(value % 10U + static_cast<unsigned>('0'));
|
|
|
+ value /= 10;
|
|
|
+ } while (value != 0);
|
|
|
+ }
|
|
|
+
|
|
|
+/** Change ',' to '.' everywhere in buffer.
|
|
|
+ *
|
|
|
+ * We had a sophisticated way, but it did not work in WinCE.
|
|
|
+ * @see https://github.com/open-source-parsers/jsoncpp/pull/9
|
|
|
+ */
|
|
|
+ template<typename Iter>
|
|
|
+ Iter fixNumericLocale(Iter begin, Iter end) {
|
|
|
+ for (; begin != end; ++begin) {
|
|
|
+ if (*begin == ',') {
|
|
|
+ *begin = '.';
|
|
|
+ }
|
|
|
+ }
|
|
|
+ return begin;
|
|
|
+ }
|
|
|
+
|
|
|
+ template<typename Iter>
|
|
|
+ void fixNumericLocaleInput(Iter begin, Iter end) {
|
|
|
+ char decimalPoint = getDecimalPoint();
|
|
|
+ if (decimalPoint == '\0' || decimalPoint == '.') {
|
|
|
+ return;
|
|
|
+ }
|
|
|
+ for (; begin != end; ++begin) {
|
|
|
+ if (*begin == '.') {
|
|
|
+ *begin = decimalPoint;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+/**
|
|
|
+ * Return iterator that would be the new end of the range [begin,end), if we
|
|
|
+ * were to delete zeros in the end of string, but not the last zero before '.'.
|
|
|
+ */
|
|
|
+ template<typename Iter>
|
|
|
+ Iter fixZerosInTheEnd(Iter begin, Iter end) {
|
|
|
+ for (; begin != end; --end) {
|
|
|
+ if (*(end - 1) != '0') {
|
|
|
+ return end;
|
|
|
+ }
|
|
|
+ // Don't delete the last zero before the decimal point.
|
|
|
+ if (begin != (end - 1) && *(end - 2) == '.') {
|
|
|
+ return end;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ return end;
|
|
|
+ }
|
|
|
+
|
|
|
+} // namespace Json
|
|
|
+
|
|
|
+#endif // LIB_JSONCPP_JSON_TOOL_H_INCLUDED
|
|
|
+
|
|
|
+// //////////////////////////////////////////////////////////////////////
|
|
|
+// End of content of file: src/lib_json/json_tool.h
|
|
|
+// //////////////////////////////////////////////////////////////////////
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+// //////////////////////////////////////////////////////////////////////
|
|
|
+// Beginning of content of file: src/lib_json/json_reader.cpp
|
|
|
+// //////////////////////////////////////////////////////////////////////
|
|
|
+
|
|
|
+// Copyright 2007-2011 Baptiste Lepilleur and The JsonCpp Authors
|
|
|
+// Copyright (C) 2016 InfoTeCS JSC. All rights reserved.
|
|
|
+// Distributed under MIT license, or public domain if desired and
|
|
|
+// recognized in your jurisdiction.
|
|
|
+// See file LICENSE for detail or copy at http://jsoncpp.sourceforge.net/LICENSE
|
|
|
+
|
|
|
+#if !defined(JSON_IS_AMALGAMATION)
|
|
|
+ #include "json_tool.h"
|
|
|
+#include <json/assertions.h>
|
|
|
+#include <json/reader.h>
|
|
|
+#include <json/value.h>
|
|
|
+#endif // if !defined(JSON_IS_AMALGAMATION)
|
|
|
+
|
|
|
+#include <algorithm>
|
|
|
+#include <cassert>
|
|
|
+#include <cstring>
|
|
|
+#include <iostream>
|
|
|
+#include <istream>
|
|
|
+#include <limits>
|
|
|
+#include <memory>
|
|
|
+#include <set>
|
|
|
+#include <sstream>
|
|
|
+#include <utility>
|
|
|
+
|
|
|
+#include <cstdio>
|
|
|
+
|
|
|
+#if __cplusplus >= 201103L
|
|
|
+
|
|
|
+#if !defined(sscanf)
|
|
|
+#define sscanf std::sscanf
|
|
|
+#endif
|
|
|
+
|
|
|
+#endif //__cplusplus
|
|
|
+
|
|
|
+#if defined(_MSC_VER)
|
|
|
+ #if !defined(_CRT_SECURE_CPP_OVERLOAD_STANDARD_NAMES)
|
|
|
+#define _CRT_SECURE_CPP_OVERLOAD_STANDARD_NAMES 1
|
|
|
+#endif //_CRT_SECURE_CPP_OVERLOAD_STANDARD_NAMES
|
|
|
+#endif //_MSC_VER
|
|
|
+
|
|
|
+#if defined(_MSC_VER)
|
|
|
+ // Disable warning about strdup being deprecated.
|
|
|
+#pragma warning(disable : 4996)
|
|
|
+#endif
|
|
|
+
|
|
|
+// Define JSONCPP_DEPRECATED_STACK_LIMIT as an appropriate integer at compile
|
|
|
+// time to change the stack limit
|
|
|
+#if !defined(JSONCPP_DEPRECATED_STACK_LIMIT)
|
|
|
+#define JSONCPP_DEPRECATED_STACK_LIMIT 1000
|
|
|
+#endif
|
|
|
+
|
|
|
+static size_t const stackLimit_g =
|
|
|
+ JSONCPP_DEPRECATED_STACK_LIMIT; // see readValue()
|
|
|
+
|
|
|
+namespace Json {
|
|
|
+
|
|
|
+#if __cplusplus >= 201103L || (defined(_CPPLIB_VER) && _CPPLIB_VER >= 520)
|
|
|
+ using CharReaderPtr = std::unique_ptr<CharReader>;
|
|
|
+#else
|
|
|
+ using CharReaderPtr = std::auto_ptr<CharReader>;
|
|
|
+#endif
|
|
|
+
|
|
|
+// Implementation of class Features
|
|
|
+// ////////////////////////////////
|
|
|
+
|
|
|
+ Features::Features() = default;
|
|
|
+
|
|
|
+ Features Features::all() { return {}; }
|
|
|
+
|
|
|
+ Features Features::strictMode() {
|
|
|
+ Features features;
|
|
|
+ features.allowComments_ = false;
|
|
|
+ features.strictRoot_ = true;
|
|
|
+ features.allowDroppedNullPlaceholders_ = false;
|
|
|
+ features.allowNumericKeys_ = false;
|
|
|
+ return features;
|
|
|
+ }
|
|
|
+
|
|
|
+// Implementation of class Reader
|
|
|
+// ////////////////////////////////
|
|
|
+
|
|
|
+ bool Reader::containsNewLine(Reader::Location begin, Reader::Location end) {
|
|
|
+ return std::any_of(begin, end, [](char b) { return b == '\n' || b == '\r'; });
|
|
|
+ }
|
|
|
+
|
|
|
+// Class Reader
|
|
|
+// //////////////////////////////////////////////////////////////////
|
|
|
+
|
|
|
+ Reader::Reader() : features_(Features::all()) {}
|
|
|
+
|
|
|
+ Reader::Reader(const Features &features) : features_(features) {}
|
|
|
+
|
|
|
+ bool Reader::parse(const std::string &document, Value &root,
|
|
|
+ bool collectComments) {
|
|
|
+ document_.assign(document.begin(), document.end());
|
|
|
+ const char *begin = document_.c_str();
|
|
|
+ const char *end = begin + document_.length();
|
|
|
+ return parse(begin, end, root, collectComments);
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Reader::parse(std::istream &is, Value &root, bool collectComments) {
|
|
|
+ // std::istream_iterator<char> begin(is);
|
|
|
+ // std::istream_iterator<char> end;
|
|
|
+ // Those would allow streamed input from a file, if parse() were a
|
|
|
+ // template function.
|
|
|
+
|
|
|
+ // Since String is reference-counted, this at least does not
|
|
|
+ // create an extra copy.
|
|
|
+ String doc;
|
|
|
+ std::getline(is, doc, static_cast<char> EOF);
|
|
|
+ return parse(doc.data(), doc.data() + doc.size(), root, collectComments);
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Reader::parse(const char *beginDoc, const char *endDoc, Value &root,
|
|
|
+ bool collectComments) {
|
|
|
+ if (!features_.allowComments_) {
|
|
|
+ collectComments = false;
|
|
|
+ }
|
|
|
+
|
|
|
+ begin_ = beginDoc;
|
|
|
+ end_ = endDoc;
|
|
|
+ collectComments_ = collectComments;
|
|
|
+ current_ = begin_;
|
|
|
+ lastValueEnd_ = nullptr;
|
|
|
+ lastValue_ = nullptr;
|
|
|
+ commentsBefore_.clear();
|
|
|
+ errors_.clear();
|
|
|
+ while (!nodes_.empty())
|
|
|
+ nodes_.pop();
|
|
|
+ nodes_.push(&root);
|
|
|
+
|
|
|
+ bool successful = readValue();
|
|
|
+ Token token;
|
|
|
+ skipCommentTokens(token);
|
|
|
+ if (collectComments_ && !commentsBefore_.empty())
|
|
|
+ root.setComment(commentsBefore_, commentAfter);
|
|
|
+ if (features_.strictRoot_) {
|
|
|
+ if (!root.isArray() && !root.isObject()) {
|
|
|
+ // Set error location to start of doc, ideally should be first token found
|
|
|
+ // in doc
|
|
|
+ token.type_ = tokenError;
|
|
|
+ token.start_ = beginDoc;
|
|
|
+ token.end_ = endDoc;
|
|
|
+ addError(
|
|
|
+ "A valid JSON document must be either an array or an object value.",
|
|
|
+ token);
|
|
|
+ return false;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ return successful;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Reader::readValue() {
|
|
|
+ // readValue() may call itself only if it calls readObject() or ReadArray().
|
|
|
+ // These methods execute nodes_.push() just before and nodes_.pop)() just
|
|
|
+ // after calling readValue(). parse() executes one nodes_.push(), so > instead
|
|
|
+ // of >=.
|
|
|
+ if (nodes_.size() > stackLimit_g)
|
|
|
+ throwRuntimeError("Exceeded stackLimit in readValue().");
|
|
|
+
|
|
|
+ Token token;
|
|
|
+ skipCommentTokens(token);
|
|
|
+ bool successful = true;
|
|
|
+
|
|
|
+ if (collectComments_ && !commentsBefore_.empty()) {
|
|
|
+ currentValue().setComment(commentsBefore_, commentBefore);
|
|
|
+ commentsBefore_.clear();
|
|
|
+ }
|
|
|
+
|
|
|
+ switch (token.type_) {
|
|
|
+ case tokenObjectBegin:
|
|
|
+ successful = readObject(token);
|
|
|
+ currentValue().setOffsetLimit(current_ - begin_);
|
|
|
+ break;
|
|
|
+ case tokenArrayBegin:
|
|
|
+ successful = readArray(token);
|
|
|
+ currentValue().setOffsetLimit(current_ - begin_);
|
|
|
+ break;
|
|
|
+ case tokenNumber:
|
|
|
+ successful = decodeNumber(token);
|
|
|
+ break;
|
|
|
+ case tokenString:
|
|
|
+ successful = decodeString(token);
|
|
|
+ break;
|
|
|
+ case tokenTrue: {
|
|
|
+ Value v(true);
|
|
|
+ currentValue().swapPayload(v);
|
|
|
+ currentValue().setOffsetStart(token.start_ - begin_);
|
|
|
+ currentValue().setOffsetLimit(token.end_ - begin_);
|
|
|
+ }
|
|
|
+ break;
|
|
|
+ case tokenFalse: {
|
|
|
+ Value v(false);
|
|
|
+ currentValue().swapPayload(v);
|
|
|
+ currentValue().setOffsetStart(token.start_ - begin_);
|
|
|
+ currentValue().setOffsetLimit(token.end_ - begin_);
|
|
|
+ }
|
|
|
+ break;
|
|
|
+ case tokenNull: {
|
|
|
+ Value v;
|
|
|
+ currentValue().swapPayload(v);
|
|
|
+ currentValue().setOffsetStart(token.start_ - begin_);
|
|
|
+ currentValue().setOffsetLimit(token.end_ - begin_);
|
|
|
+ }
|
|
|
+ break;
|
|
|
+ case tokenArraySeparator:
|
|
|
+ case tokenObjectEnd:
|
|
|
+ case tokenArrayEnd:
|
|
|
+ if (features_.allowDroppedNullPlaceholders_) {
|
|
|
+ // "Un-read" the current token and mark the current value as a null
|
|
|
+ // token.
|
|
|
+ current_--;
|
|
|
+ Value v;
|
|
|
+ currentValue().swapPayload(v);
|
|
|
+ currentValue().setOffsetStart(current_ - begin_ - 1);
|
|
|
+ currentValue().setOffsetLimit(current_ - begin_);
|
|
|
+ break;
|
|
|
+ } // Else, fall through...
|
|
|
+ default:
|
|
|
+ currentValue().setOffsetStart(token.start_ - begin_);
|
|
|
+ currentValue().setOffsetLimit(token.end_ - begin_);
|
|
|
+ return addError("Syntax error: value, object or array expected.", token);
|
|
|
+ }
|
|
|
+
|
|
|
+ if (collectComments_) {
|
|
|
+ lastValueEnd_ = current_;
|
|
|
+ lastValue_ = ¤tValue();
|
|
|
+ }
|
|
|
+
|
|
|
+ return successful;
|
|
|
+ }
|
|
|
+
|
|
|
+ void Reader::skipCommentTokens(Token &token) {
|
|
|
+ if (features_.allowComments_) {
|
|
|
+ do {
|
|
|
+ readToken(token);
|
|
|
+ } while (token.type_ == tokenComment);
|
|
|
+ } else {
|
|
|
+ readToken(token);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Reader::readToken(Token &token) {
|
|
|
+ skipSpaces();
|
|
|
+ token.start_ = current_;
|
|
|
+ Char c = getNextChar();
|
|
|
+ bool ok = true;
|
|
|
+ switch (c) {
|
|
|
+ case '{':
|
|
|
+ token.type_ = tokenObjectBegin;
|
|
|
+ break;
|
|
|
+ case '}':
|
|
|
+ token.type_ = tokenObjectEnd;
|
|
|
+ break;
|
|
|
+ case '[':
|
|
|
+ token.type_ = tokenArrayBegin;
|
|
|
+ break;
|
|
|
+ case ']':
|
|
|
+ token.type_ = tokenArrayEnd;
|
|
|
+ break;
|
|
|
+ case '"':
|
|
|
+ token.type_ = tokenString;
|
|
|
+ ok = readString();
|
|
|
+ break;
|
|
|
+ case '/':
|
|
|
+ token.type_ = tokenComment;
|
|
|
+ ok = readComment();
|
|
|
+ break;
|
|
|
+ case '0':
|
|
|
+ case '1':
|
|
|
+ case '2':
|
|
|
+ case '3':
|
|
|
+ case '4':
|
|
|
+ case '5':
|
|
|
+ case '6':
|
|
|
+ case '7':
|
|
|
+ case '8':
|
|
|
+ case '9':
|
|
|
+ case '-':
|
|
|
+ token.type_ = tokenNumber;
|
|
|
+ readNumber();
|
|
|
+ break;
|
|
|
+ case 't':
|
|
|
+ token.type_ = tokenTrue;
|
|
|
+ ok = match("rue", 3);
|
|
|
+ break;
|
|
|
+ case 'f':
|
|
|
+ token.type_ = tokenFalse;
|
|
|
+ ok = match("alse", 4);
|
|
|
+ break;
|
|
|
+ case 'n':
|
|
|
+ token.type_ = tokenNull;
|
|
|
+ ok = match("ull", 3);
|
|
|
+ break;
|
|
|
+ case ',':
|
|
|
+ token.type_ = tokenArraySeparator;
|
|
|
+ break;
|
|
|
+ case ':':
|
|
|
+ token.type_ = tokenMemberSeparator;
|
|
|
+ break;
|
|
|
+ case 0:
|
|
|
+ token.type_ = tokenEndOfStream;
|
|
|
+ break;
|
|
|
+ default:
|
|
|
+ ok = false;
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ if (!ok)
|
|
|
+ token.type_ = tokenError;
|
|
|
+ token.end_ = current_;
|
|
|
+ return ok;
|
|
|
+ }
|
|
|
+
|
|
|
+ void Reader::skipSpaces() {
|
|
|
+ while (current_ != end_) {
|
|
|
+ Char c = *current_;
|
|
|
+ if (c == ' ' || c == '\t' || c == '\r' || c == '\n')
|
|
|
+ ++current_;
|
|
|
+ else
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Reader::match(const Char *pattern, int patternLength) {
|
|
|
+ if (end_ - current_ < patternLength)
|
|
|
+ return false;
|
|
|
+ int index = patternLength;
|
|
|
+ while (index--)
|
|
|
+ if (current_[index] != pattern[index])
|
|
|
+ return false;
|
|
|
+ current_ += patternLength;
|
|
|
+ return true;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Reader::readComment() {
|
|
|
+ Location commentBegin = current_ - 1;
|
|
|
+ Char c = getNextChar();
|
|
|
+ bool successful = false;
|
|
|
+ if (c == '*')
|
|
|
+ successful = readCStyleComment();
|
|
|
+ else if (c == '/')
|
|
|
+ successful = readCppStyleComment();
|
|
|
+ if (!successful)
|
|
|
+ return false;
|
|
|
+
|
|
|
+ if (collectComments_) {
|
|
|
+ CommentPlacement placement = commentBefore;
|
|
|
+ if (lastValueEnd_ && !containsNewLine(lastValueEnd_, commentBegin)) {
|
|
|
+ if (c != '*' || !containsNewLine(commentBegin, current_))
|
|
|
+ placement = commentAfterOnSameLine;
|
|
|
+ }
|
|
|
+
|
|
|
+ addComment(commentBegin, current_, placement);
|
|
|
+ }
|
|
|
+ return true;
|
|
|
+ }
|
|
|
+
|
|
|
+ String Reader::normalizeEOL(Reader::Location begin, Reader::Location end) {
|
|
|
+ String normalized;
|
|
|
+ normalized.reserve(static_cast<size_t>(end - begin));
|
|
|
+ Reader::Location current = begin;
|
|
|
+ while (current != end) {
|
|
|
+ char c = *current++;
|
|
|
+ if (c == '\r') {
|
|
|
+ if (current != end && *current == '\n')
|
|
|
+ // convert dos EOL
|
|
|
+ ++current;
|
|
|
+ // convert Mac EOL
|
|
|
+ normalized += '\n';
|
|
|
+ } else {
|
|
|
+ normalized += c;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ return normalized;
|
|
|
+ }
|
|
|
+
|
|
|
+ void Reader::addComment(Location begin, Location end,
|
|
|
+ CommentPlacement placement) {
|
|
|
+ assert(collectComments_);
|
|
|
+ const String &normalized = normalizeEOL(begin, end);
|
|
|
+ if (placement == commentAfterOnSameLine) {
|
|
|
+ assert(lastValue_ != nullptr);
|
|
|
+ lastValue_->setComment(normalized, placement);
|
|
|
+ } else {
|
|
|
+ commentsBefore_ += normalized;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Reader::readCStyleComment() {
|
|
|
+ while ((current_ + 1) < end_) {
|
|
|
+ Char c = getNextChar();
|
|
|
+ if (c == '*' && *current_ == '/')
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ return getNextChar() == '/';
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Reader::readCppStyleComment() {
|
|
|
+ while (current_ != end_) {
|
|
|
+ Char c = getNextChar();
|
|
|
+ if (c == '\n')
|
|
|
+ break;
|
|
|
+ if (c == '\r') {
|
|
|
+ // Consume DOS EOL. It will be normalized in addComment.
|
|
|
+ if (current_ != end_ && *current_ == '\n')
|
|
|
+ getNextChar();
|
|
|
+ // Break on Moc OS 9 EOL.
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ return true;
|
|
|
+ }
|
|
|
+
|
|
|
+ void Reader::readNumber() {
|
|
|
+ Location p = current_;
|
|
|
+ char c = '0'; // stopgap for already consumed character
|
|
|
+ // integral part
|
|
|
+ while (c >= '0' && c <= '9')
|
|
|
+ c = (current_ = p) < end_ ? *p++ : '\0';
|
|
|
+ // fractional part
|
|
|
+ if (c == '.') {
|
|
|
+ c = (current_ = p) < end_ ? *p++ : '\0';
|
|
|
+ while (c >= '0' && c <= '9')
|
|
|
+ c = (current_ = p) < end_ ? *p++ : '\0';
|
|
|
+ }
|
|
|
+ // exponential part
|
|
|
+ if (c == 'e' || c == 'E') {
|
|
|
+ c = (current_ = p) < end_ ? *p++ : '\0';
|
|
|
+ if (c == '+' || c == '-')
|
|
|
+ c = (current_ = p) < end_ ? *p++ : '\0';
|
|
|
+ while (c >= '0' && c <= '9')
|
|
|
+ c = (current_ = p) < end_ ? *p++ : '\0';
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Reader::readString() {
|
|
|
+ Char c = '\0';
|
|
|
+ while (current_ != end_) {
|
|
|
+ c = getNextChar();
|
|
|
+ if (c == '\\')
|
|
|
+ getNextChar();
|
|
|
+ else if (c == '"')
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ return c == '"';
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Reader::readObject(Token &token) {
|
|
|
+ Token tokenName;
|
|
|
+ String name;
|
|
|
+ Value init(objectValue);
|
|
|
+ currentValue().swapPayload(init);
|
|
|
+ currentValue().setOffsetStart(token.start_ - begin_);
|
|
|
+ while (readToken(tokenName)) {
|
|
|
+ bool initialTokenOk = true;
|
|
|
+ while (tokenName.type_ == tokenComment && initialTokenOk)
|
|
|
+ initialTokenOk = readToken(tokenName);
|
|
|
+ if (!initialTokenOk)
|
|
|
+ break;
|
|
|
+ if (tokenName.type_ == tokenObjectEnd && name.empty()) // empty object
|
|
|
+ return true;
|
|
|
+ name.clear();
|
|
|
+ if (tokenName.type_ == tokenString) {
|
|
|
+ if (!decodeString(tokenName, name))
|
|
|
+ return recoverFromError(tokenObjectEnd);
|
|
|
+ } else if (tokenName.type_ == tokenNumber && features_.allowNumericKeys_) {
|
|
|
+ Value numberName;
|
|
|
+ if (!decodeNumber(tokenName, numberName))
|
|
|
+ return recoverFromError(tokenObjectEnd);
|
|
|
+ name = numberName.asString();
|
|
|
+ } else {
|
|
|
+ break;
|
|
|
+ }
|
|
|
+
|
|
|
+ Token colon;
|
|
|
+ if (!readToken(colon) || colon.type_ != tokenMemberSeparator) {
|
|
|
+ return addErrorAndRecover("Missing ':' after object member name", colon,
|
|
|
+ tokenObjectEnd);
|
|
|
+ }
|
|
|
+ Value &value = currentValue()[name];
|
|
|
+ nodes_.push(&value);
|
|
|
+ bool ok = readValue();
|
|
|
+ nodes_.pop();
|
|
|
+ if (!ok) // error already put
|
|
|
+ return recoverFromError(tokenObjectEnd);
|
|
|
+
|
|
|
+ Token comma;
|
|
|
+ if (!readToken(comma) ||
|
|
|
+ (comma.type_ != tokenObjectEnd && comma.type_ != tokenArraySeparator &&
|
|
|
+ comma.type_ != tokenComment)) {
|
|
|
+ return addErrorAndRecover("Missing ',' or '}' in object declaration",
|
|
|
+ comma, tokenObjectEnd);
|
|
|
+ }
|
|
|
+ bool finalizeTokenOk = true;
|
|
|
+ while (comma.type_ == tokenComment && finalizeTokenOk)
|
|
|
+ finalizeTokenOk = readToken(comma);
|
|
|
+ if (comma.type_ == tokenObjectEnd)
|
|
|
+ return true;
|
|
|
+ }
|
|
|
+ return addErrorAndRecover("Missing '}' or object member name", tokenName,
|
|
|
+ tokenObjectEnd);
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Reader::readArray(Token &token) {
|
|
|
+ Value init(arrayValue);
|
|
|
+ currentValue().swapPayload(init);
|
|
|
+ currentValue().setOffsetStart(token.start_ - begin_);
|
|
|
+ skipSpaces();
|
|
|
+ if (current_ != end_ && *current_ == ']') // empty array
|
|
|
+ {
|
|
|
+ Token endArray;
|
|
|
+ readToken(endArray);
|
|
|
+ return true;
|
|
|
+ }
|
|
|
+ int index = 0;
|
|
|
+ for (;;) {
|
|
|
+ Value &value = currentValue()[index++];
|
|
|
+ nodes_.push(&value);
|
|
|
+ bool ok = readValue();
|
|
|
+ nodes_.pop();
|
|
|
+ if (!ok) // error already put
|
|
|
+ return recoverFromError(tokenArrayEnd);
|
|
|
+
|
|
|
+ Token currentToken;
|
|
|
+ // Accept Comment after last item in the array.
|
|
|
+ ok = readToken(currentToken);
|
|
|
+ while (currentToken.type_ == tokenComment && ok) {
|
|
|
+ ok = readToken(currentToken);
|
|
|
+ }
|
|
|
+ bool badTokenType = (currentToken.type_ != tokenArraySeparator &&
|
|
|
+ currentToken.type_ != tokenArrayEnd);
|
|
|
+ if (!ok || badTokenType) {
|
|
|
+ return addErrorAndRecover("Missing ',' or ']' in array declaration",
|
|
|
+ currentToken, tokenArrayEnd);
|
|
|
+ }
|
|
|
+ if (currentToken.type_ == tokenArrayEnd)
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ return true;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Reader::decodeNumber(Token &token) {
|
|
|
+ Value decoded;
|
|
|
+ if (!decodeNumber(token, decoded))
|
|
|
+ return false;
|
|
|
+ currentValue().swapPayload(decoded);
|
|
|
+ currentValue().setOffsetStart(token.start_ - begin_);
|
|
|
+ currentValue().setOffsetLimit(token.end_ - begin_);
|
|
|
+ return true;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Reader::decodeNumber(Token &token, Value &decoded) {
|
|
|
+ // Attempts to parse the number as an integer. If the number is
|
|
|
+ // larger than the maximum supported value of an integer then
|
|
|
+ // we decode the number as a double.
|
|
|
+ Location current = token.start_;
|
|
|
+ bool isNegative = *current == '-';
|
|
|
+ if (isNegative)
|
|
|
+ ++current;
|
|
|
+ // TODO: Help the compiler do the div and mod at compile time or get rid of
|
|
|
+ // them.
|
|
|
+ Value::LargestUInt maxIntegerValue =
|
|
|
+ isNegative ? Value::LargestUInt(Value::maxLargestInt) + 1
|
|
|
+ : Value::maxLargestUInt;
|
|
|
+ Value::LargestUInt threshold = maxIntegerValue / 10;
|
|
|
+ Value::LargestUInt value = 0;
|
|
|
+ while (current < token.end_) {
|
|
|
+ Char c = *current++;
|
|
|
+ if (c < '0' || c > '9')
|
|
|
+ return decodeDouble(token, decoded);
|
|
|
+ auto digit(static_cast<Value::UInt>(c - '0'));
|
|
|
+ if (value >= threshold) {
|
|
|
+ // We've hit or exceeded the max value divided by 10 (rounded down). If
|
|
|
+ // a) we've only just touched the limit, b) this is the last digit, and
|
|
|
+ // c) it's small enough to fit in that rounding delta, we're okay.
|
|
|
+ // Otherwise treat this number as a double to avoid overflow.
|
|
|
+ if (value > threshold || current != token.end_ ||
|
|
|
+ digit > maxIntegerValue % 10) {
|
|
|
+ return decodeDouble(token, decoded);
|
|
|
+ }
|
|
|
+ }
|
|
|
+ value = value * 10 + digit;
|
|
|
+ }
|
|
|
+ if (isNegative && value == maxIntegerValue)
|
|
|
+ decoded = Value::minLargestInt;
|
|
|
+ else if (isNegative)
|
|
|
+ decoded = -Value::LargestInt(value);
|
|
|
+ else if (value <= Value::LargestUInt(Value::maxInt))
|
|
|
+ decoded = Value::LargestInt(value);
|
|
|
+ else
|
|
|
+ decoded = value;
|
|
|
+ return true;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Reader::decodeDouble(Token &token) {
|
|
|
+ Value decoded;
|
|
|
+ if (!decodeDouble(token, decoded))
|
|
|
+ return false;
|
|
|
+ currentValue().swapPayload(decoded);
|
|
|
+ currentValue().setOffsetStart(token.start_ - begin_);
|
|
|
+ currentValue().setOffsetLimit(token.end_ - begin_);
|
|
|
+ return true;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Reader::decodeDouble(Token &token, Value &decoded) {
|
|
|
+ double value = 0;
|
|
|
+ String buffer(token.start_, token.end_);
|
|
|
+ IStringStream is(buffer);
|
|
|
+ if (!(is >> value))
|
|
|
+ return addError(
|
|
|
+ "'" + String(token.start_, token.end_) + "' is not a number.", token);
|
|
|
+ decoded = value;
|
|
|
+ return true;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Reader::decodeString(Token &token) {
|
|
|
+ String decoded_string;
|
|
|
+ if (!decodeString(token, decoded_string))
|
|
|
+ return false;
|
|
|
+ Value decoded(decoded_string);
|
|
|
+ currentValue().swapPayload(decoded);
|
|
|
+ currentValue().setOffsetStart(token.start_ - begin_);
|
|
|
+ currentValue().setOffsetLimit(token.end_ - begin_);
|
|
|
+ return true;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Reader::decodeString(Token &token, String &decoded) {
|
|
|
+ decoded.reserve(static_cast<size_t>(token.end_ - token.start_ - 2));
|
|
|
+ Location current = token.start_ + 1; // skip '"'
|
|
|
+ Location end = token.end_ - 1; // do not include '"'
|
|
|
+ while (current != end) {
|
|
|
+ Char c = *current++;
|
|
|
+ if (c == '"')
|
|
|
+ break;
|
|
|
+ if (c == '\\') {
|
|
|
+ if (current == end)
|
|
|
+ return addError("Empty escape sequence in string", token, current);
|
|
|
+ Char escape = *current++;
|
|
|
+ switch (escape) {
|
|
|
+ case '"':
|
|
|
+ decoded += '"';
|
|
|
+ break;
|
|
|
+ case '/':
|
|
|
+ decoded += '/';
|
|
|
+ break;
|
|
|
+ case '\\':
|
|
|
+ decoded += '\\';
|
|
|
+ break;
|
|
|
+ case 'b':
|
|
|
+ decoded += '\b';
|
|
|
+ break;
|
|
|
+ case 'f':
|
|
|
+ decoded += '\f';
|
|
|
+ break;
|
|
|
+ case 'n':
|
|
|
+ decoded += '\n';
|
|
|
+ break;
|
|
|
+ case 'r':
|
|
|
+ decoded += '\r';
|
|
|
+ break;
|
|
|
+ case 't':
|
|
|
+ decoded += '\t';
|
|
|
+ break;
|
|
|
+ case 'u': {
|
|
|
+ unsigned int unicode;
|
|
|
+ if (!decodeUnicodeCodePoint(token, current, end, unicode))
|
|
|
+ return false;
|
|
|
+ decoded += codePointToUTF8(unicode);
|
|
|
+ }
|
|
|
+ break;
|
|
|
+ default:
|
|
|
+ return addError("Bad escape sequence in string", token, current);
|
|
|
+ }
|
|
|
+ } else {
|
|
|
+ decoded += c;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ return true;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Reader::decodeUnicodeCodePoint(Token &token, Location ¤t,
|
|
|
+ Location end, unsigned int &unicode) {
|
|
|
+
|
|
|
+ if (!decodeUnicodeEscapeSequence(token, current, end, unicode))
|
|
|
+ return false;
|
|
|
+ if (unicode >= 0xD800 && unicode <= 0xDBFF) {
|
|
|
+ // surrogate pairs
|
|
|
+ if (end - current < 6)
|
|
|
+ return addError(
|
|
|
+ "additional six characters expected to parse unicode surrogate pair.",
|
|
|
+ token, current);
|
|
|
+ if (*(current++) == '\\' && *(current++) == 'u') {
|
|
|
+ unsigned int surrogatePair;
|
|
|
+ if (decodeUnicodeEscapeSequence(token, current, end, surrogatePair)) {
|
|
|
+ unicode = 0x10000 + ((unicode & 0x3FF) << 10) + (surrogatePair & 0x3FF);
|
|
|
+ } else
|
|
|
+ return false;
|
|
|
+ } else
|
|
|
+ return addError("expecting another \\u token to begin the second half of "
|
|
|
+ "a unicode surrogate pair",
|
|
|
+ token, current);
|
|
|
+ }
|
|
|
+ return true;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Reader::decodeUnicodeEscapeSequence(Token &token, Location ¤t,
|
|
|
+ Location end,
|
|
|
+ unsigned int &ret_unicode) {
|
|
|
+ if (end - current < 4)
|
|
|
+ return addError(
|
|
|
+ "Bad unicode escape sequence in string: four digits expected.", token,
|
|
|
+ current);
|
|
|
+ int unicode = 0;
|
|
|
+ for (int index = 0; index < 4; ++index) {
|
|
|
+ Char c = *current++;
|
|
|
+ unicode *= 16;
|
|
|
+ if (c >= '0' && c <= '9')
|
|
|
+ unicode += c - '0';
|
|
|
+ else if (c >= 'a' && c <= 'f')
|
|
|
+ unicode += c - 'a' + 10;
|
|
|
+ else if (c >= 'A' && c <= 'F')
|
|
|
+ unicode += c - 'A' + 10;
|
|
|
+ else
|
|
|
+ return addError(
|
|
|
+ "Bad unicode escape sequence in string: hexadecimal digit expected.",
|
|
|
+ token, current);
|
|
|
+ }
|
|
|
+ ret_unicode = static_cast<unsigned int>(unicode);
|
|
|
+ return true;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Reader::addError(const String &message, Token &token, Location extra) {
|
|
|
+ ErrorInfo info;
|
|
|
+ info.token_ = token;
|
|
|
+ info.message_ = message;
|
|
|
+ info.extra_ = extra;
|
|
|
+ errors_.push_back(info);
|
|
|
+ return false;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Reader::recoverFromError(TokenType skipUntilToken) {
|
|
|
+ size_t const errorCount = errors_.size();
|
|
|
+ Token skip;
|
|
|
+ for (;;) {
|
|
|
+ if (!readToken(skip))
|
|
|
+ errors_.resize(errorCount); // discard errors caused by recovery
|
|
|
+ if (skip.type_ == skipUntilToken || skip.type_ == tokenEndOfStream)
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ errors_.resize(errorCount);
|
|
|
+ return false;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Reader::addErrorAndRecover(const String &message, Token &token,
|
|
|
+ TokenType skipUntilToken) {
|
|
|
+ addError(message, token);
|
|
|
+ return recoverFromError(skipUntilToken);
|
|
|
+ }
|
|
|
+
|
|
|
+ Value &Reader::currentValue() { return *(nodes_.top()); }
|
|
|
+
|
|
|
+ Reader::Char Reader::getNextChar() {
|
|
|
+ if (current_ == end_)
|
|
|
+ return 0;
|
|
|
+ return *current_++;
|
|
|
+ }
|
|
|
+
|
|
|
+ void Reader::getLocationLineAndColumn(Location location, int &line,
|
|
|
+ int &column) const {
|
|
|
+ Location current = begin_;
|
|
|
+ Location lastLineStart = current;
|
|
|
+ line = 0;
|
|
|
+ while (current < location && current != end_) {
|
|
|
+ Char c = *current++;
|
|
|
+ if (c == '\r') {
|
|
|
+ if (*current == '\n')
|
|
|
+ ++current;
|
|
|
+ lastLineStart = current;
|
|
|
+ ++line;
|
|
|
+ } else if (c == '\n') {
|
|
|
+ lastLineStart = current;
|
|
|
+ ++line;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ // column & line start at 1
|
|
|
+ column = int(location - lastLineStart) + 1;
|
|
|
+ ++line;
|
|
|
+ }
|
|
|
+
|
|
|
+ String Reader::getLocationLineAndColumn(Location location) const {
|
|
|
+ int line, column;
|
|
|
+ getLocationLineAndColumn(location, line, column);
|
|
|
+ char buffer[18 + 16 + 16 + 1];
|
|
|
+ jsoncpp_snprintf(buffer, sizeof(buffer), "Line %d, Column %d", line, column);
|
|
|
+ return buffer;
|
|
|
+ }
|
|
|
+
|
|
|
+// Deprecated. Preserved for backward compatibility
|
|
|
+ String Reader::getFormatedErrorMessages() const {
|
|
|
+ return getFormattedErrorMessages();
|
|
|
+ }
|
|
|
+
|
|
|
+ String Reader::getFormattedErrorMessages() const {
|
|
|
+ String formattedMessage;
|
|
|
+ for (const auto &error : errors_) {
|
|
|
+ formattedMessage +=
|
|
|
+ "* " + getLocationLineAndColumn(error.token_.start_) + "\n";
|
|
|
+ formattedMessage += " " + error.message_ + "\n";
|
|
|
+ if (error.extra_)
|
|
|
+ formattedMessage +=
|
|
|
+ "See " + getLocationLineAndColumn(error.extra_) + " for detail.\n";
|
|
|
+ }
|
|
|
+ return formattedMessage;
|
|
|
+ }
|
|
|
+
|
|
|
+ std::vector<Reader::StructuredError> Reader::getStructuredErrors() const {
|
|
|
+ std::vector<Reader::StructuredError> allErrors;
|
|
|
+ for (const auto &error : errors_) {
|
|
|
+ Reader::StructuredError structured;
|
|
|
+ structured.offset_start = error.token_.start_ - begin_;
|
|
|
+ structured.offset_limit = error.token_.end_ - begin_;
|
|
|
+ structured.message = error.message_;
|
|
|
+ allErrors.push_back(structured);
|
|
|
+ }
|
|
|
+ return allErrors;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Reader::pushError(const Value &value, const String &message) {
|
|
|
+ ptrdiff_t const length = end_ - begin_;
|
|
|
+ if (value.getOffsetStart() > length || value.getOffsetLimit() > length)
|
|
|
+ return false;
|
|
|
+ Token token;
|
|
|
+ token.type_ = tokenError;
|
|
|
+ token.start_ = begin_ + value.getOffsetStart();
|
|
|
+ token.end_ = begin_ + value.getOffsetLimit();
|
|
|
+ ErrorInfo info;
|
|
|
+ info.token_ = token;
|
|
|
+ info.message_ = message;
|
|
|
+ info.extra_ = nullptr;
|
|
|
+ errors_.push_back(info);
|
|
|
+ return true;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Reader::pushError(const Value &value, const String &message,
|
|
|
+ const Value &extra) {
|
|
|
+ ptrdiff_t const length = end_ - begin_;
|
|
|
+ if (value.getOffsetStart() > length || value.getOffsetLimit() > length ||
|
|
|
+ extra.getOffsetLimit() > length)
|
|
|
+ return false;
|
|
|
+ Token token;
|
|
|
+ token.type_ = tokenError;
|
|
|
+ token.start_ = begin_ + value.getOffsetStart();
|
|
|
+ token.end_ = begin_ + value.getOffsetLimit();
|
|
|
+ ErrorInfo info;
|
|
|
+ info.token_ = token;
|
|
|
+ info.message_ = message;
|
|
|
+ info.extra_ = begin_ + extra.getOffsetStart();
|
|
|
+ errors_.push_back(info);
|
|
|
+ return true;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Reader::good() const { return errors_.empty(); }
|
|
|
+
|
|
|
+// Originally copied from the Features class (now deprecated), used internally
|
|
|
+// for features implementation.
|
|
|
+ class OurFeatures {
|
|
|
+ public:
|
|
|
+ static OurFeatures all();
|
|
|
+
|
|
|
+ bool allowComments_;
|
|
|
+ bool allowTrailingCommas_;
|
|
|
+ bool strictRoot_;
|
|
|
+ bool allowDroppedNullPlaceholders_;
|
|
|
+ bool allowNumericKeys_;
|
|
|
+ bool allowSingleQuotes_;
|
|
|
+ bool failIfExtra_;
|
|
|
+ bool rejectDupKeys_;
|
|
|
+ bool allowSpecialFloats_;
|
|
|
+ bool skipBom_;
|
|
|
+ size_t stackLimit_;
|
|
|
+ }; // OurFeatures
|
|
|
+
|
|
|
+ OurFeatures OurFeatures::all() { return {}; }
|
|
|
+
|
|
|
+// Implementation of class Reader
|
|
|
+// ////////////////////////////////
|
|
|
+
|
|
|
+// Originally copied from the Reader class (now deprecated), used internally
|
|
|
+// for implementing JSON reading.
|
|
|
+ class OurReader {
|
|
|
+ public:
|
|
|
+ using Char = char;
|
|
|
+ using Location = const Char *;
|
|
|
+ struct StructuredError {
|
|
|
+ ptrdiff_t offset_start;
|
|
|
+ ptrdiff_t offset_limit;
|
|
|
+ String message;
|
|
|
+ };
|
|
|
+
|
|
|
+ explicit OurReader(OurFeatures const &features);
|
|
|
+
|
|
|
+ bool parse(const char *beginDoc, const char *endDoc, Value &root,
|
|
|
+ bool collectComments = true);
|
|
|
+
|
|
|
+ String getFormattedErrorMessages() const;
|
|
|
+
|
|
|
+ std::vector<StructuredError> getStructuredErrors() const;
|
|
|
+
|
|
|
+ private:
|
|
|
+ OurReader(OurReader const &); // no impl
|
|
|
+ void operator=(OurReader const &); // no impl
|
|
|
+
|
|
|
+ enum TokenType {
|
|
|
+ tokenEndOfStream = 0,
|
|
|
+ tokenObjectBegin,
|
|
|
+ tokenObjectEnd,
|
|
|
+ tokenArrayBegin,
|
|
|
+ tokenArrayEnd,
|
|
|
+ tokenString,
|
|
|
+ tokenNumber,
|
|
|
+ tokenTrue,
|
|
|
+ tokenFalse,
|
|
|
+ tokenNull,
|
|
|
+ tokenNaN,
|
|
|
+ tokenPosInf,
|
|
|
+ tokenNegInf,
|
|
|
+ tokenArraySeparator,
|
|
|
+ tokenMemberSeparator,
|
|
|
+ tokenComment,
|
|
|
+ tokenError
|
|
|
+ };
|
|
|
+
|
|
|
+ class Token {
|
|
|
+ public:
|
|
|
+ TokenType type_;
|
|
|
+ Location start_;
|
|
|
+ Location end_;
|
|
|
+ };
|
|
|
+
|
|
|
+ class ErrorInfo {
|
|
|
+ public:
|
|
|
+ Token token_;
|
|
|
+ String message_;
|
|
|
+ Location extra_;
|
|
|
+ };
|
|
|
+
|
|
|
+ using Errors = std::deque<ErrorInfo>;
|
|
|
+
|
|
|
+ bool readToken(Token &token);
|
|
|
+
|
|
|
+ void skipSpaces();
|
|
|
+
|
|
|
+ void skipBom(bool skipBom);
|
|
|
+
|
|
|
+ bool match(const Char *pattern, int patternLength);
|
|
|
+
|
|
|
+ bool readComment();
|
|
|
+
|
|
|
+ bool readCStyleComment(bool *containsNewLineResult);
|
|
|
+
|
|
|
+ bool readCppStyleComment();
|
|
|
+
|
|
|
+ bool readString();
|
|
|
+
|
|
|
+ bool readStringSingleQuote();
|
|
|
+
|
|
|
+ bool readNumber(bool checkInf);
|
|
|
+
|
|
|
+ bool readValue();
|
|
|
+
|
|
|
+ bool readObject(Token &token);
|
|
|
+
|
|
|
+ bool readArray(Token &token);
|
|
|
+
|
|
|
+ bool decodeNumber(Token &token);
|
|
|
+
|
|
|
+ bool decodeNumber(Token &token, Value &decoded);
|
|
|
+
|
|
|
+ bool decodeString(Token &token);
|
|
|
+
|
|
|
+ bool decodeString(Token &token, String &decoded);
|
|
|
+
|
|
|
+ bool decodeDouble(Token &token);
|
|
|
+
|
|
|
+ bool decodeDouble(Token &token, Value &decoded);
|
|
|
+
|
|
|
+ bool decodeUnicodeCodePoint(Token &token, Location ¤t, Location end,
|
|
|
+ unsigned int &unicode);
|
|
|
+
|
|
|
+ bool decodeUnicodeEscapeSequence(Token &token, Location ¤t,
|
|
|
+ Location end, unsigned int &unicode);
|
|
|
+
|
|
|
+ bool addError(const String &message, Token &token, Location extra = nullptr);
|
|
|
+
|
|
|
+ bool recoverFromError(TokenType skipUntilToken);
|
|
|
+
|
|
|
+ bool addErrorAndRecover(const String &message, Token &token,
|
|
|
+ TokenType skipUntilToken);
|
|
|
+
|
|
|
+ void skipUntilSpace();
|
|
|
+
|
|
|
+ Value ¤tValue();
|
|
|
+
|
|
|
+ Char getNextChar();
|
|
|
+
|
|
|
+ void getLocationLineAndColumn(Location location, int &line,
|
|
|
+ int &column) const;
|
|
|
+
|
|
|
+ String getLocationLineAndColumn(Location location) const;
|
|
|
+
|
|
|
+ void addComment(Location begin, Location end, CommentPlacement placement);
|
|
|
+
|
|
|
+ void skipCommentTokens(Token &token);
|
|
|
+
|
|
|
+ static String normalizeEOL(Location begin, Location end);
|
|
|
+
|
|
|
+ static bool containsNewLine(Location begin, Location end);
|
|
|
+
|
|
|
+ using Nodes = std::stack<Value *>;
|
|
|
+
|
|
|
+ Nodes nodes_{};
|
|
|
+ Errors errors_{};
|
|
|
+ String document_{};
|
|
|
+ Location begin_ = nullptr;
|
|
|
+ Location end_ = nullptr;
|
|
|
+ Location current_ = nullptr;
|
|
|
+ Location lastValueEnd_ = nullptr;
|
|
|
+ Value *lastValue_ = nullptr;
|
|
|
+ bool lastValueHasAComment_ = false;
|
|
|
+ String commentsBefore_{};
|
|
|
+
|
|
|
+ OurFeatures const features_;
|
|
|
+ bool collectComments_ = false;
|
|
|
+ }; // OurReader
|
|
|
+
|
|
|
+// complete copy of Read impl, for OurReader
|
|
|
+
|
|
|
+ bool OurReader::containsNewLine(OurReader::Location begin,
|
|
|
+ OurReader::Location end) {
|
|
|
+ return std::any_of(begin, end, [](char b) { return b == '\n' || b == '\r'; });
|
|
|
+ }
|
|
|
+
|
|
|
+ OurReader::OurReader(OurFeatures const &features) : features_(features) {}
|
|
|
+
|
|
|
+ bool OurReader::parse(const char *beginDoc, const char *endDoc, Value &root,
|
|
|
+ bool collectComments) {
|
|
|
+ if (!features_.allowComments_) {
|
|
|
+ collectComments = false;
|
|
|
+ }
|
|
|
+
|
|
|
+ begin_ = beginDoc;
|
|
|
+ end_ = endDoc;
|
|
|
+ collectComments_ = collectComments;
|
|
|
+ current_ = begin_;
|
|
|
+ lastValueEnd_ = nullptr;
|
|
|
+ lastValue_ = nullptr;
|
|
|
+ commentsBefore_.clear();
|
|
|
+ errors_.clear();
|
|
|
+ while (!nodes_.empty())
|
|
|
+ nodes_.pop();
|
|
|
+ nodes_.push(&root);
|
|
|
+
|
|
|
+ // skip byte order mark if it exists at the beginning of the UTF-8 text.
|
|
|
+ skipBom(features_.skipBom_);
|
|
|
+ bool successful = readValue();
|
|
|
+ nodes_.pop();
|
|
|
+ Token token;
|
|
|
+ skipCommentTokens(token);
|
|
|
+ if (features_.failIfExtra_ && (token.type_ != tokenEndOfStream)) {
|
|
|
+ addError("Extra non-whitespace after JSON value.", token);
|
|
|
+ return false;
|
|
|
+ }
|
|
|
+ if (collectComments_ && !commentsBefore_.empty())
|
|
|
+ root.setComment(commentsBefore_, commentAfter);
|
|
|
+ if (features_.strictRoot_) {
|
|
|
+ if (!root.isArray() && !root.isObject()) {
|
|
|
+ // Set error location to start of doc, ideally should be first token found
|
|
|
+ // in doc
|
|
|
+ token.type_ = tokenError;
|
|
|
+ token.start_ = beginDoc;
|
|
|
+ token.end_ = endDoc;
|
|
|
+ addError(
|
|
|
+ "A valid JSON document must be either an array or an object value.",
|
|
|
+ token);
|
|
|
+ return false;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ return successful;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool OurReader::readValue() {
|
|
|
+ // To preserve the old behaviour we cast size_t to int.
|
|
|
+ if (nodes_.size() > features_.stackLimit_)
|
|
|
+ throwRuntimeError("Exceeded stackLimit in readValue().");
|
|
|
+ Token token;
|
|
|
+ skipCommentTokens(token);
|
|
|
+ bool successful = true;
|
|
|
+
|
|
|
+ if (collectComments_ && !commentsBefore_.empty()) {
|
|
|
+ currentValue().setComment(commentsBefore_, commentBefore);
|
|
|
+ commentsBefore_.clear();
|
|
|
+ }
|
|
|
+
|
|
|
+ switch (token.type_) {
|
|
|
+ case tokenObjectBegin:
|
|
|
+ successful = readObject(token);
|
|
|
+ currentValue().setOffsetLimit(current_ - begin_);
|
|
|
+ break;
|
|
|
+ case tokenArrayBegin:
|
|
|
+ successful = readArray(token);
|
|
|
+ currentValue().setOffsetLimit(current_ - begin_);
|
|
|
+ break;
|
|
|
+ case tokenNumber:
|
|
|
+ successful = decodeNumber(token);
|
|
|
+ break;
|
|
|
+ case tokenString:
|
|
|
+ successful = decodeString(token);
|
|
|
+ break;
|
|
|
+ case tokenTrue: {
|
|
|
+ Value v(true);
|
|
|
+ currentValue().swapPayload(v);
|
|
|
+ currentValue().setOffsetStart(token.start_ - begin_);
|
|
|
+ currentValue().setOffsetLimit(token.end_ - begin_);
|
|
|
+ }
|
|
|
+ break;
|
|
|
+ case tokenFalse: {
|
|
|
+ Value v(false);
|
|
|
+ currentValue().swapPayload(v);
|
|
|
+ currentValue().setOffsetStart(token.start_ - begin_);
|
|
|
+ currentValue().setOffsetLimit(token.end_ - begin_);
|
|
|
+ }
|
|
|
+ break;
|
|
|
+ case tokenNull: {
|
|
|
+ Value v;
|
|
|
+ currentValue().swapPayload(v);
|
|
|
+ currentValue().setOffsetStart(token.start_ - begin_);
|
|
|
+ currentValue().setOffsetLimit(token.end_ - begin_);
|
|
|
+ }
|
|
|
+ break;
|
|
|
+ case tokenNaN: {
|
|
|
+ Value v(std::numeric_limits<double>::quiet_NaN());
|
|
|
+ currentValue().swapPayload(v);
|
|
|
+ currentValue().setOffsetStart(token.start_ - begin_);
|
|
|
+ currentValue().setOffsetLimit(token.end_ - begin_);
|
|
|
+ }
|
|
|
+ break;
|
|
|
+ case tokenPosInf: {
|
|
|
+ Value v(std::numeric_limits<double>::infinity());
|
|
|
+ currentValue().swapPayload(v);
|
|
|
+ currentValue().setOffsetStart(token.start_ - begin_);
|
|
|
+ currentValue().setOffsetLimit(token.end_ - begin_);
|
|
|
+ }
|
|
|
+ break;
|
|
|
+ case tokenNegInf: {
|
|
|
+ Value v(-std::numeric_limits<double>::infinity());
|
|
|
+ currentValue().swapPayload(v);
|
|
|
+ currentValue().setOffsetStart(token.start_ - begin_);
|
|
|
+ currentValue().setOffsetLimit(token.end_ - begin_);
|
|
|
+ }
|
|
|
+ break;
|
|
|
+ case tokenArraySeparator:
|
|
|
+ case tokenObjectEnd:
|
|
|
+ case tokenArrayEnd:
|
|
|
+ if (features_.allowDroppedNullPlaceholders_) {
|
|
|
+ // "Un-read" the current token and mark the current value as a null
|
|
|
+ // token.
|
|
|
+ current_--;
|
|
|
+ Value v;
|
|
|
+ currentValue().swapPayload(v);
|
|
|
+ currentValue().setOffsetStart(current_ - begin_ - 1);
|
|
|
+ currentValue().setOffsetLimit(current_ - begin_);
|
|
|
+ break;
|
|
|
+ } // else, fall through ...
|
|
|
+ default:
|
|
|
+ currentValue().setOffsetStart(token.start_ - begin_);
|
|
|
+ currentValue().setOffsetLimit(token.end_ - begin_);
|
|
|
+ return addError("Syntax error: value, object or array expected.", token);
|
|
|
+ }
|
|
|
+
|
|
|
+ if (collectComments_) {
|
|
|
+ lastValueEnd_ = current_;
|
|
|
+ lastValueHasAComment_ = false;
|
|
|
+ lastValue_ = ¤tValue();
|
|
|
+ }
|
|
|
+
|
|
|
+ return successful;
|
|
|
+ }
|
|
|
+
|
|
|
+ void OurReader::skipCommentTokens(Token &token) {
|
|
|
+ if (features_.allowComments_) {
|
|
|
+ do {
|
|
|
+ readToken(token);
|
|
|
+ } while (token.type_ == tokenComment);
|
|
|
+ } else {
|
|
|
+ readToken(token);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ bool OurReader::readToken(Token &token) {
|
|
|
+ skipSpaces();
|
|
|
+ token.start_ = current_;
|
|
|
+ Char c = getNextChar();
|
|
|
+ bool ok = true;
|
|
|
+ switch (c) {
|
|
|
+ case '{':
|
|
|
+ token.type_ = tokenObjectBegin;
|
|
|
+ break;
|
|
|
+ case '}':
|
|
|
+ token.type_ = tokenObjectEnd;
|
|
|
+ break;
|
|
|
+ case '[':
|
|
|
+ token.type_ = tokenArrayBegin;
|
|
|
+ break;
|
|
|
+ case ']':
|
|
|
+ token.type_ = tokenArrayEnd;
|
|
|
+ break;
|
|
|
+ case '"':
|
|
|
+ token.type_ = tokenString;
|
|
|
+ ok = readString();
|
|
|
+ break;
|
|
|
+ case '\'':
|
|
|
+ if (features_.allowSingleQuotes_) {
|
|
|
+ token.type_ = tokenString;
|
|
|
+ ok = readStringSingleQuote();
|
|
|
+ } else {
|
|
|
+ // If we don't allow single quotes, this is a failure case.
|
|
|
+ ok = false;
|
|
|
+ }
|
|
|
+ break;
|
|
|
+ case '/':
|
|
|
+ token.type_ = tokenComment;
|
|
|
+ ok = readComment();
|
|
|
+ break;
|
|
|
+ case '0':
|
|
|
+ case '1':
|
|
|
+ case '2':
|
|
|
+ case '3':
|
|
|
+ case '4':
|
|
|
+ case '5':
|
|
|
+ case '6':
|
|
|
+ case '7':
|
|
|
+ case '8':
|
|
|
+ case '9':
|
|
|
+ token.type_ = tokenNumber;
|
|
|
+ readNumber(false);
|
|
|
+ break;
|
|
|
+ case '-':
|
|
|
+ if (readNumber(true)) {
|
|
|
+ token.type_ = tokenNumber;
|
|
|
+ } else {
|
|
|
+ token.type_ = tokenNegInf;
|
|
|
+ ok = features_.allowSpecialFloats_ && match("nfinity", 7);
|
|
|
+ }
|
|
|
+ break;
|
|
|
+ case '+':
|
|
|
+ if (readNumber(true)) {
|
|
|
+ token.type_ = tokenNumber;
|
|
|
+ } else {
|
|
|
+ token.type_ = tokenPosInf;
|
|
|
+ ok = features_.allowSpecialFloats_ && match("nfinity", 7);
|
|
|
+ }
|
|
|
+ break;
|
|
|
+ case 't':
|
|
|
+ token.type_ = tokenTrue;
|
|
|
+ ok = match("rue", 3);
|
|
|
+ break;
|
|
|
+ case 'f':
|
|
|
+ token.type_ = tokenFalse;
|
|
|
+ ok = match("alse", 4);
|
|
|
+ break;
|
|
|
+ case 'n':
|
|
|
+ token.type_ = tokenNull;
|
|
|
+ ok = match("ull", 3);
|
|
|
+ break;
|
|
|
+ case 'N':
|
|
|
+ if (features_.allowSpecialFloats_) {
|
|
|
+ token.type_ = tokenNaN;
|
|
|
+ ok = match("aN", 2);
|
|
|
+ } else {
|
|
|
+ ok = false;
|
|
|
+ }
|
|
|
+ break;
|
|
|
+ case 'I':
|
|
|
+ if (features_.allowSpecialFloats_) {
|
|
|
+ token.type_ = tokenPosInf;
|
|
|
+ ok = match("nfinity", 7);
|
|
|
+ } else {
|
|
|
+ ok = false;
|
|
|
+ }
|
|
|
+ break;
|
|
|
+ case ',':
|
|
|
+ token.type_ = tokenArraySeparator;
|
|
|
+ break;
|
|
|
+ case ':':
|
|
|
+ token.type_ = tokenMemberSeparator;
|
|
|
+ break;
|
|
|
+ case 0:
|
|
|
+ token.type_ = tokenEndOfStream;
|
|
|
+ break;
|
|
|
+ default:
|
|
|
+ ok = false;
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ if (!ok)
|
|
|
+ token.type_ = tokenError;
|
|
|
+ token.end_ = current_;
|
|
|
+ return ok;
|
|
|
+ }
|
|
|
+
|
|
|
+ void OurReader::skipSpaces() {
|
|
|
+ while (current_ != end_) {
|
|
|
+ Char c = *current_;
|
|
|
+ if (c == ' ' || c == '\t' || c == '\r' || c == '\n')
|
|
|
+ ++current_;
|
|
|
+ else
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ void OurReader::skipBom(bool skipBom) {
|
|
|
+ // The default behavior is to skip BOM.
|
|
|
+ if (skipBom) {
|
|
|
+ if ((end_ - begin_) >= 3 && strncmp(begin_, "\xEF\xBB\xBF", 3) == 0) {
|
|
|
+ begin_ += 3;
|
|
|
+ current_ = begin_;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ bool OurReader::match(const Char *pattern, int patternLength) {
|
|
|
+ if (end_ - current_ < patternLength)
|
|
|
+ return false;
|
|
|
+ int index = patternLength;
|
|
|
+ while (index--)
|
|
|
+ if (current_[index] != pattern[index])
|
|
|
+ return false;
|
|
|
+ current_ += patternLength;
|
|
|
+ return true;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool OurReader::readComment() {
|
|
|
+ const Location commentBegin = current_ - 1;
|
|
|
+ const Char c = getNextChar();
|
|
|
+ bool successful = false;
|
|
|
+ bool cStyleWithEmbeddedNewline = false;
|
|
|
+
|
|
|
+ const bool isCStyleComment = (c == '*');
|
|
|
+ const bool isCppStyleComment = (c == '/');
|
|
|
+ if (isCStyleComment) {
|
|
|
+ successful = readCStyleComment(&cStyleWithEmbeddedNewline);
|
|
|
+ } else if (isCppStyleComment) {
|
|
|
+ successful = readCppStyleComment();
|
|
|
+ }
|
|
|
+
|
|
|
+ if (!successful)
|
|
|
+ return false;
|
|
|
+
|
|
|
+ if (collectComments_) {
|
|
|
+ CommentPlacement placement = commentBefore;
|
|
|
+
|
|
|
+ if (!lastValueHasAComment_) {
|
|
|
+ if (lastValueEnd_ && !containsNewLine(lastValueEnd_, commentBegin)) {
|
|
|
+ if (isCppStyleComment || !cStyleWithEmbeddedNewline) {
|
|
|
+ placement = commentAfterOnSameLine;
|
|
|
+ lastValueHasAComment_ = true;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ addComment(commentBegin, current_, placement);
|
|
|
+ }
|
|
|
+ return true;
|
|
|
+ }
|
|
|
+
|
|
|
+ String OurReader::normalizeEOL(OurReader::Location begin,
|
|
|
+ OurReader::Location end) {
|
|
|
+ String normalized;
|
|
|
+ normalized.reserve(static_cast<size_t>(end - begin));
|
|
|
+ OurReader::Location current = begin;
|
|
|
+ while (current != end) {
|
|
|
+ char c = *current++;
|
|
|
+ if (c == '\r') {
|
|
|
+ if (current != end && *current == '\n')
|
|
|
+ // convert dos EOL
|
|
|
+ ++current;
|
|
|
+ // convert Mac EOL
|
|
|
+ normalized += '\n';
|
|
|
+ } else {
|
|
|
+ normalized += c;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ return normalized;
|
|
|
+ }
|
|
|
+
|
|
|
+ void OurReader::addComment(Location begin, Location end,
|
|
|
+ CommentPlacement placement) {
|
|
|
+ assert(collectComments_);
|
|
|
+ const String &normalized = normalizeEOL(begin, end);
|
|
|
+ if (placement == commentAfterOnSameLine) {
|
|
|
+ assert(lastValue_ != nullptr);
|
|
|
+ lastValue_->setComment(normalized, placement);
|
|
|
+ } else {
|
|
|
+ commentsBefore_ += normalized;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ bool OurReader::readCStyleComment(bool *containsNewLineResult) {
|
|
|
+ *containsNewLineResult = false;
|
|
|
+
|
|
|
+ while ((current_ + 1) < end_) {
|
|
|
+ Char c = getNextChar();
|
|
|
+ if (c == '*' && *current_ == '/')
|
|
|
+ break;
|
|
|
+ if (c == '\n')
|
|
|
+ *containsNewLineResult = true;
|
|
|
+ }
|
|
|
+
|
|
|
+ return getNextChar() == '/';
|
|
|
+ }
|
|
|
+
|
|
|
+ bool OurReader::readCppStyleComment() {
|
|
|
+ while (current_ != end_) {
|
|
|
+ Char c = getNextChar();
|
|
|
+ if (c == '\n')
|
|
|
+ break;
|
|
|
+ if (c == '\r') {
|
|
|
+ // Consume DOS EOL. It will be normalized in addComment.
|
|
|
+ if (current_ != end_ && *current_ == '\n')
|
|
|
+ getNextChar();
|
|
|
+ // Break on Moc OS 9 EOL.
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ return true;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool OurReader::readNumber(bool checkInf) {
|
|
|
+ Location p = current_;
|
|
|
+ if (checkInf && p != end_ && *p == 'I') {
|
|
|
+ current_ = ++p;
|
|
|
+ return false;
|
|
|
+ }
|
|
|
+ char c = '0'; // stopgap for already consumed character
|
|
|
+ // integral part
|
|
|
+ while (c >= '0' && c <= '9')
|
|
|
+ c = (current_ = p) < end_ ? *p++ : '\0';
|
|
|
+ // fractional part
|
|
|
+ if (c == '.') {
|
|
|
+ c = (current_ = p) < end_ ? *p++ : '\0';
|
|
|
+ while (c >= '0' && c <= '9')
|
|
|
+ c = (current_ = p) < end_ ? *p++ : '\0';
|
|
|
+ }
|
|
|
+ // exponential part
|
|
|
+ if (c == 'e' || c == 'E') {
|
|
|
+ c = (current_ = p) < end_ ? *p++ : '\0';
|
|
|
+ if (c == '+' || c == '-')
|
|
|
+ c = (current_ = p) < end_ ? *p++ : '\0';
|
|
|
+ while (c >= '0' && c <= '9')
|
|
|
+ c = (current_ = p) < end_ ? *p++ : '\0';
|
|
|
+ }
|
|
|
+ return true;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool OurReader::readString() {
|
|
|
+ Char c = 0;
|
|
|
+ while (current_ != end_) {
|
|
|
+ c = getNextChar();
|
|
|
+ if (c == '\\')
|
|
|
+ getNextChar();
|
|
|
+ else if (c == '"')
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ return c == '"';
|
|
|
+ }
|
|
|
+
|
|
|
+ bool OurReader::readStringSingleQuote() {
|
|
|
+ Char c = 0;
|
|
|
+ while (current_ != end_) {
|
|
|
+ c = getNextChar();
|
|
|
+ if (c == '\\')
|
|
|
+ getNextChar();
|
|
|
+ else if (c == '\'')
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ return c == '\'';
|
|
|
+ }
|
|
|
+
|
|
|
+ bool OurReader::readObject(Token &token) {
|
|
|
+ Token tokenName;
|
|
|
+ String name;
|
|
|
+ Value init(objectValue);
|
|
|
+ currentValue().swapPayload(init);
|
|
|
+ currentValue().setOffsetStart(token.start_ - begin_);
|
|
|
+ while (readToken(tokenName)) {
|
|
|
+ bool initialTokenOk = true;
|
|
|
+ while (tokenName.type_ == tokenComment && initialTokenOk)
|
|
|
+ initialTokenOk = readToken(tokenName);
|
|
|
+ if (!initialTokenOk)
|
|
|
+ break;
|
|
|
+ if (tokenName.type_ == tokenObjectEnd &&
|
|
|
+ (name.empty() ||
|
|
|
+ features_.allowTrailingCommas_)) // empty object or trailing comma
|
|
|
+ return true;
|
|
|
+ name.clear();
|
|
|
+ if (tokenName.type_ == tokenString) {
|
|
|
+ if (!decodeString(tokenName, name))
|
|
|
+ return recoverFromError(tokenObjectEnd);
|
|
|
+ } else if (tokenName.type_ == tokenNumber && features_.allowNumericKeys_) {
|
|
|
+ Value numberName;
|
|
|
+ if (!decodeNumber(tokenName, numberName))
|
|
|
+ return recoverFromError(tokenObjectEnd);
|
|
|
+ name = numberName.asString();
|
|
|
+ } else {
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ if (name.length() >= (1U << 30))
|
|
|
+ throwRuntimeError("keylength >= 2^30");
|
|
|
+ if (features_.rejectDupKeys_ && currentValue().isMember(name)) {
|
|
|
+ String msg = "Duplicate key: '" + name + "'";
|
|
|
+ return addErrorAndRecover(msg, tokenName, tokenObjectEnd);
|
|
|
+ }
|
|
|
+
|
|
|
+ Token colon;
|
|
|
+ if (!readToken(colon) || colon.type_ != tokenMemberSeparator) {
|
|
|
+ return addErrorAndRecover("Missing ':' after object member name", colon,
|
|
|
+ tokenObjectEnd);
|
|
|
+ }
|
|
|
+ Value &value = currentValue()[name];
|
|
|
+ nodes_.push(&value);
|
|
|
+ bool ok = readValue();
|
|
|
+ nodes_.pop();
|
|
|
+ if (!ok) // error already put
|
|
|
+ return recoverFromError(tokenObjectEnd);
|
|
|
+
|
|
|
+ Token comma;
|
|
|
+ if (!readToken(comma) ||
|
|
|
+ (comma.type_ != tokenObjectEnd && comma.type_ != tokenArraySeparator &&
|
|
|
+ comma.type_ != tokenComment)) {
|
|
|
+ return addErrorAndRecover("Missing ',' or '}' in object declaration",
|
|
|
+ comma, tokenObjectEnd);
|
|
|
+ }
|
|
|
+ bool finalizeTokenOk = true;
|
|
|
+ while (comma.type_ == tokenComment && finalizeTokenOk)
|
|
|
+ finalizeTokenOk = readToken(comma);
|
|
|
+ if (comma.type_ == tokenObjectEnd)
|
|
|
+ return true;
|
|
|
+ }
|
|
|
+ return addErrorAndRecover("Missing '}' or object member name", tokenName,
|
|
|
+ tokenObjectEnd);
|
|
|
+ }
|
|
|
+
|
|
|
+ bool OurReader::readArray(Token &token) {
|
|
|
+ Value init(arrayValue);
|
|
|
+ currentValue().swapPayload(init);
|
|
|
+ currentValue().setOffsetStart(token.start_ - begin_);
|
|
|
+ int index = 0;
|
|
|
+ for (;;) {
|
|
|
+ skipSpaces();
|
|
|
+ if (current_ != end_ && *current_ == ']' &&
|
|
|
+ (index == 0 ||
|
|
|
+ (features_.allowTrailingCommas_ &&
|
|
|
+ !features_.allowDroppedNullPlaceholders_))) // empty array or trailing
|
|
|
+ // comma
|
|
|
+ {
|
|
|
+ Token endArray;
|
|
|
+ readToken(endArray);
|
|
|
+ return true;
|
|
|
+ }
|
|
|
+ Value &value = currentValue()[index++];
|
|
|
+ nodes_.push(&value);
|
|
|
+ bool ok = readValue();
|
|
|
+ nodes_.pop();
|
|
|
+ if (!ok) // error already put
|
|
|
+ return recoverFromError(tokenArrayEnd);
|
|
|
+
|
|
|
+ Token currentToken;
|
|
|
+ // Accept Comment after last item in the array.
|
|
|
+ ok = readToken(currentToken);
|
|
|
+ while (currentToken.type_ == tokenComment && ok) {
|
|
|
+ ok = readToken(currentToken);
|
|
|
+ }
|
|
|
+ bool badTokenType = (currentToken.type_ != tokenArraySeparator &&
|
|
|
+ currentToken.type_ != tokenArrayEnd);
|
|
|
+ if (!ok || badTokenType) {
|
|
|
+ return addErrorAndRecover("Missing ',' or ']' in array declaration",
|
|
|
+ currentToken, tokenArrayEnd);
|
|
|
+ }
|
|
|
+ if (currentToken.type_ == tokenArrayEnd)
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ return true;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool OurReader::decodeNumber(Token &token) {
|
|
|
+ Value decoded;
|
|
|
+ if (!decodeNumber(token, decoded))
|
|
|
+ return false;
|
|
|
+ currentValue().swapPayload(decoded);
|
|
|
+ currentValue().setOffsetStart(token.start_ - begin_);
|
|
|
+ currentValue().setOffsetLimit(token.end_ - begin_);
|
|
|
+ return true;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool OurReader::decodeNumber(Token &token, Value &decoded) {
|
|
|
+ // Attempts to parse the number as an integer. If the number is
|
|
|
+ // larger than the maximum supported value of an integer then
|
|
|
+ // we decode the number as a double.
|
|
|
+ Location current = token.start_;
|
|
|
+ const bool isNegative = *current == '-';
|
|
|
+ if (isNegative) {
|
|
|
+ ++current;
|
|
|
+ }
|
|
|
+
|
|
|
+ // We assume we can represent the largest and smallest integer types as
|
|
|
+ // unsigned integers with separate sign. This is only true if they can fit
|
|
|
+ // into an unsigned integer.
|
|
|
+ static_assert(Value::maxLargestInt <= Value::maxLargestUInt,
|
|
|
+ "Int must be smaller than UInt");
|
|
|
+
|
|
|
+ // We need to convert minLargestInt into a positive number. The easiest way
|
|
|
+ // to do this conversion is to assume our "threshold" value of minLargestInt
|
|
|
+ // divided by 10 can fit in maxLargestInt when absolute valued. This should
|
|
|
+ // be a safe assumption.
|
|
|
+ static_assert(Value::minLargestInt <= -Value::maxLargestInt,
|
|
|
+ "The absolute value of minLargestInt must be greater than or "
|
|
|
+ "equal to maxLargestInt");
|
|
|
+ static_assert(Value::minLargestInt / 10 >= -Value::maxLargestInt,
|
|
|
+ "The absolute value of minLargestInt must be only 1 magnitude "
|
|
|
+ "larger than maxLargest Int");
|
|
|
+
|
|
|
+ static constexpr Value::LargestUInt positive_threshold =
|
|
|
+ Value::maxLargestUInt / 10;
|
|
|
+ static constexpr Value::UInt positive_last_digit = Value::maxLargestUInt % 10;
|
|
|
+
|
|
|
+ // For the negative values, we have to be more careful. Since typically
|
|
|
+ // -Value::minLargestInt will cause an overflow, we first divide by 10 and
|
|
|
+ // then take the inverse. This assumes that minLargestInt is only a single
|
|
|
+ // power of 10 different in magnitude, which we check above. For the last
|
|
|
+ // digit, we take the modulus before negating for the same reason.
|
|
|
+ static constexpr auto negative_threshold =
|
|
|
+ Value::LargestUInt(-(Value::minLargestInt / 10));
|
|
|
+ static constexpr auto negative_last_digit =
|
|
|
+ Value::UInt(-(Value::minLargestInt % 10));
|
|
|
+
|
|
|
+ const Value::LargestUInt threshold =
|
|
|
+ isNegative ? negative_threshold : positive_threshold;
|
|
|
+ const Value::UInt max_last_digit =
|
|
|
+ isNegative ? negative_last_digit : positive_last_digit;
|
|
|
+
|
|
|
+ Value::LargestUInt value = 0;
|
|
|
+ while (current < token.end_) {
|
|
|
+ Char c = *current++;
|
|
|
+ if (c < '0' || c > '9')
|
|
|
+ return decodeDouble(token, decoded);
|
|
|
+
|
|
|
+ const auto digit(static_cast<Value::UInt>(c - '0'));
|
|
|
+ if (value >= threshold) {
|
|
|
+ // We've hit or exceeded the max value divided by 10 (rounded down). If
|
|
|
+ // a) we've only just touched the limit, meaing value == threshold,
|
|
|
+ // b) this is the last digit, or
|
|
|
+ // c) it's small enough to fit in that rounding delta, we're okay.
|
|
|
+ // Otherwise treat this number as a double to avoid overflow.
|
|
|
+ if (value > threshold || current != token.end_ ||
|
|
|
+ digit > max_last_digit) {
|
|
|
+ return decodeDouble(token, decoded);
|
|
|
+ }
|
|
|
+ }
|
|
|
+ value = value * 10 + digit;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (isNegative) {
|
|
|
+ // We use the same magnitude assumption here, just in case.
|
|
|
+ const auto last_digit = static_cast<Value::UInt>(value % 10);
|
|
|
+ decoded = -Value::LargestInt(value / 10) * 10 - last_digit;
|
|
|
+ } else if (value <= Value::LargestUInt(Value::maxLargestInt)) {
|
|
|
+ decoded = Value::LargestInt(value);
|
|
|
+ } else {
|
|
|
+ decoded = value;
|
|
|
+ }
|
|
|
+
|
|
|
+ return true;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool OurReader::decodeDouble(Token &token) {
|
|
|
+ Value decoded;
|
|
|
+ if (!decodeDouble(token, decoded))
|
|
|
+ return false;
|
|
|
+ currentValue().swapPayload(decoded);
|
|
|
+ currentValue().setOffsetStart(token.start_ - begin_);
|
|
|
+ currentValue().setOffsetLimit(token.end_ - begin_);
|
|
|
+ return true;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool OurReader::decodeDouble(Token &token, Value &decoded) {
|
|
|
+ double value = 0;
|
|
|
+ const String buffer(token.start_, token.end_);
|
|
|
+ IStringStream is(buffer);
|
|
|
+ if (!(is >> value)) {
|
|
|
+ return addError(
|
|
|
+ "'" + String(token.start_, token.end_) + "' is not a number.", token);
|
|
|
+ }
|
|
|
+ decoded = value;
|
|
|
+ return true;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool OurReader::decodeString(Token &token) {
|
|
|
+ String decoded_string;
|
|
|
+ if (!decodeString(token, decoded_string))
|
|
|
+ return false;
|
|
|
+ Value decoded(decoded_string);
|
|
|
+ currentValue().swapPayload(decoded);
|
|
|
+ currentValue().setOffsetStart(token.start_ - begin_);
|
|
|
+ currentValue().setOffsetLimit(token.end_ - begin_);
|
|
|
+ return true;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool OurReader::decodeString(Token &token, String &decoded) {
|
|
|
+ decoded.reserve(static_cast<size_t>(token.end_ - token.start_ - 2));
|
|
|
+ Location current = token.start_ + 1; // skip '"'
|
|
|
+ Location end = token.end_ - 1; // do not include '"'
|
|
|
+ while (current != end) {
|
|
|
+ Char c = *current++;
|
|
|
+ if (c == '"')
|
|
|
+ break;
|
|
|
+ if (c == '\\') {
|
|
|
+ if (current == end)
|
|
|
+ return addError("Empty escape sequence in string", token, current);
|
|
|
+ Char escape = *current++;
|
|
|
+ switch (escape) {
|
|
|
+ case '"':
|
|
|
+ decoded += '"';
|
|
|
+ break;
|
|
|
+ case '/':
|
|
|
+ decoded += '/';
|
|
|
+ break;
|
|
|
+ case '\\':
|
|
|
+ decoded += '\\';
|
|
|
+ break;
|
|
|
+ case 'b':
|
|
|
+ decoded += '\b';
|
|
|
+ break;
|
|
|
+ case 'f':
|
|
|
+ decoded += '\f';
|
|
|
+ break;
|
|
|
+ case 'n':
|
|
|
+ decoded += '\n';
|
|
|
+ break;
|
|
|
+ case 'r':
|
|
|
+ decoded += '\r';
|
|
|
+ break;
|
|
|
+ case 't':
|
|
|
+ decoded += '\t';
|
|
|
+ break;
|
|
|
+ case 'u': {
|
|
|
+ unsigned int unicode;
|
|
|
+ if (!decodeUnicodeCodePoint(token, current, end, unicode))
|
|
|
+ return false;
|
|
|
+ decoded += codePointToUTF8(unicode);
|
|
|
+ }
|
|
|
+ break;
|
|
|
+ default:
|
|
|
+ return addError("Bad escape sequence in string", token, current);
|
|
|
+ }
|
|
|
+ } else {
|
|
|
+ decoded += c;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ return true;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool OurReader::decodeUnicodeCodePoint(Token &token, Location ¤t,
|
|
|
+ Location end, unsigned int &unicode) {
|
|
|
+
|
|
|
+ if (!decodeUnicodeEscapeSequence(token, current, end, unicode))
|
|
|
+ return false;
|
|
|
+ if (unicode >= 0xD800 && unicode <= 0xDBFF) {
|
|
|
+ // surrogate pairs
|
|
|
+ if (end - current < 6)
|
|
|
+ return addError(
|
|
|
+ "additional six characters expected to parse unicode surrogate pair.",
|
|
|
+ token, current);
|
|
|
+ if (*(current++) == '\\' && *(current++) == 'u') {
|
|
|
+ unsigned int surrogatePair;
|
|
|
+ if (decodeUnicodeEscapeSequence(token, current, end, surrogatePair)) {
|
|
|
+ unicode = 0x10000 + ((unicode & 0x3FF) << 10) + (surrogatePair & 0x3FF);
|
|
|
+ } else
|
|
|
+ return false;
|
|
|
+ } else
|
|
|
+ return addError("expecting another \\u token to begin the second half of "
|
|
|
+ "a unicode surrogate pair",
|
|
|
+ token, current);
|
|
|
+ }
|
|
|
+ return true;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool OurReader::decodeUnicodeEscapeSequence(Token &token, Location ¤t,
|
|
|
+ Location end,
|
|
|
+ unsigned int &ret_unicode) {
|
|
|
+ if (end - current < 4)
|
|
|
+ return addError(
|
|
|
+ "Bad unicode escape sequence in string: four digits expected.", token,
|
|
|
+ current);
|
|
|
+ int unicode = 0;
|
|
|
+ for (int index = 0; index < 4; ++index) {
|
|
|
+ Char c = *current++;
|
|
|
+ unicode *= 16;
|
|
|
+ if (c >= '0' && c <= '9')
|
|
|
+ unicode += c - '0';
|
|
|
+ else if (c >= 'a' && c <= 'f')
|
|
|
+ unicode += c - 'a' + 10;
|
|
|
+ else if (c >= 'A' && c <= 'F')
|
|
|
+ unicode += c - 'A' + 10;
|
|
|
+ else
|
|
|
+ return addError(
|
|
|
+ "Bad unicode escape sequence in string: hexadecimal digit expected.",
|
|
|
+ token, current);
|
|
|
+ }
|
|
|
+ ret_unicode = static_cast<unsigned int>(unicode);
|
|
|
+ return true;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool OurReader::addError(const String &message, Token &token, Location extra) {
|
|
|
+ ErrorInfo info;
|
|
|
+ info.token_ = token;
|
|
|
+ info.message_ = message;
|
|
|
+ info.extra_ = extra;
|
|
|
+ errors_.push_back(info);
|
|
|
+ return false;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool OurReader::recoverFromError(TokenType skipUntilToken) {
|
|
|
+ size_t errorCount = errors_.size();
|
|
|
+ Token skip;
|
|
|
+ for (;;) {
|
|
|
+ if (!readToken(skip))
|
|
|
+ errors_.resize(errorCount); // discard errors caused by recovery
|
|
|
+ if (skip.type_ == skipUntilToken || skip.type_ == tokenEndOfStream)
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ errors_.resize(errorCount);
|
|
|
+ return false;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool OurReader::addErrorAndRecover(const String &message, Token &token,
|
|
|
+ TokenType skipUntilToken) {
|
|
|
+ addError(message, token);
|
|
|
+ return recoverFromError(skipUntilToken);
|
|
|
+ }
|
|
|
+
|
|
|
+ Value &OurReader::currentValue() { return *(nodes_.top()); }
|
|
|
+
|
|
|
+ OurReader::Char OurReader::getNextChar() {
|
|
|
+ if (current_ == end_)
|
|
|
+ return 0;
|
|
|
+ return *current_++;
|
|
|
+ }
|
|
|
+
|
|
|
+ void OurReader::getLocationLineAndColumn(Location location, int &line,
|
|
|
+ int &column) const {
|
|
|
+ Location current = begin_;
|
|
|
+ Location lastLineStart = current;
|
|
|
+ line = 0;
|
|
|
+ while (current < location && current != end_) {
|
|
|
+ Char c = *current++;
|
|
|
+ if (c == '\r') {
|
|
|
+ if (*current == '\n')
|
|
|
+ ++current;
|
|
|
+ lastLineStart = current;
|
|
|
+ ++line;
|
|
|
+ } else if (c == '\n') {
|
|
|
+ lastLineStart = current;
|
|
|
+ ++line;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ // column & line start at 1
|
|
|
+ column = int(location - lastLineStart) + 1;
|
|
|
+ ++line;
|
|
|
+ }
|
|
|
+
|
|
|
+ String OurReader::getLocationLineAndColumn(Location location) const {
|
|
|
+ int line, column;
|
|
|
+ getLocationLineAndColumn(location, line, column);
|
|
|
+ char buffer[18 + 16 + 16 + 1];
|
|
|
+ jsoncpp_snprintf(buffer, sizeof(buffer), "Line %d, Column %d", line, column);
|
|
|
+ return buffer;
|
|
|
+ }
|
|
|
+
|
|
|
+ String OurReader::getFormattedErrorMessages() const {
|
|
|
+ String formattedMessage;
|
|
|
+ for (const auto &error : errors_) {
|
|
|
+ formattedMessage +=
|
|
|
+ "* " + getLocationLineAndColumn(error.token_.start_) + "\n";
|
|
|
+ formattedMessage += " " + error.message_ + "\n";
|
|
|
+ if (error.extra_)
|
|
|
+ formattedMessage +=
|
|
|
+ "See " + getLocationLineAndColumn(error.extra_) + " for detail.\n";
|
|
|
+ }
|
|
|
+ return formattedMessage;
|
|
|
+ }
|
|
|
+
|
|
|
+ std::vector<OurReader::StructuredError> OurReader::getStructuredErrors() const {
|
|
|
+ std::vector<OurReader::StructuredError> allErrors;
|
|
|
+ for (const auto &error : errors_) {
|
|
|
+ OurReader::StructuredError structured;
|
|
|
+ structured.offset_start = error.token_.start_ - begin_;
|
|
|
+ structured.offset_limit = error.token_.end_ - begin_;
|
|
|
+ structured.message = error.message_;
|
|
|
+ allErrors.push_back(structured);
|
|
|
+ }
|
|
|
+ return allErrors;
|
|
|
+ }
|
|
|
+
|
|
|
+ class OurCharReader : public CharReader {
|
|
|
+ bool const collectComments_;
|
|
|
+ OurReader reader_;
|
|
|
+
|
|
|
+ public:
|
|
|
+ OurCharReader(bool collectComments, OurFeatures const &features)
|
|
|
+ : collectComments_(collectComments), reader_(features) {}
|
|
|
+
|
|
|
+ bool parse(char const *beginDoc, char const *endDoc, Value *root,
|
|
|
+ String *errs) override {
|
|
|
+ bool ok = reader_.parse(beginDoc, endDoc, *root, collectComments_);
|
|
|
+ if (errs) {
|
|
|
+ *errs = reader_.getFormattedErrorMessages();
|
|
|
+ }
|
|
|
+ return ok;
|
|
|
+ }
|
|
|
+ };
|
|
|
+
|
|
|
+ CharReaderBuilder::CharReaderBuilder() { setDefaults(&settings_); }
|
|
|
+
|
|
|
+ CharReaderBuilder::~CharReaderBuilder() = default;
|
|
|
+
|
|
|
+ CharReader *CharReaderBuilder::newCharReader() const {
|
|
|
+ bool collectComments = settings_["collectComments"].asBool();
|
|
|
+ OurFeatures features = OurFeatures::all();
|
|
|
+ features.allowComments_ = settings_["allowComments"].asBool();
|
|
|
+ features.allowTrailingCommas_ = settings_["allowTrailingCommas"].asBool();
|
|
|
+ features.strictRoot_ = settings_["strictRoot"].asBool();
|
|
|
+ features.allowDroppedNullPlaceholders_ =
|
|
|
+ settings_["allowDroppedNullPlaceholders"].asBool();
|
|
|
+ features.allowNumericKeys_ = settings_["allowNumericKeys"].asBool();
|
|
|
+ features.allowSingleQuotes_ = settings_["allowSingleQuotes"].asBool();
|
|
|
+
|
|
|
+ // Stack limit is always a size_t, so we get this as an unsigned int
|
|
|
+ // regardless of it we have 64-bit integer support enabled.
|
|
|
+ features.stackLimit_ = static_cast<size_t>(settings_["stackLimit"].asUInt());
|
|
|
+ features.failIfExtra_ = settings_["failIfExtra"].asBool();
|
|
|
+ features.rejectDupKeys_ = settings_["rejectDupKeys"].asBool();
|
|
|
+ features.allowSpecialFloats_ = settings_["allowSpecialFloats"].asBool();
|
|
|
+ features.skipBom_ = settings_["skipBom"].asBool();
|
|
|
+ return new OurCharReader(collectComments, features);
|
|
|
+ }
|
|
|
+
|
|
|
+ bool CharReaderBuilder::validate(Json::Value *invalid) const {
|
|
|
+ static const auto &valid_keys = *new std::set<String>{
|
|
|
+ "collectComments",
|
|
|
+ "allowComments",
|
|
|
+ "allowTrailingCommas",
|
|
|
+ "strictRoot",
|
|
|
+ "allowDroppedNullPlaceholders",
|
|
|
+ "allowNumericKeys",
|
|
|
+ "allowSingleQuotes",
|
|
|
+ "stackLimit",
|
|
|
+ "failIfExtra",
|
|
|
+ "rejectDupKeys",
|
|
|
+ "allowSpecialFloats",
|
|
|
+ "skipBom",
|
|
|
+ };
|
|
|
+ for (auto si = settings_.begin(); si != settings_.end(); ++si) {
|
|
|
+ auto key = si.name();
|
|
|
+ if (valid_keys.count(key))
|
|
|
+ continue;
|
|
|
+ if (invalid)
|
|
|
+ (*invalid)[std::move(key)] = *si;
|
|
|
+ else
|
|
|
+ return false;
|
|
|
+ }
|
|
|
+ return invalid ? invalid->empty() : true;
|
|
|
+ }
|
|
|
+
|
|
|
+ Value &CharReaderBuilder::operator[](const String &key) {
|
|
|
+ return settings_[key];
|
|
|
+ }
|
|
|
+
|
|
|
+// static
|
|
|
+ void CharReaderBuilder::strictMode(Json::Value *settings) {
|
|
|
+ //! [CharReaderBuilderStrictMode]
|
|
|
+ (*settings)["allowComments"] = false;
|
|
|
+ (*settings)["allowTrailingCommas"] = false;
|
|
|
+ (*settings)["strictRoot"] = true;
|
|
|
+ (*settings)["allowDroppedNullPlaceholders"] = false;
|
|
|
+ (*settings)["allowNumericKeys"] = false;
|
|
|
+ (*settings)["allowSingleQuotes"] = false;
|
|
|
+ (*settings)["stackLimit"] = 1000;
|
|
|
+ (*settings)["failIfExtra"] = true;
|
|
|
+ (*settings)["rejectDupKeys"] = true;
|
|
|
+ (*settings)["allowSpecialFloats"] = false;
|
|
|
+ (*settings)["skipBom"] = true;
|
|
|
+ //! [CharReaderBuilderStrictMode]
|
|
|
+ }
|
|
|
+
|
|
|
+// static
|
|
|
+ void CharReaderBuilder::setDefaults(Json::Value *settings) {
|
|
|
+ //! [CharReaderBuilderDefaults]
|
|
|
+ (*settings)["collectComments"] = true;
|
|
|
+ (*settings)["allowComments"] = true;
|
|
|
+ (*settings)["allowTrailingCommas"] = true;
|
|
|
+ (*settings)["strictRoot"] = false;
|
|
|
+ (*settings)["allowDroppedNullPlaceholders"] = false;
|
|
|
+ (*settings)["allowNumericKeys"] = false;
|
|
|
+ (*settings)["allowSingleQuotes"] = false;
|
|
|
+ (*settings)["stackLimit"] = 1000;
|
|
|
+ (*settings)["failIfExtra"] = false;
|
|
|
+ (*settings)["rejectDupKeys"] = false;
|
|
|
+ (*settings)["allowSpecialFloats"] = false;
|
|
|
+ (*settings)["skipBom"] = true;
|
|
|
+ //! [CharReaderBuilderDefaults]
|
|
|
+ }
|
|
|
+
|
|
|
+//////////////////////////////////
|
|
|
+// global functions
|
|
|
+
|
|
|
+ bool parseFromStream(CharReader::Factory const &fact, IStream &sin, Value *root,
|
|
|
+ String *errs) {
|
|
|
+ OStringStream ssin;
|
|
|
+ ssin << sin.rdbuf();
|
|
|
+ String doc = ssin.str();
|
|
|
+ char const *begin = doc.data();
|
|
|
+ char const *end = begin + doc.size();
|
|
|
+ // Note that we do not actually need a null-terminator.
|
|
|
+ CharReaderPtr const reader(fact.newCharReader());
|
|
|
+ return reader->parse(begin, end, root, errs);
|
|
|
+ }
|
|
|
+
|
|
|
+ IStream &operator>>(IStream &sin, Value &root) {
|
|
|
+ CharReaderBuilder b;
|
|
|
+ String errs;
|
|
|
+ bool ok = parseFromStream(b, sin, &root, &errs);
|
|
|
+ if (!ok) {
|
|
|
+ throwRuntimeError(errs);
|
|
|
+ }
|
|
|
+ return sin;
|
|
|
+ }
|
|
|
+
|
|
|
+} // namespace Json
|
|
|
+
|
|
|
+// //////////////////////////////////////////////////////////////////////
|
|
|
+// End of content of file: src/lib_json/json_reader.cpp
|
|
|
+// //////////////////////////////////////////////////////////////////////
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+// //////////////////////////////////////////////////////////////////////
|
|
|
+// Beginning of content of file: src/lib_json/json_valueiterator.inl
|
|
|
+// //////////////////////////////////////////////////////////////////////
|
|
|
+
|
|
|
+// Copyright 2007-2010 Baptiste Lepilleur and The JsonCpp Authors
|
|
|
+// Distributed under MIT license, or public domain if desired and
|
|
|
+// recognized in your jurisdiction.
|
|
|
+// See file LICENSE for detail or copy at http://jsoncpp.sourceforge.net/LICENSE
|
|
|
+
|
|
|
+// included by json_value.cpp
|
|
|
+
|
|
|
+namespace Json {
|
|
|
+
|
|
|
+// //////////////////////////////////////////////////////////////////
|
|
|
+// //////////////////////////////////////////////////////////////////
|
|
|
+// //////////////////////////////////////////////////////////////////
|
|
|
+// class ValueIteratorBase
|
|
|
+// //////////////////////////////////////////////////////////////////
|
|
|
+// //////////////////////////////////////////////////////////////////
|
|
|
+// //////////////////////////////////////////////////////////////////
|
|
|
+
|
|
|
+ ValueIteratorBase::ValueIteratorBase() : current_() {}
|
|
|
+
|
|
|
+ ValueIteratorBase::ValueIteratorBase(
|
|
|
+ const Value::ObjectValues::iterator ¤t)
|
|
|
+ : current_(current), isNull_(false) {}
|
|
|
+
|
|
|
+ Value &ValueIteratorBase::deref() { return current_->second; }
|
|
|
+
|
|
|
+ const Value &ValueIteratorBase::deref() const { return current_->second; }
|
|
|
+
|
|
|
+ void ValueIteratorBase::increment() { ++current_; }
|
|
|
+
|
|
|
+ void ValueIteratorBase::decrement() { --current_; }
|
|
|
+
|
|
|
+ ValueIteratorBase::difference_type
|
|
|
+ ValueIteratorBase::computeDistance(const SelfType &other) const {
|
|
|
+ // Iterator for null value are initialized using the default
|
|
|
+ // constructor, which initialize current_ to the default
|
|
|
+ // std::map::iterator. As begin() and end() are two instance
|
|
|
+ // of the default std::map::iterator, they can not be compared.
|
|
|
+ // To allow this, we handle this comparison specifically.
|
|
|
+ if (isNull_ && other.isNull_) {
|
|
|
+ return 0;
|
|
|
+ }
|
|
|
+
|
|
|
+ // Usage of std::distance is not portable (does not compile with Sun Studio 12
|
|
|
+ // RogueWave STL,
|
|
|
+ // which is the one used by default).
|
|
|
+ // Using a portable hand-made version for non random iterator instead:
|
|
|
+ // return difference_type( std::distance( current_, other.current_ ) );
|
|
|
+ difference_type myDistance = 0;
|
|
|
+ for (Value::ObjectValues::iterator it = current_; it != other.current_;
|
|
|
+ ++it) {
|
|
|
+ ++myDistance;
|
|
|
+ }
|
|
|
+ return myDistance;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool ValueIteratorBase::isEqual(const SelfType &other) const {
|
|
|
+ if (isNull_) {
|
|
|
+ return other.isNull_;
|
|
|
+ }
|
|
|
+ return current_ == other.current_;
|
|
|
+ }
|
|
|
+
|
|
|
+ void ValueIteratorBase::copy(const SelfType &other) {
|
|
|
+ current_ = other.current_;
|
|
|
+ isNull_ = other.isNull_;
|
|
|
+ }
|
|
|
+
|
|
|
+ Value ValueIteratorBase::key() const {
|
|
|
+ const Value::CZString czstring = (*current_).first;
|
|
|
+ if (czstring.data()) {
|
|
|
+ if (czstring.isStaticString())
|
|
|
+ return Value(StaticString(czstring.data()));
|
|
|
+ return Value(czstring.data(), czstring.data() + czstring.length());
|
|
|
+ }
|
|
|
+ return Value(czstring.index());
|
|
|
+ }
|
|
|
+
|
|
|
+ UInt ValueIteratorBase::index() const {
|
|
|
+ const Value::CZString czstring = (*current_).first;
|
|
|
+ if (!czstring.data())
|
|
|
+ return czstring.index();
|
|
|
+ return Value::UInt(-1);
|
|
|
+ }
|
|
|
+
|
|
|
+ String ValueIteratorBase::name() const {
|
|
|
+ char const *keey;
|
|
|
+ char const *end;
|
|
|
+ keey = memberName(&end);
|
|
|
+ if (!keey)
|
|
|
+ return String();
|
|
|
+ return String(keey, end);
|
|
|
+ }
|
|
|
+
|
|
|
+ char const *ValueIteratorBase::memberName() const {
|
|
|
+ const char *cname = (*current_).first.data();
|
|
|
+ return cname ? cname : "";
|
|
|
+ }
|
|
|
+
|
|
|
+ char const *ValueIteratorBase::memberName(char const **end) const {
|
|
|
+ const char *cname = (*current_).first.data();
|
|
|
+ if (!cname) {
|
|
|
+ *end = nullptr;
|
|
|
+ return nullptr;
|
|
|
+ }
|
|
|
+ *end = cname + (*current_).first.length();
|
|
|
+ return cname;
|
|
|
+ }
|
|
|
+
|
|
|
+// //////////////////////////////////////////////////////////////////
|
|
|
+// //////////////////////////////////////////////////////////////////
|
|
|
+// //////////////////////////////////////////////////////////////////
|
|
|
+// class ValueConstIterator
|
|
|
+// //////////////////////////////////////////////////////////////////
|
|
|
+// //////////////////////////////////////////////////////////////////
|
|
|
+// //////////////////////////////////////////////////////////////////
|
|
|
+
|
|
|
+ ValueConstIterator::ValueConstIterator() = default;
|
|
|
+
|
|
|
+ ValueConstIterator::ValueConstIterator(
|
|
|
+ const Value::ObjectValues::iterator ¤t)
|
|
|
+ : ValueIteratorBase(current) {}
|
|
|
+
|
|
|
+ ValueConstIterator::ValueConstIterator(ValueIterator const &other)
|
|
|
+ : ValueIteratorBase(other) {}
|
|
|
+
|
|
|
+ ValueConstIterator &ValueConstIterator::
|
|
|
+ operator=(const ValueIteratorBase &other) {
|
|
|
+ copy(other);
|
|
|
+ return *this;
|
|
|
+ }
|
|
|
+
|
|
|
+// //////////////////////////////////////////////////////////////////
|
|
|
+// //////////////////////////////////////////////////////////////////
|
|
|
+// //////////////////////////////////////////////////////////////////
|
|
|
+// class ValueIterator
|
|
|
+// //////////////////////////////////////////////////////////////////
|
|
|
+// //////////////////////////////////////////////////////////////////
|
|
|
+// //////////////////////////////////////////////////////////////////
|
|
|
+
|
|
|
+ ValueIterator::ValueIterator() = default;
|
|
|
+
|
|
|
+ ValueIterator::ValueIterator(const Value::ObjectValues::iterator ¤t)
|
|
|
+ : ValueIteratorBase(current) {}
|
|
|
+
|
|
|
+ ValueIterator::ValueIterator(const ValueConstIterator &other)
|
|
|
+ : ValueIteratorBase(other) {
|
|
|
+ throwRuntimeError("ConstIterator to Iterator should never be allowed.");
|
|
|
+ }
|
|
|
+
|
|
|
+ ValueIterator::ValueIterator(const ValueIterator &other) = default;
|
|
|
+
|
|
|
+ ValueIterator &ValueIterator::operator=(const SelfType &other) {
|
|
|
+ copy(other);
|
|
|
+ return *this;
|
|
|
+ }
|
|
|
+
|
|
|
+} // namespace Json
|
|
|
+
|
|
|
+// //////////////////////////////////////////////////////////////////////
|
|
|
+// End of content of file: src/lib_json/json_valueiterator.inl
|
|
|
+// //////////////////////////////////////////////////////////////////////
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+// //////////////////////////////////////////////////////////////////////
|
|
|
+// Beginning of content of file: src/lib_json/json_value.cpp
|
|
|
+// //////////////////////////////////////////////////////////////////////
|
|
|
+
|
|
|
+// Copyright 2011 Baptiste Lepilleur and The JsonCpp Authors
|
|
|
+// Distributed under MIT license, or public domain if desired and
|
|
|
+// recognized in your jurisdiction.
|
|
|
+// See file LICENSE for detail or copy at http://jsoncpp.sourceforge.net/LICENSE
|
|
|
+
|
|
|
+#if !defined(JSON_IS_AMALGAMATION)
|
|
|
+ #include <json/assertions.h>
|
|
|
+#include <json/value.h>
|
|
|
+#include <json/writer.h>
|
|
|
+#endif // if !defined(JSON_IS_AMALGAMATION)
|
|
|
+
|
|
|
+#include <algorithm>
|
|
|
+#include <cassert>
|
|
|
+#include <cmath>
|
|
|
+#include <cstddef>
|
|
|
+#include <cstring>
|
|
|
+#include <iostream>
|
|
|
+#include <sstream>
|
|
|
+#include <utility>
|
|
|
+
|
|
|
+// Provide implementation equivalent of std::snprintf for older _MSC compilers
|
|
|
+#if defined(_MSC_VER) && _MSC_VER < 1900
|
|
|
+ #include <stdarg.h>
|
|
|
+static int msvc_pre1900_c99_vsnprintf(char* outBuf, size_t size,
|
|
|
+ const char* format, va_list ap) {
|
|
|
+ int count = -1;
|
|
|
+ if (size != 0)
|
|
|
+ count = _vsnprintf_s(outBuf, size, _TRUNCATE, format, ap);
|
|
|
+ if (count == -1)
|
|
|
+ count = _vscprintf(format, ap);
|
|
|
+ return count;
|
|
|
+}
|
|
|
+
|
|
|
+int JSON_API msvc_pre1900_c99_snprintf(char* outBuf, size_t size,
|
|
|
+ const char* format, ...) {
|
|
|
+ va_list ap;
|
|
|
+ va_start(ap, format);
|
|
|
+ const int count = msvc_pre1900_c99_vsnprintf(outBuf, size, format, ap);
|
|
|
+ va_end(ap);
|
|
|
+ return count;
|
|
|
+}
|
|
|
+#endif
|
|
|
+
|
|
|
+// Disable warning C4702 : unreachable code
|
|
|
+#if defined(_MSC_VER)
|
|
|
+#pragma warning(disable : 4702)
|
|
|
+#endif
|
|
|
+
|
|
|
+#define JSON_ASSERT_UNREACHABLE assert(false)
|
|
|
+
|
|
|
+namespace Json {
|
|
|
+ template<typename T>
|
|
|
+ static std::unique_ptr<T> cloneUnique(const std::unique_ptr<T> &p) {
|
|
|
+ std::unique_ptr<T> r;
|
|
|
+ if (p) {
|
|
|
+ r = std::unique_ptr<T>(new T(*p));
|
|
|
+ }
|
|
|
+ return r;
|
|
|
+ }
|
|
|
+
|
|
|
+// This is a walkaround to avoid the static initialization of Value::null.
|
|
|
+// kNull must be word-aligned to avoid crashing on ARM. We use an alignment of
|
|
|
+// 8 (instead of 4) as a bit of future-proofing.
|
|
|
+#if defined(__ARMEL__)
|
|
|
+#define ALIGNAS(byte_alignment) __attribute__((aligned(byte_alignment)))
|
|
|
+#else
|
|
|
+#define ALIGNAS(byte_alignment)
|
|
|
+#endif
|
|
|
+
|
|
|
+// static
|
|
|
+ Value const &Value::nullSingleton() {
|
|
|
+ static Value const nullStatic;
|
|
|
+ return nullStatic;
|
|
|
+ }
|
|
|
+
|
|
|
+#if JSON_USE_NULLREF
|
|
|
+// for backwards compatibility, we'll leave these global references around, but
|
|
|
+// DO NOT use them in JSONCPP library code any more!
|
|
|
+// static
|
|
|
+ Value const &Value::null = Value::nullSingleton();
|
|
|
+
|
|
|
+// static
|
|
|
+ Value const &Value::nullRef = Value::nullSingleton();
|
|
|
+#endif
|
|
|
+
|
|
|
+#if !defined(JSON_USE_INT64_DOUBLE_CONVERSION)
|
|
|
+ template <typename T, typename U>
|
|
|
+static inline bool InRange(double d, T min, U max) {
|
|
|
+ // The casts can lose precision, but we are looking only for
|
|
|
+ // an approximate range. Might fail on edge cases though. ~cdunn
|
|
|
+ return d >= static_cast<double>(min) && d <= static_cast<double>(max);
|
|
|
+}
|
|
|
+#else // if !defined(JSON_USE_INT64_DOUBLE_CONVERSION)
|
|
|
+
|
|
|
+ static inline double integerToDouble(Json::UInt64 value) {
|
|
|
+ return static_cast<double>(Int64(value / 2)) * 2.0 +
|
|
|
+ static_cast<double>(Int64(value & 1));
|
|
|
+ }
|
|
|
+
|
|
|
+ template<typename T>
|
|
|
+ static inline double integerToDouble(T value) {
|
|
|
+ return static_cast<double>(value);
|
|
|
+ }
|
|
|
+
|
|
|
+ template<typename T, typename U>
|
|
|
+ static inline bool InRange(double d, T min, U max) {
|
|
|
+ return d >= integerToDouble(min) && d <= integerToDouble(max);
|
|
|
+ }
|
|
|
+
|
|
|
+#endif // if !defined(JSON_USE_INT64_DOUBLE_CONVERSION)
|
|
|
+
|
|
|
+/** Duplicates the specified string value.
|
|
|
+ * @param value Pointer to the string to duplicate. Must be zero-terminated if
|
|
|
+ * length is "unknown".
|
|
|
+ * @param length Length of the value. if equals to unknown, then it will be
|
|
|
+ * computed using strlen(value).
|
|
|
+ * @return Pointer on the duplicate instance of string.
|
|
|
+ */
|
|
|
+ static inline char *duplicateStringValue(const char *value, size_t length) {
|
|
|
+ // Avoid an integer overflow in the call to malloc below by limiting length
|
|
|
+ // to a sane value.
|
|
|
+ if (length >= static_cast<size_t>(Value::maxInt))
|
|
|
+ length = Value::maxInt - 1;
|
|
|
+
|
|
|
+ auto newString = static_cast<char *>(malloc(length + 1));
|
|
|
+ if (newString == nullptr) {
|
|
|
+ throwRuntimeError("in Json::Value::duplicateStringValue(): "
|
|
|
+ "Failed to allocate string value buffer");
|
|
|
+ }
|
|
|
+ memcpy(newString, value, length);
|
|
|
+ newString[length] = 0;
|
|
|
+ return newString;
|
|
|
+ }
|
|
|
+
|
|
|
+/* Record the length as a prefix.
|
|
|
+ */
|
|
|
+ static inline char *duplicateAndPrefixStringValue(const char *value,
|
|
|
+ unsigned int length) {
|
|
|
+ // Avoid an integer overflow in the call to malloc below by limiting length
|
|
|
+ // to a sane value.
|
|
|
+ JSON_ASSERT_MESSAGE(length <= static_cast<unsigned>(Value::maxInt) -
|
|
|
+ sizeof(unsigned) - 1U,
|
|
|
+ "in Json::Value::duplicateAndPrefixStringValue(): "
|
|
|
+ "length too big for prefixing");
|
|
|
+ size_t actualLength = sizeof(length) + length + 1;
|
|
|
+ auto newString = static_cast<char *>(malloc(actualLength));
|
|
|
+ if (newString == nullptr) {
|
|
|
+ throwRuntimeError("in Json::Value::duplicateAndPrefixStringValue(): "
|
|
|
+ "Failed to allocate string value buffer");
|
|
|
+ }
|
|
|
+ *reinterpret_cast<unsigned *>(newString) = length;
|
|
|
+ memcpy(newString + sizeof(unsigned), value, length);
|
|
|
+ newString[actualLength - 1U] =
|
|
|
+ 0; // to avoid buffer over-run accidents by users later
|
|
|
+ return newString;
|
|
|
+ }
|
|
|
+
|
|
|
+ inline static void decodePrefixedString(bool isPrefixed, char const *prefixed,
|
|
|
+ unsigned *length, char const **value) {
|
|
|
+ if (!isPrefixed) {
|
|
|
+ *length = static_cast<unsigned>(strlen(prefixed));
|
|
|
+ *value = prefixed;
|
|
|
+ } else {
|
|
|
+ *length = *reinterpret_cast<unsigned const *>(prefixed);
|
|
|
+ *value = prefixed + sizeof(unsigned);
|
|
|
+ }
|
|
|
+ }
|
|
|
+/** Free the string duplicated by
|
|
|
+ * duplicateStringValue()/duplicateAndPrefixStringValue().
|
|
|
+ */
|
|
|
+#if JSONCPP_USING_SECURE_MEMORY
|
|
|
+ static inline void releasePrefixedStringValue(char* value) {
|
|
|
+ unsigned length = 0;
|
|
|
+ char const* valueDecoded;
|
|
|
+ decodePrefixedString(true, value, &length, &valueDecoded);
|
|
|
+ size_t const size = sizeof(unsigned) + length + 1U;
|
|
|
+ memset(value, 0, size);
|
|
|
+ free(value);
|
|
|
+}
|
|
|
+static inline void releaseStringValue(char* value, unsigned length) {
|
|
|
+ // length==0 => we allocated the strings memory
|
|
|
+ size_t size = (length == 0) ? strlen(value) : length;
|
|
|
+ memset(value, 0, size);
|
|
|
+ free(value);
|
|
|
+}
|
|
|
+#else // !JSONCPP_USING_SECURE_MEMORY
|
|
|
+
|
|
|
+ static inline void releasePrefixedStringValue(char *value) { free(value); }
|
|
|
+
|
|
|
+ static inline void releaseStringValue(char *value, unsigned) { free(value); }
|
|
|
+
|
|
|
+#endif // JSONCPP_USING_SECURE_MEMORY
|
|
|
+
|
|
|
+} // namespace Json
|
|
|
+
|
|
|
+// //////////////////////////////////////////////////////////////////
|
|
|
+// //////////////////////////////////////////////////////////////////
|
|
|
+// //////////////////////////////////////////////////////////////////
|
|
|
+// ValueInternals...
|
|
|
+// //////////////////////////////////////////////////////////////////
|
|
|
+// //////////////////////////////////////////////////////////////////
|
|
|
+// //////////////////////////////////////////////////////////////////
|
|
|
+#if !defined(JSON_IS_AMALGAMATION)
|
|
|
+
|
|
|
+#include "json_valueiterator.inl"
|
|
|
+#endif // if !defined(JSON_IS_AMALGAMATION)
|
|
|
+
|
|
|
+namespace Json {
|
|
|
+
|
|
|
+#if JSON_USE_EXCEPTION
|
|
|
+
|
|
|
+ Exception::Exception(String msg) : msg_(std::move(msg)) {}
|
|
|
+
|
|
|
+ Exception::~Exception() noexcept = default;
|
|
|
+
|
|
|
+ char const *Exception::what() const noexcept { return msg_.c_str(); }
|
|
|
+
|
|
|
+ RuntimeError::RuntimeError(String const &msg) : Exception(msg) {}
|
|
|
+
|
|
|
+ LogicError::LogicError(String const &msg) : Exception(msg) {}
|
|
|
+
|
|
|
+ JSONCPP_NORETURN void throwRuntimeError(String const &msg) {
|
|
|
+ throw RuntimeError(msg);
|
|
|
+ }
|
|
|
+
|
|
|
+ JSONCPP_NORETURN void throwLogicError(String const &msg) {
|
|
|
+ throw LogicError(msg);
|
|
|
+ }
|
|
|
+
|
|
|
+#else // !JSON_USE_EXCEPTION
|
|
|
+ JSONCPP_NORETURN void throwRuntimeError(String const& msg) {
|
|
|
+ std::cerr << msg << std::endl;
|
|
|
+ abort();
|
|
|
+}
|
|
|
+JSONCPP_NORETURN void throwLogicError(String const& msg) {
|
|
|
+ std::cerr << msg << std::endl;
|
|
|
+ abort();
|
|
|
+}
|
|
|
+#endif
|
|
|
+
|
|
|
+// //////////////////////////////////////////////////////////////////
|
|
|
+// //////////////////////////////////////////////////////////////////
|
|
|
+// //////////////////////////////////////////////////////////////////
|
|
|
+// class Value::CZString
|
|
|
+// //////////////////////////////////////////////////////////////////
|
|
|
+// //////////////////////////////////////////////////////////////////
|
|
|
+// //////////////////////////////////////////////////////////////////
|
|
|
+
|
|
|
+// Notes: policy_ indicates if the string was allocated when
|
|
|
+// a string is stored.
|
|
|
+
|
|
|
+ Value::CZString::CZString(ArrayIndex index) : cstr_(nullptr), index_(index) {}
|
|
|
+
|
|
|
+ Value::CZString::CZString(char const *str, unsigned length,
|
|
|
+ DuplicationPolicy allocate)
|
|
|
+ : cstr_(str) {
|
|
|
+ // allocate != duplicate
|
|
|
+ storage_.policy_ = allocate & 0x3;
|
|
|
+ storage_.length_ = length & 0x3FFFFFFF;
|
|
|
+ }
|
|
|
+
|
|
|
+ Value::CZString::CZString(const CZString &other) {
|
|
|
+ cstr_ = (other.storage_.policy_ != noDuplication && other.cstr_ != nullptr
|
|
|
+ ? duplicateStringValue(other.cstr_, other.storage_.length_)
|
|
|
+ : other.cstr_);
|
|
|
+ storage_.policy_ =
|
|
|
+ static_cast<unsigned>(
|
|
|
+ other.cstr_
|
|
|
+ ? (static_cast<DuplicationPolicy>(other.storage_.policy_) ==
|
|
|
+ noDuplication
|
|
|
+ ? noDuplication
|
|
|
+ : duplicate)
|
|
|
+ : static_cast<DuplicationPolicy>(other.storage_.policy_)) &
|
|
|
+ 3U;
|
|
|
+ storage_.length_ = other.storage_.length_;
|
|
|
+ }
|
|
|
+
|
|
|
+ Value::CZString::CZString(CZString &&other)
|
|
|
+ : cstr_(other.cstr_), index_(other.index_) {
|
|
|
+ other.cstr_ = nullptr;
|
|
|
+ }
|
|
|
+
|
|
|
+ Value::CZString::~CZString() {
|
|
|
+ if (cstr_ && storage_.policy_ == duplicate) {
|
|
|
+ releaseStringValue(const_cast<char *>(cstr_),
|
|
|
+ storage_.length_ + 1U); // +1 for null terminating
|
|
|
+ // character for sake of
|
|
|
+ // completeness but not actually
|
|
|
+ // necessary
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ void Value::CZString::swap(CZString &other) {
|
|
|
+ std::swap(cstr_, other.cstr_);
|
|
|
+ std::swap(index_, other.index_);
|
|
|
+ }
|
|
|
+
|
|
|
+ Value::CZString &Value::CZString::operator=(const CZString &other) {
|
|
|
+ cstr_ = other.cstr_;
|
|
|
+ index_ = other.index_;
|
|
|
+ return *this;
|
|
|
+ }
|
|
|
+
|
|
|
+ Value::CZString &Value::CZString::operator=(CZString &&other) {
|
|
|
+ cstr_ = other.cstr_;
|
|
|
+ index_ = other.index_;
|
|
|
+ other.cstr_ = nullptr;
|
|
|
+ return *this;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Value::CZString::operator<(const CZString &other) const {
|
|
|
+ if (!cstr_)
|
|
|
+ return index_ < other.index_;
|
|
|
+ // return strcmp(cstr_, other.cstr_) < 0;
|
|
|
+ // Assume both are strings.
|
|
|
+ unsigned this_len = this->storage_.length_;
|
|
|
+ unsigned other_len = other.storage_.length_;
|
|
|
+ unsigned min_len = std::min<unsigned>(this_len, other_len);
|
|
|
+ JSON_ASSERT(this->cstr_ && other.cstr_);
|
|
|
+ int comp = memcmp(this->cstr_, other.cstr_, min_len);
|
|
|
+ if (comp < 0)
|
|
|
+ return true;
|
|
|
+ if (comp > 0)
|
|
|
+ return false;
|
|
|
+ return (this_len < other_len);
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Value::CZString::operator==(const CZString &other) const {
|
|
|
+ if (!cstr_)
|
|
|
+ return index_ == other.index_;
|
|
|
+ // return strcmp(cstr_, other.cstr_) == 0;
|
|
|
+ // Assume both are strings.
|
|
|
+ unsigned this_len = this->storage_.length_;
|
|
|
+ unsigned other_len = other.storage_.length_;
|
|
|
+ if (this_len != other_len)
|
|
|
+ return false;
|
|
|
+ JSON_ASSERT(this->cstr_ && other.cstr_);
|
|
|
+ int comp = memcmp(this->cstr_, other.cstr_, this_len);
|
|
|
+ return comp == 0;
|
|
|
+ }
|
|
|
+
|
|
|
+ ArrayIndex Value::CZString::index() const { return index_; }
|
|
|
+
|
|
|
+// const char* Value::CZString::c_str() const { return cstr_; }
|
|
|
+ const char *Value::CZString::data() const { return cstr_; }
|
|
|
+
|
|
|
+ unsigned Value::CZString::length() const { return storage_.length_; }
|
|
|
+
|
|
|
+ bool Value::CZString::isStaticString() const {
|
|
|
+ return storage_.policy_ == noDuplication;
|
|
|
+ }
|
|
|
+
|
|
|
+// //////////////////////////////////////////////////////////////////
|
|
|
+// //////////////////////////////////////////////////////////////////
|
|
|
+// //////////////////////////////////////////////////////////////////
|
|
|
+// class Value::Value
|
|
|
+// //////////////////////////////////////////////////////////////////
|
|
|
+// //////////////////////////////////////////////////////////////////
|
|
|
+// //////////////////////////////////////////////////////////////////
|
|
|
+
|
|
|
+/*! \internal Default constructor initialization must be equivalent to:
|
|
|
+ * memset( this, 0, sizeof(Value) )
|
|
|
+ * This optimization is used in ValueInternalMap fast allocator.
|
|
|
+ */
|
|
|
+ Value::Value(ValueType type) {
|
|
|
+ static char const emptyString[] = "";
|
|
|
+ initBasic(type);
|
|
|
+ switch (type) {
|
|
|
+ case nullValue:
|
|
|
+ break;
|
|
|
+ case intValue:
|
|
|
+ case uintValue:
|
|
|
+ value_.int_ = 0;
|
|
|
+ break;
|
|
|
+ case realValue:
|
|
|
+ value_.real_ = 0.0;
|
|
|
+ break;
|
|
|
+ case stringValue:
|
|
|
+ // allocated_ == false, so this is safe.
|
|
|
+ value_.string_ = const_cast<char *>(static_cast<char const *>(emptyString));
|
|
|
+ break;
|
|
|
+ case arrayValue:
|
|
|
+ case objectValue:
|
|
|
+ value_.map_ = new ObjectValues();
|
|
|
+ break;
|
|
|
+ case booleanValue:
|
|
|
+ value_.bool_ = false;
|
|
|
+ break;
|
|
|
+ default:
|
|
|
+ JSON_ASSERT_UNREACHABLE;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ Value::Value(Int value) {
|
|
|
+ initBasic(intValue);
|
|
|
+ value_.int_ = value;
|
|
|
+ }
|
|
|
+
|
|
|
+ Value::Value(UInt value) {
|
|
|
+ initBasic(uintValue);
|
|
|
+ value_.uint_ = value;
|
|
|
+ }
|
|
|
+
|
|
|
+#if defined(JSON_HAS_INT64)
|
|
|
+
|
|
|
+ Value::Value(Int64 value) {
|
|
|
+ initBasic(intValue);
|
|
|
+ value_.int_ = value;
|
|
|
+ }
|
|
|
+
|
|
|
+ Value::Value(UInt64 value) {
|
|
|
+ initBasic(uintValue);
|
|
|
+ value_.uint_ = value;
|
|
|
+ }
|
|
|
+
|
|
|
+#endif // defined(JSON_HAS_INT64)
|
|
|
+
|
|
|
+ Value::Value(double value) {
|
|
|
+ initBasic(realValue);
|
|
|
+ value_.real_ = value;
|
|
|
+ }
|
|
|
+
|
|
|
+ Value::Value(const char *value) {
|
|
|
+ initBasic(stringValue, true);
|
|
|
+ JSON_ASSERT_MESSAGE(value != nullptr,
|
|
|
+ "Null Value Passed to Value Constructor");
|
|
|
+ value_.string_ = duplicateAndPrefixStringValue(
|
|
|
+ value, static_cast<unsigned>(strlen(value)));
|
|
|
+ }
|
|
|
+
|
|
|
+ Value::Value(const char *begin, const char *end) {
|
|
|
+ initBasic(stringValue, true);
|
|
|
+ value_.string_ =
|
|
|
+ duplicateAndPrefixStringValue(begin, static_cast<unsigned>(end - begin));
|
|
|
+ }
|
|
|
+
|
|
|
+ Value::Value(const String &value) {
|
|
|
+ initBasic(stringValue, true);
|
|
|
+ value_.string_ = duplicateAndPrefixStringValue(
|
|
|
+ value.data(), static_cast<unsigned>(value.length()));
|
|
|
+ }
|
|
|
+
|
|
|
+ Value::Value(const StaticString &value) {
|
|
|
+ initBasic(stringValue);
|
|
|
+ value_.string_ = const_cast<char *>(value.c_str());
|
|
|
+ }
|
|
|
+
|
|
|
+ Value::Value(bool value) {
|
|
|
+ initBasic(booleanValue);
|
|
|
+ value_.bool_ = value;
|
|
|
+ }
|
|
|
+
|
|
|
+ Value::Value(const Value &other) {
|
|
|
+ dupPayload(other);
|
|
|
+ dupMeta(other);
|
|
|
+ }
|
|
|
+
|
|
|
+ Value::Value(Value &&other) {
|
|
|
+ initBasic(nullValue);
|
|
|
+ swap(other);
|
|
|
+ }
|
|
|
+
|
|
|
+ Value::~Value() {
|
|
|
+ releasePayload();
|
|
|
+ value_.uint_ = 0;
|
|
|
+ }
|
|
|
+
|
|
|
+ Value &Value::operator=(const Value &other) {
|
|
|
+ Value(other).swap(*this);
|
|
|
+ return *this;
|
|
|
+ }
|
|
|
+
|
|
|
+ Value &Value::operator=(Value &&other) {
|
|
|
+ other.swap(*this);
|
|
|
+ return *this;
|
|
|
+ }
|
|
|
+
|
|
|
+ void Value::swapPayload(Value &other) {
|
|
|
+ std::swap(bits_, other.bits_);
|
|
|
+ std::swap(value_, other.value_);
|
|
|
+ }
|
|
|
+
|
|
|
+ void Value::copyPayload(const Value &other) {
|
|
|
+ releasePayload();
|
|
|
+ dupPayload(other);
|
|
|
+ }
|
|
|
+
|
|
|
+ void Value::swap(Value &other) {
|
|
|
+ swapPayload(other);
|
|
|
+ std::swap(comments_, other.comments_);
|
|
|
+ std::swap(start_, other.start_);
|
|
|
+ std::swap(limit_, other.limit_);
|
|
|
+ }
|
|
|
+
|
|
|
+ void Value::copy(const Value &other) {
|
|
|
+ copyPayload(other);
|
|
|
+ dupMeta(other);
|
|
|
+ }
|
|
|
+
|
|
|
+ ValueType Value::type() const {
|
|
|
+ return static_cast<ValueType>(bits_.value_type_);
|
|
|
+ }
|
|
|
+
|
|
|
+ int Value::compare(const Value &other) const {
|
|
|
+ if (*this < other)
|
|
|
+ return -1;
|
|
|
+ if (*this > other)
|
|
|
+ return 1;
|
|
|
+ return 0;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Value::operator<(const Value &other) const {
|
|
|
+ int typeDelta = type() - other.type();
|
|
|
+ if (typeDelta)
|
|
|
+ return typeDelta < 0;
|
|
|
+ switch (type()) {
|
|
|
+ case nullValue:
|
|
|
+ return false;
|
|
|
+ case intValue:
|
|
|
+ return value_.int_ < other.value_.int_;
|
|
|
+ case uintValue:
|
|
|
+ return value_.uint_ < other.value_.uint_;
|
|
|
+ case realValue:
|
|
|
+ return value_.real_ < other.value_.real_;
|
|
|
+ case booleanValue:
|
|
|
+ return value_.bool_ < other.value_.bool_;
|
|
|
+ case stringValue: {
|
|
|
+ if ((value_.string_ == nullptr) || (other.value_.string_ == nullptr)) {
|
|
|
+ return other.value_.string_ != nullptr;
|
|
|
+ }
|
|
|
+ unsigned this_len;
|
|
|
+ unsigned other_len;
|
|
|
+ char const *this_str;
|
|
|
+ char const *other_str;
|
|
|
+ decodePrefixedString(this->isAllocated(), this->value_.string_, &this_len,
|
|
|
+ &this_str);
|
|
|
+ decodePrefixedString(other.isAllocated(), other.value_.string_, &other_len,
|
|
|
+ &other_str);
|
|
|
+ unsigned min_len = std::min<unsigned>(this_len, other_len);
|
|
|
+ JSON_ASSERT(this_str && other_str);
|
|
|
+ int comp = memcmp(this_str, other_str, min_len);
|
|
|
+ if (comp < 0)
|
|
|
+ return true;
|
|
|
+ if (comp > 0)
|
|
|
+ return false;
|
|
|
+ return (this_len < other_len);
|
|
|
+ }
|
|
|
+ case arrayValue:
|
|
|
+ case objectValue: {
|
|
|
+ auto thisSize = value_.map_->size();
|
|
|
+ auto otherSize = other.value_.map_->size();
|
|
|
+ if (thisSize != otherSize)
|
|
|
+ return thisSize < otherSize;
|
|
|
+ return (*value_.map_) < (*other.value_.map_);
|
|
|
+ }
|
|
|
+ default:
|
|
|
+ JSON_ASSERT_UNREACHABLE;
|
|
|
+ }
|
|
|
+ return false; // unreachable
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Value::operator<=(const Value &other) const { return !(other < *this); }
|
|
|
+
|
|
|
+ bool Value::operator>=(const Value &other) const { return !(*this < other); }
|
|
|
+
|
|
|
+ bool Value::operator>(const Value &other) const { return other < *this; }
|
|
|
+
|
|
|
+ bool Value::operator==(const Value &other) const {
|
|
|
+ if (type() != other.type())
|
|
|
+ return false;
|
|
|
+ switch (type()) {
|
|
|
+ case nullValue:
|
|
|
+ return true;
|
|
|
+ case intValue:
|
|
|
+ return value_.int_ == other.value_.int_;
|
|
|
+ case uintValue:
|
|
|
+ return value_.uint_ == other.value_.uint_;
|
|
|
+ case realValue:
|
|
|
+ return value_.real_ == other.value_.real_;
|
|
|
+ case booleanValue:
|
|
|
+ return value_.bool_ == other.value_.bool_;
|
|
|
+ case stringValue: {
|
|
|
+ if ((value_.string_ == nullptr) || (other.value_.string_ == nullptr)) {
|
|
|
+ return (value_.string_ == other.value_.string_);
|
|
|
+ }
|
|
|
+ unsigned this_len;
|
|
|
+ unsigned other_len;
|
|
|
+ char const *this_str;
|
|
|
+ char const *other_str;
|
|
|
+ decodePrefixedString(this->isAllocated(), this->value_.string_, &this_len,
|
|
|
+ &this_str);
|
|
|
+ decodePrefixedString(other.isAllocated(), other.value_.string_, &other_len,
|
|
|
+ &other_str);
|
|
|
+ if (this_len != other_len)
|
|
|
+ return false;
|
|
|
+ JSON_ASSERT(this_str && other_str);
|
|
|
+ int comp = memcmp(this_str, other_str, this_len);
|
|
|
+ return comp == 0;
|
|
|
+ }
|
|
|
+ case arrayValue:
|
|
|
+ case objectValue:
|
|
|
+ return value_.map_->size() == other.value_.map_->size() &&
|
|
|
+ (*value_.map_) == (*other.value_.map_);
|
|
|
+ default:
|
|
|
+ JSON_ASSERT_UNREACHABLE;
|
|
|
+ }
|
|
|
+ return false; // unreachable
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Value::operator!=(const Value &other) const { return !(*this == other); }
|
|
|
+
|
|
|
+ const char *Value::asCString() const {
|
|
|
+ JSON_ASSERT_MESSAGE(type() == stringValue,
|
|
|
+ "in Json::Value::asCString(): requires stringValue");
|
|
|
+ if (value_.string_ == nullptr)
|
|
|
+ return nullptr;
|
|
|
+ unsigned this_len;
|
|
|
+ char const *this_str;
|
|
|
+ decodePrefixedString(this->isAllocated(), this->value_.string_, &this_len,
|
|
|
+ &this_str);
|
|
|
+ return this_str;
|
|
|
+ }
|
|
|
+
|
|
|
+#if JSONCPP_USING_SECURE_MEMORY
|
|
|
+ unsigned Value::getCStringLength() const {
|
|
|
+ JSON_ASSERT_MESSAGE(type() == stringValue,
|
|
|
+ "in Json::Value::asCString(): requires stringValue");
|
|
|
+ if (value_.string_ == 0)
|
|
|
+ return 0;
|
|
|
+ unsigned this_len;
|
|
|
+ char const* this_str;
|
|
|
+ decodePrefixedString(this->isAllocated(), this->value_.string_, &this_len,
|
|
|
+ &this_str);
|
|
|
+ return this_len;
|
|
|
+}
|
|
|
+#endif
|
|
|
+
|
|
|
+ bool Value::getString(char const **begin, char const **end) const {
|
|
|
+ if (type() != stringValue)
|
|
|
+ return false;
|
|
|
+ if (value_.string_ == nullptr)
|
|
|
+ return false;
|
|
|
+ unsigned length;
|
|
|
+ decodePrefixedString(this->isAllocated(), this->value_.string_, &length,
|
|
|
+ begin);
|
|
|
+ *end = *begin + length;
|
|
|
+ return true;
|
|
|
+ }
|
|
|
+
|
|
|
+ String Value::asString() const {
|
|
|
+ switch (type()) {
|
|
|
+ case nullValue:
|
|
|
+ return "";
|
|
|
+ case stringValue: {
|
|
|
+ if (value_.string_ == nullptr)
|
|
|
+ return "";
|
|
|
+ unsigned this_len;
|
|
|
+ char const *this_str;
|
|
|
+ decodePrefixedString(this->isAllocated(), this->value_.string_, &this_len,
|
|
|
+ &this_str);
|
|
|
+ return String(this_str, this_len);
|
|
|
+ }
|
|
|
+ case booleanValue:
|
|
|
+ return value_.bool_ ? "true" : "false";
|
|
|
+ case intValue:
|
|
|
+ return valueToString(value_.int_);
|
|
|
+ case uintValue:
|
|
|
+ return valueToString(value_.uint_);
|
|
|
+ case realValue:
|
|
|
+ return valueToString(value_.real_);
|
|
|
+ default:
|
|
|
+ JSON_FAIL_MESSAGE("Type is not convertible to string");
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ Value::Int Value::asInt() const {
|
|
|
+ switch (type()) {
|
|
|
+ case intValue:
|
|
|
+ JSON_ASSERT_MESSAGE(isInt(), "LargestInt out of Int range");
|
|
|
+ return Int(value_.int_);
|
|
|
+ case uintValue:
|
|
|
+ JSON_ASSERT_MESSAGE(isInt(), "LargestUInt out of Int range");
|
|
|
+ return Int(value_.uint_);
|
|
|
+ case realValue:
|
|
|
+ JSON_ASSERT_MESSAGE(InRange(value_.real_, minInt, maxInt),
|
|
|
+ "double out of Int range");
|
|
|
+ return Int(value_.real_);
|
|
|
+ case nullValue:
|
|
|
+ return 0;
|
|
|
+ case booleanValue:
|
|
|
+ return value_.bool_ ? 1 : 0;
|
|
|
+ default:
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ JSON_FAIL_MESSAGE("Value is not convertible to Int.");
|
|
|
+ }
|
|
|
+
|
|
|
+ Value::UInt Value::asUInt() const {
|
|
|
+ switch (type()) {
|
|
|
+ case intValue:
|
|
|
+ JSON_ASSERT_MESSAGE(isUInt(), "LargestInt out of UInt range");
|
|
|
+ return UInt(value_.int_);
|
|
|
+ case uintValue:
|
|
|
+ JSON_ASSERT_MESSAGE(isUInt(), "LargestUInt out of UInt range");
|
|
|
+ return UInt(value_.uint_);
|
|
|
+ case realValue:
|
|
|
+ JSON_ASSERT_MESSAGE(InRange(value_.real_, 0, maxUInt),
|
|
|
+ "double out of UInt range");
|
|
|
+ return UInt(value_.real_);
|
|
|
+ case nullValue:
|
|
|
+ return 0;
|
|
|
+ case booleanValue:
|
|
|
+ return value_.bool_ ? 1 : 0;
|
|
|
+ default:
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ JSON_FAIL_MESSAGE("Value is not convertible to UInt.");
|
|
|
+ }
|
|
|
+
|
|
|
+#if defined(JSON_HAS_INT64)
|
|
|
+
|
|
|
+ Value::Int64 Value::asInt64() const {
|
|
|
+ switch (type()) {
|
|
|
+ case intValue:
|
|
|
+ return Int64(value_.int_);
|
|
|
+ case uintValue:
|
|
|
+ JSON_ASSERT_MESSAGE(isInt64(), "LargestUInt out of Int64 range");
|
|
|
+ return Int64(value_.uint_);
|
|
|
+ case realValue:
|
|
|
+ JSON_ASSERT_MESSAGE(InRange(value_.real_, minInt64, maxInt64),
|
|
|
+ "double out of Int64 range");
|
|
|
+ return Int64(value_.real_);
|
|
|
+ case nullValue:
|
|
|
+ return 0;
|
|
|
+ case booleanValue:
|
|
|
+ return value_.bool_ ? 1 : 0;
|
|
|
+ default:
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ JSON_FAIL_MESSAGE("Value is not convertible to Int64.");
|
|
|
+ }
|
|
|
+
|
|
|
+ Value::UInt64 Value::asUInt64() const {
|
|
|
+ switch (type()) {
|
|
|
+ case intValue:
|
|
|
+ JSON_ASSERT_MESSAGE(isUInt64(), "LargestInt out of UInt64 range");
|
|
|
+ return UInt64(value_.int_);
|
|
|
+ case uintValue:
|
|
|
+ return UInt64(value_.uint_);
|
|
|
+ case realValue:
|
|
|
+ JSON_ASSERT_MESSAGE(InRange(value_.real_, 0, maxUInt64),
|
|
|
+ "double out of UInt64 range");
|
|
|
+ return UInt64(value_.real_);
|
|
|
+ case nullValue:
|
|
|
+ return 0;
|
|
|
+ case booleanValue:
|
|
|
+ return value_.bool_ ? 1 : 0;
|
|
|
+ default:
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ JSON_FAIL_MESSAGE("Value is not convertible to UInt64.");
|
|
|
+ }
|
|
|
+
|
|
|
+#endif // if defined(JSON_HAS_INT64)
|
|
|
+
|
|
|
+ LargestInt Value::asLargestInt() const {
|
|
|
+#if defined(JSON_NO_INT64)
|
|
|
+ return asInt();
|
|
|
+#else
|
|
|
+ return asInt64();
|
|
|
+#endif
|
|
|
+ }
|
|
|
+
|
|
|
+ LargestUInt Value::asLargestUInt() const {
|
|
|
+#if defined(JSON_NO_INT64)
|
|
|
+ return asUInt();
|
|
|
+#else
|
|
|
+ return asUInt64();
|
|
|
+#endif
|
|
|
+ }
|
|
|
+
|
|
|
+ double Value::asDouble() const {
|
|
|
+ switch (type()) {
|
|
|
+ case intValue:
|
|
|
+ return static_cast<double>(value_.int_);
|
|
|
+ case uintValue:
|
|
|
+#if !defined(JSON_USE_INT64_DOUBLE_CONVERSION)
|
|
|
+ return static_cast<double>(value_.uint_);
|
|
|
+#else // if !defined(JSON_USE_INT64_DOUBLE_CONVERSION)
|
|
|
+ return integerToDouble(value_.uint_);
|
|
|
+#endif // if !defined(JSON_USE_INT64_DOUBLE_CONVERSION)
|
|
|
+ case realValue:
|
|
|
+ return value_.real_;
|
|
|
+ case nullValue:
|
|
|
+ return 0.0;
|
|
|
+ case booleanValue:
|
|
|
+ return value_.bool_ ? 1.0 : 0.0;
|
|
|
+ default:
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ JSON_FAIL_MESSAGE("Value is not convertible to double.");
|
|
|
+ }
|
|
|
+
|
|
|
+ float Value::asFloat() const {
|
|
|
+ switch (type()) {
|
|
|
+ case intValue:
|
|
|
+ return static_cast<float>(value_.int_);
|
|
|
+ case uintValue:
|
|
|
+#if !defined(JSON_USE_INT64_DOUBLE_CONVERSION)
|
|
|
+ return static_cast<float>(value_.uint_);
|
|
|
+#else // if !defined(JSON_USE_INT64_DOUBLE_CONVERSION)
|
|
|
+ // This can fail (silently?) if the value is bigger than MAX_FLOAT.
|
|
|
+ return static_cast<float>(integerToDouble(value_.uint_));
|
|
|
+#endif // if !defined(JSON_USE_INT64_DOUBLE_CONVERSION)
|
|
|
+ case realValue:
|
|
|
+ return static_cast<float>(value_.real_);
|
|
|
+ case nullValue:
|
|
|
+ return 0.0;
|
|
|
+ case booleanValue:
|
|
|
+ return value_.bool_ ? 1.0F : 0.0F;
|
|
|
+ default:
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ JSON_FAIL_MESSAGE("Value is not convertible to float.");
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Value::asBool() const {
|
|
|
+ switch (type()) {
|
|
|
+ case booleanValue:
|
|
|
+ return value_.bool_;
|
|
|
+ case nullValue:
|
|
|
+ return false;
|
|
|
+ case intValue:
|
|
|
+ return value_.int_ != 0;
|
|
|
+ case uintValue:
|
|
|
+ return value_.uint_ != 0;
|
|
|
+ case realValue: {
|
|
|
+ // According to JavaScript language zero or NaN is regarded as false
|
|
|
+ const auto value_classification = std::fpclassify(value_.real_);
|
|
|
+ return value_classification != FP_ZERO && value_classification != FP_NAN;
|
|
|
+ }
|
|
|
+ default:
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ JSON_FAIL_MESSAGE("Value is not convertible to bool.");
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Value::isConvertibleTo(ValueType other) const {
|
|
|
+ switch (other) {
|
|
|
+ case nullValue:
|
|
|
+ return (isNumeric() && asDouble() == 0.0) ||
|
|
|
+ (type() == booleanValue && !value_.bool_) ||
|
|
|
+ (type() == stringValue && asString().empty()) ||
|
|
|
+ (type() == arrayValue && value_.map_->empty()) ||
|
|
|
+ (type() == objectValue && value_.map_->empty()) ||
|
|
|
+ type() == nullValue;
|
|
|
+ case intValue:
|
|
|
+ return isInt() ||
|
|
|
+ (type() == realValue && InRange(value_.real_, minInt, maxInt)) ||
|
|
|
+ type() == booleanValue || type() == nullValue;
|
|
|
+ case uintValue:
|
|
|
+ return isUInt() ||
|
|
|
+ (type() == realValue && InRange(value_.real_, 0, maxUInt)) ||
|
|
|
+ type() == booleanValue || type() == nullValue;
|
|
|
+ case realValue:
|
|
|
+ return isNumeric() || type() == booleanValue || type() == nullValue;
|
|
|
+ case booleanValue:
|
|
|
+ return isNumeric() || type() == booleanValue || type() == nullValue;
|
|
|
+ case stringValue:
|
|
|
+ return isNumeric() || type() == booleanValue || type() == stringValue ||
|
|
|
+ type() == nullValue;
|
|
|
+ case arrayValue:
|
|
|
+ return type() == arrayValue || type() == nullValue;
|
|
|
+ case objectValue:
|
|
|
+ return type() == objectValue || type() == nullValue;
|
|
|
+ }
|
|
|
+ JSON_ASSERT_UNREACHABLE;
|
|
|
+ return false;
|
|
|
+ }
|
|
|
+
|
|
|
+/// Number of values in array or object
|
|
|
+ ArrayIndex Value::size() const {
|
|
|
+ switch (type()) {
|
|
|
+ case nullValue:
|
|
|
+ case intValue:
|
|
|
+ case uintValue:
|
|
|
+ case realValue:
|
|
|
+ case booleanValue:
|
|
|
+ case stringValue:
|
|
|
+ return 0;
|
|
|
+ case arrayValue: // size of the array is highest index + 1
|
|
|
+ if (!value_.map_->empty()) {
|
|
|
+ ObjectValues::const_iterator itLast = value_.map_->end();
|
|
|
+ --itLast;
|
|
|
+ return (*itLast).first.index() + 1;
|
|
|
+ }
|
|
|
+ return 0;
|
|
|
+ case objectValue:
|
|
|
+ return ArrayIndex(value_.map_->size());
|
|
|
+ }
|
|
|
+ JSON_ASSERT_UNREACHABLE;
|
|
|
+ return 0; // unreachable;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Value::empty() const {
|
|
|
+ if (isNull() || isArray() || isObject())
|
|
|
+ return size() == 0U;
|
|
|
+ return false;
|
|
|
+ }
|
|
|
+
|
|
|
+ Value::operator bool() const { return !isNull(); }
|
|
|
+
|
|
|
+ void Value::clear() {
|
|
|
+ JSON_ASSERT_MESSAGE(type() == nullValue || type() == arrayValue ||
|
|
|
+ type() == objectValue,
|
|
|
+ "in Json::Value::clear(): requires complex value");
|
|
|
+ start_ = 0;
|
|
|
+ limit_ = 0;
|
|
|
+ switch (type()) {
|
|
|
+ case arrayValue:
|
|
|
+ case objectValue:
|
|
|
+ value_.map_->clear();
|
|
|
+ break;
|
|
|
+ default:
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ void Value::resize(ArrayIndex newSize) {
|
|
|
+ JSON_ASSERT_MESSAGE(type() == nullValue || type() == arrayValue,
|
|
|
+ "in Json::Value::resize(): requires arrayValue");
|
|
|
+ if (type() == nullValue)
|
|
|
+ *this = Value(arrayValue);
|
|
|
+ ArrayIndex oldSize = size();
|
|
|
+ if (newSize == 0)
|
|
|
+ clear();
|
|
|
+ else if (newSize > oldSize)
|
|
|
+ this->operator[](newSize - 1);
|
|
|
+ else {
|
|
|
+ for (ArrayIndex index = newSize; index < oldSize; ++index) {
|
|
|
+ value_.map_->erase(index);
|
|
|
+ }
|
|
|
+ JSON_ASSERT(size() == newSize);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ Value &Value::operator[](ArrayIndex index) {
|
|
|
+ JSON_ASSERT_MESSAGE(
|
|
|
+ type() == nullValue || type() == arrayValue,
|
|
|
+ "in Json::Value::operator[](ArrayIndex): requires arrayValue");
|
|
|
+ if (type() == nullValue)
|
|
|
+ *this = Value(arrayValue);
|
|
|
+ CZString key(index);
|
|
|
+ auto it = value_.map_->lower_bound(key);
|
|
|
+ if (it != value_.map_->end() && (*it).first == key)
|
|
|
+ return (*it).second;
|
|
|
+
|
|
|
+ ObjectValues::value_type defaultValue(key, nullSingleton());
|
|
|
+ it = value_.map_->insert(it, defaultValue);
|
|
|
+ return (*it).second;
|
|
|
+ }
|
|
|
+
|
|
|
+ Value &Value::operator[](int index) {
|
|
|
+ JSON_ASSERT_MESSAGE(
|
|
|
+ index >= 0,
|
|
|
+ "in Json::Value::operator[](int index): index cannot be negative");
|
|
|
+ return (*this)[ArrayIndex(index)];
|
|
|
+ }
|
|
|
+
|
|
|
+ const Value &Value::operator[](ArrayIndex index) const {
|
|
|
+ JSON_ASSERT_MESSAGE(
|
|
|
+ type() == nullValue || type() == arrayValue,
|
|
|
+ "in Json::Value::operator[](ArrayIndex)const: requires arrayValue");
|
|
|
+ if (type() == nullValue)
|
|
|
+ return nullSingleton();
|
|
|
+ CZString key(index);
|
|
|
+ ObjectValues::const_iterator it = value_.map_->find(key);
|
|
|
+ if (it == value_.map_->end())
|
|
|
+ return nullSingleton();
|
|
|
+ return (*it).second;
|
|
|
+ }
|
|
|
+
|
|
|
+ const Value &Value::operator[](int index) const {
|
|
|
+ JSON_ASSERT_MESSAGE(
|
|
|
+ index >= 0,
|
|
|
+ "in Json::Value::operator[](int index) const: index cannot be negative");
|
|
|
+ return (*this)[ArrayIndex(index)];
|
|
|
+ }
|
|
|
+
|
|
|
+ void Value::initBasic(ValueType type, bool allocated) {
|
|
|
+ setType(type);
|
|
|
+ setIsAllocated(allocated);
|
|
|
+ comments_ = Comments{};
|
|
|
+ start_ = 0;
|
|
|
+ limit_ = 0;
|
|
|
+ }
|
|
|
+
|
|
|
+ void Value::dupPayload(const Value &other) {
|
|
|
+ setType(other.type());
|
|
|
+ setIsAllocated(false);
|
|
|
+ switch (type()) {
|
|
|
+ case nullValue:
|
|
|
+ case intValue:
|
|
|
+ case uintValue:
|
|
|
+ case realValue:
|
|
|
+ case booleanValue:
|
|
|
+ value_ = other.value_;
|
|
|
+ break;
|
|
|
+ case stringValue:
|
|
|
+ if (other.value_.string_ && other.isAllocated()) {
|
|
|
+ unsigned len;
|
|
|
+ char const *str;
|
|
|
+ decodePrefixedString(other.isAllocated(), other.value_.string_, &len,
|
|
|
+ &str);
|
|
|
+ value_.string_ = duplicateAndPrefixStringValue(str, len);
|
|
|
+ setIsAllocated(true);
|
|
|
+ } else {
|
|
|
+ value_.string_ = other.value_.string_;
|
|
|
+ }
|
|
|
+ break;
|
|
|
+ case arrayValue:
|
|
|
+ case objectValue:
|
|
|
+ value_.map_ = new ObjectValues(*other.value_.map_);
|
|
|
+ break;
|
|
|
+ default:
|
|
|
+ JSON_ASSERT_UNREACHABLE;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ void Value::releasePayload() {
|
|
|
+ switch (type()) {
|
|
|
+ case nullValue:
|
|
|
+ case intValue:
|
|
|
+ case uintValue:
|
|
|
+ case realValue:
|
|
|
+ case booleanValue:
|
|
|
+ break;
|
|
|
+ case stringValue:
|
|
|
+ if (isAllocated())
|
|
|
+ releasePrefixedStringValue(value_.string_);
|
|
|
+ break;
|
|
|
+ case arrayValue:
|
|
|
+ case objectValue:
|
|
|
+ delete value_.map_;
|
|
|
+ break;
|
|
|
+ default:
|
|
|
+ JSON_ASSERT_UNREACHABLE;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ void Value::dupMeta(const Value &other) {
|
|
|
+ comments_ = other.comments_;
|
|
|
+ start_ = other.start_;
|
|
|
+ limit_ = other.limit_;
|
|
|
+ }
|
|
|
+
|
|
|
+// Access an object value by name, create a null member if it does not exist.
|
|
|
+// @pre Type of '*this' is object or null.
|
|
|
+// @param key is null-terminated.
|
|
|
+ Value &Value::resolveReference(const char *key) {
|
|
|
+ JSON_ASSERT_MESSAGE(
|
|
|
+ type() == nullValue || type() == objectValue,
|
|
|
+ "in Json::Value::resolveReference(): requires objectValue");
|
|
|
+ if (type() == nullValue)
|
|
|
+ *this = Value(objectValue);
|
|
|
+ CZString actualKey(key, static_cast<unsigned>(strlen(key)),
|
|
|
+ CZString::noDuplication); // NOTE!
|
|
|
+ auto it = value_.map_->lower_bound(actualKey);
|
|
|
+ if (it != value_.map_->end() && (*it).first == actualKey)
|
|
|
+ return (*it).second;
|
|
|
+
|
|
|
+ ObjectValues::value_type defaultValue(actualKey, nullSingleton());
|
|
|
+ it = value_.map_->insert(it, defaultValue);
|
|
|
+ Value &value = (*it).second;
|
|
|
+ return value;
|
|
|
+ }
|
|
|
+
|
|
|
+// @param key is not null-terminated.
|
|
|
+ Value &Value::resolveReference(char const *key, char const *end) {
|
|
|
+ JSON_ASSERT_MESSAGE(
|
|
|
+ type() == nullValue || type() == objectValue,
|
|
|
+ "in Json::Value::resolveReference(key, end): requires objectValue");
|
|
|
+ if (type() == nullValue)
|
|
|
+ *this = Value(objectValue);
|
|
|
+ CZString actualKey(key, static_cast<unsigned>(end - key),
|
|
|
+ CZString::duplicateOnCopy);
|
|
|
+ auto it = value_.map_->lower_bound(actualKey);
|
|
|
+ if (it != value_.map_->end() && (*it).first == actualKey)
|
|
|
+ return (*it).second;
|
|
|
+
|
|
|
+ ObjectValues::value_type defaultValue(actualKey, nullSingleton());
|
|
|
+ it = value_.map_->insert(it, defaultValue);
|
|
|
+ Value &value = (*it).second;
|
|
|
+ return value;
|
|
|
+ }
|
|
|
+
|
|
|
+ Value Value::get(ArrayIndex index, const Value &defaultValue) const {
|
|
|
+ const Value *value = &((*this)[index]);
|
|
|
+ return value == &nullSingleton() ? defaultValue : *value;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Value::isValidIndex(ArrayIndex index) const { return index < size(); }
|
|
|
+
|
|
|
+ Value const *Value::find(char const *begin, char const *end) const {
|
|
|
+ JSON_ASSERT_MESSAGE(type() == nullValue || type() == objectValue,
|
|
|
+ "in Json::Value::find(begin, end): requires "
|
|
|
+ "objectValue or nullValue");
|
|
|
+ if (type() == nullValue)
|
|
|
+ return nullptr;
|
|
|
+ CZString actualKey(begin, static_cast<unsigned>(end - begin),
|
|
|
+ CZString::noDuplication);
|
|
|
+ ObjectValues::const_iterator it = value_.map_->find(actualKey);
|
|
|
+ if (it == value_.map_->end())
|
|
|
+ return nullptr;
|
|
|
+ return &(*it).second;
|
|
|
+ }
|
|
|
+
|
|
|
+ Value *Value::demand(char const *begin, char const *end) {
|
|
|
+ JSON_ASSERT_MESSAGE(type() == nullValue || type() == objectValue,
|
|
|
+ "in Json::Value::demand(begin, end): requires "
|
|
|
+ "objectValue or nullValue");
|
|
|
+ return &resolveReference(begin, end);
|
|
|
+ }
|
|
|
+
|
|
|
+ const Value &Value::operator[](const char *key) const {
|
|
|
+ Value const *found = find(key, key + strlen(key));
|
|
|
+ if (!found)
|
|
|
+ return nullSingleton();
|
|
|
+ return *found;
|
|
|
+ }
|
|
|
+
|
|
|
+ Value const &Value::operator[](const String &key) const {
|
|
|
+ Value const *found = find(key.data(), key.data() + key.length());
|
|
|
+ if (!found)
|
|
|
+ return nullSingleton();
|
|
|
+ return *found;
|
|
|
+ }
|
|
|
+
|
|
|
+ Value &Value::operator[](const char *key) {
|
|
|
+ return resolveReference(key, key + strlen(key));
|
|
|
+ }
|
|
|
+
|
|
|
+ Value &Value::operator[](const String &key) {
|
|
|
+ return resolveReference(key.data(), key.data() + key.length());
|
|
|
+ }
|
|
|
+
|
|
|
+ Value &Value::operator[](const StaticString &key) {
|
|
|
+ return resolveReference(key.c_str());
|
|
|
+ }
|
|
|
+
|
|
|
+ Value &Value::append(const Value &value) { return append(Value(value)); }
|
|
|
+
|
|
|
+ Value &Value::append(Value &&value) {
|
|
|
+ JSON_ASSERT_MESSAGE(type() == nullValue || type() == arrayValue,
|
|
|
+ "in Json::Value::append: requires arrayValue");
|
|
|
+ if (type() == nullValue) {
|
|
|
+ *this = Value(arrayValue);
|
|
|
+ }
|
|
|
+ return this->value_.map_->emplace(size(), std::move(value)).first->second;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Value::insert(ArrayIndex index, const Value &newValue) {
|
|
|
+ return insert(index, Value(newValue));
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Value::insert(ArrayIndex index, Value &&newValue) {
|
|
|
+ JSON_ASSERT_MESSAGE(type() == nullValue || type() == arrayValue,
|
|
|
+ "in Json::Value::insert: requires arrayValue");
|
|
|
+ ArrayIndex length = size();
|
|
|
+ if (index > length) {
|
|
|
+ return false;
|
|
|
+ }
|
|
|
+ for (ArrayIndex i = length; i > index; i--) {
|
|
|
+ (*this)[i] = std::move((*this)[i - 1]);
|
|
|
+ }
|
|
|
+ (*this)[index] = std::move(newValue);
|
|
|
+ return true;
|
|
|
+ }
|
|
|
+
|
|
|
+ Value Value::get(char const *begin, char const *end,
|
|
|
+ Value const &defaultValue) const {
|
|
|
+ Value const *found = find(begin, end);
|
|
|
+ return !found ? defaultValue : *found;
|
|
|
+ }
|
|
|
+
|
|
|
+ Value Value::get(char const *key, Value const &defaultValue) const {
|
|
|
+ return get(key, key + strlen(key), defaultValue);
|
|
|
+ }
|
|
|
+
|
|
|
+ Value Value::get(String const &key, Value const &defaultValue) const {
|
|
|
+ return get(key.data(), key.data() + key.length(), defaultValue);
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Value::removeMember(const char *begin, const char *end, Value *removed) {
|
|
|
+ if (type() != objectValue) {
|
|
|
+ return false;
|
|
|
+ }
|
|
|
+ CZString actualKey(begin, static_cast<unsigned>(end - begin),
|
|
|
+ CZString::noDuplication);
|
|
|
+ auto it = value_.map_->find(actualKey);
|
|
|
+ if (it == value_.map_->end())
|
|
|
+ return false;
|
|
|
+ if (removed)
|
|
|
+ *removed = std::move(it->second);
|
|
|
+ value_.map_->erase(it);
|
|
|
+ return true;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Value::removeMember(const char *key, Value *removed) {
|
|
|
+ return removeMember(key, key + strlen(key), removed);
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Value::removeMember(String const &key, Value *removed) {
|
|
|
+ return removeMember(key.data(), key.data() + key.length(), removed);
|
|
|
+ }
|
|
|
+
|
|
|
+ void Value::removeMember(const char *key) {
|
|
|
+ JSON_ASSERT_MESSAGE(type() == nullValue || type() == objectValue,
|
|
|
+ "in Json::Value::removeMember(): requires objectValue");
|
|
|
+ if (type() == nullValue)
|
|
|
+ return;
|
|
|
+
|
|
|
+ CZString actualKey(key, unsigned(strlen(key)), CZString::noDuplication);
|
|
|
+ value_.map_->erase(actualKey);
|
|
|
+ }
|
|
|
+
|
|
|
+ void Value::removeMember(const String &key) { removeMember(key.c_str()); }
|
|
|
+
|
|
|
+ bool Value::removeIndex(ArrayIndex index, Value *removed) {
|
|
|
+ if (type() != arrayValue) {
|
|
|
+ return false;
|
|
|
+ }
|
|
|
+ CZString key(index);
|
|
|
+ auto it = value_.map_->find(key);
|
|
|
+ if (it == value_.map_->end()) {
|
|
|
+ return false;
|
|
|
+ }
|
|
|
+ if (removed)
|
|
|
+ *removed = it->second;
|
|
|
+ ArrayIndex oldSize = size();
|
|
|
+ // shift left all items left, into the place of the "removed"
|
|
|
+ for (ArrayIndex i = index; i < (oldSize - 1); ++i) {
|
|
|
+ CZString keey(i);
|
|
|
+ (*value_.map_)[keey] = (*this)[i + 1];
|
|
|
+ }
|
|
|
+ // erase the last one ("leftover")
|
|
|
+ CZString keyLast(oldSize - 1);
|
|
|
+ auto itLast = value_.map_->find(keyLast);
|
|
|
+ value_.map_->erase(itLast);
|
|
|
+ return true;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Value::isMember(char const *begin, char const *end) const {
|
|
|
+ Value const *value = find(begin, end);
|
|
|
+ return nullptr != value;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Value::isMember(char const *key) const {
|
|
|
+ return isMember(key, key + strlen(key));
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Value::isMember(String const &key) const {
|
|
|
+ return isMember(key.data(), key.data() + key.length());
|
|
|
+ }
|
|
|
+
|
|
|
+ Value::Members Value::getMemberNames() const {
|
|
|
+ JSON_ASSERT_MESSAGE(
|
|
|
+ type() == nullValue || type() == objectValue,
|
|
|
+ "in Json::Value::getMemberNames(), value must be objectValue");
|
|
|
+ if (type() == nullValue)
|
|
|
+ return Value::Members();
|
|
|
+ Members members;
|
|
|
+ members.reserve(value_.map_->size());
|
|
|
+ ObjectValues::const_iterator it = value_.map_->begin();
|
|
|
+ ObjectValues::const_iterator itEnd = value_.map_->end();
|
|
|
+ for (; it != itEnd; ++it) {
|
|
|
+ members.push_back(String((*it).first.data(), (*it).first.length()));
|
|
|
+ }
|
|
|
+ return members;
|
|
|
+ }
|
|
|
+
|
|
|
+ static bool IsIntegral(double d) {
|
|
|
+ double integral_part;
|
|
|
+ return modf(d, &integral_part) == 0.0;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Value::isNull() const { return type() == nullValue; }
|
|
|
+
|
|
|
+ bool Value::isBool() const { return type() == booleanValue; }
|
|
|
+
|
|
|
+ bool Value::isInt() const {
|
|
|
+ switch (type()) {
|
|
|
+ case intValue:
|
|
|
+#if defined(JSON_HAS_INT64)
|
|
|
+ return value_.int_ >= minInt && value_.int_ <= maxInt;
|
|
|
+#else
|
|
|
+ return true;
|
|
|
+#endif
|
|
|
+ case uintValue:
|
|
|
+ return value_.uint_ <= UInt(maxInt);
|
|
|
+ case realValue:
|
|
|
+ return value_.real_ >= minInt && value_.real_ <= maxInt &&
|
|
|
+ IsIntegral(value_.real_);
|
|
|
+ default:
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ return false;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Value::isUInt() const {
|
|
|
+ switch (type()) {
|
|
|
+ case intValue:
|
|
|
+#if defined(JSON_HAS_INT64)
|
|
|
+ return value_.int_ >= 0 && LargestUInt(value_.int_) <= LargestUInt(maxUInt);
|
|
|
+#else
|
|
|
+ return value_.int_ >= 0;
|
|
|
+#endif
|
|
|
+ case uintValue:
|
|
|
+#if defined(JSON_HAS_INT64)
|
|
|
+ return value_.uint_ <= maxUInt;
|
|
|
+#else
|
|
|
+ return true;
|
|
|
+#endif
|
|
|
+ case realValue:
|
|
|
+ return value_.real_ >= 0 && value_.real_ <= maxUInt &&
|
|
|
+ IsIntegral(value_.real_);
|
|
|
+ default:
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ return false;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Value::isInt64() const {
|
|
|
+#if defined(JSON_HAS_INT64)
|
|
|
+ switch (type()) {
|
|
|
+ case intValue:
|
|
|
+ return true;
|
|
|
+ case uintValue:
|
|
|
+ return value_.uint_ <= UInt64(maxInt64);
|
|
|
+ case realValue:
|
|
|
+ // Note that maxInt64 (= 2^63 - 1) is not exactly representable as a
|
|
|
+ // double, so double(maxInt64) will be rounded up to 2^63. Therefore we
|
|
|
+ // require the value to be strictly less than the limit.
|
|
|
+ return value_.real_ >= double(minInt64) &&
|
|
|
+ value_.real_ < double(maxInt64) && IsIntegral(value_.real_);
|
|
|
+ default:
|
|
|
+ break;
|
|
|
+ }
|
|
|
+#endif // JSON_HAS_INT64
|
|
|
+ return false;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Value::isUInt64() const {
|
|
|
+#if defined(JSON_HAS_INT64)
|
|
|
+ switch (type()) {
|
|
|
+ case intValue:
|
|
|
+ return value_.int_ >= 0;
|
|
|
+ case uintValue:
|
|
|
+ return true;
|
|
|
+ case realValue:
|
|
|
+ // Note that maxUInt64 (= 2^64 - 1) is not exactly representable as a
|
|
|
+ // double, so double(maxUInt64) will be rounded up to 2^64. Therefore we
|
|
|
+ // require the value to be strictly less than the limit.
|
|
|
+ return value_.real_ >= 0 && value_.real_ < maxUInt64AsDouble &&
|
|
|
+ IsIntegral(value_.real_);
|
|
|
+ default:
|
|
|
+ break;
|
|
|
+ }
|
|
|
+#endif // JSON_HAS_INT64
|
|
|
+ return false;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Value::isIntegral() const {
|
|
|
+ switch (type()) {
|
|
|
+ case intValue:
|
|
|
+ case uintValue:
|
|
|
+ return true;
|
|
|
+ case realValue:
|
|
|
+#if defined(JSON_HAS_INT64)
|
|
|
+ // Note that maxUInt64 (= 2^64 - 1) is not exactly representable as a
|
|
|
+ // double, so double(maxUInt64) will be rounded up to 2^64. Therefore we
|
|
|
+ // require the value to be strictly less than the limit.
|
|
|
+ return value_.real_ >= double(minInt64) &&
|
|
|
+ value_.real_ < maxUInt64AsDouble && IsIntegral(value_.real_);
|
|
|
+#else
|
|
|
+ return value_.real_ >= minInt && value_.real_ <= maxUInt &&
|
|
|
+ IsIntegral(value_.real_);
|
|
|
+#endif // JSON_HAS_INT64
|
|
|
+ default:
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ return false;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Value::isDouble() const {
|
|
|
+ return type() == intValue || type() == uintValue || type() == realValue;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Value::isNumeric() const { return isDouble(); }
|
|
|
+
|
|
|
+ bool Value::isString() const { return type() == stringValue; }
|
|
|
+
|
|
|
+ bool Value::isArray() const { return type() == arrayValue; }
|
|
|
+
|
|
|
+ bool Value::isObject() const { return type() == objectValue; }
|
|
|
+
|
|
|
+ Value::Comments::Comments(const Comments &that)
|
|
|
+ : ptr_{cloneUnique(that.ptr_)} {}
|
|
|
+
|
|
|
+ Value::Comments::Comments(Comments &&that) : ptr_{std::move(that.ptr_)} {}
|
|
|
+
|
|
|
+ Value::Comments &Value::Comments::operator=(const Comments &that) {
|
|
|
+ ptr_ = cloneUnique(that.ptr_);
|
|
|
+ return *this;
|
|
|
+ }
|
|
|
+
|
|
|
+ Value::Comments &Value::Comments::operator=(Comments &&that) {
|
|
|
+ ptr_ = std::move(that.ptr_);
|
|
|
+ return *this;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Value::Comments::has(CommentPlacement slot) const {
|
|
|
+ return ptr_ && !(*ptr_)[slot].empty();
|
|
|
+ }
|
|
|
+
|
|
|
+ String Value::Comments::get(CommentPlacement slot) const {
|
|
|
+ if (!ptr_)
|
|
|
+ return {};
|
|
|
+ return (*ptr_)[slot];
|
|
|
+ }
|
|
|
+
|
|
|
+ void Value::Comments::set(CommentPlacement slot, String comment) {
|
|
|
+ if (!ptr_) {
|
|
|
+ ptr_ = std::unique_ptr<Array>(new Array());
|
|
|
+ }
|
|
|
+ // check comments array boundry.
|
|
|
+ if (slot < CommentPlacement::numberOfCommentPlacement) {
|
|
|
+ (*ptr_)[slot] = std::move(comment);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ void Value::setComment(String comment, CommentPlacement placement) {
|
|
|
+ if (!comment.empty() && (comment.back() == '\n')) {
|
|
|
+ // Always discard trailing newline, to aid indentation.
|
|
|
+ comment.pop_back();
|
|
|
+ }
|
|
|
+ JSON_ASSERT(!comment.empty());
|
|
|
+ JSON_ASSERT_MESSAGE(
|
|
|
+ comment[0] == '\0' || comment[0] == '/',
|
|
|
+ "in Json::Value::setComment(): Comments must start with /");
|
|
|
+ comments_.set(placement, std::move(comment));
|
|
|
+ }
|
|
|
+
|
|
|
+ bool Value::hasComment(CommentPlacement placement) const {
|
|
|
+ return comments_.has(placement);
|
|
|
+ }
|
|
|
+
|
|
|
+ String Value::getComment(CommentPlacement placement) const {
|
|
|
+ return comments_.get(placement);
|
|
|
+ }
|
|
|
+
|
|
|
+ void Value::setOffsetStart(ptrdiff_t start) { start_ = start; }
|
|
|
+
|
|
|
+ void Value::setOffsetLimit(ptrdiff_t limit) { limit_ = limit; }
|
|
|
+
|
|
|
+ ptrdiff_t Value::getOffsetStart() const { return start_; }
|
|
|
+
|
|
|
+ ptrdiff_t Value::getOffsetLimit() const { return limit_; }
|
|
|
+
|
|
|
+ String Value::toStyledString() const {
|
|
|
+ StreamWriterBuilder builder;
|
|
|
+
|
|
|
+ String out = this->hasComment(commentBefore) ? "\n" : "";
|
|
|
+ out += Json::writeString(builder, *this);
|
|
|
+ out += '\n';
|
|
|
+
|
|
|
+ return out;
|
|
|
+ }
|
|
|
+
|
|
|
+ Value::const_iterator Value::begin() const {
|
|
|
+ switch (type()) {
|
|
|
+ case arrayValue:
|
|
|
+ case objectValue:
|
|
|
+ if (value_.map_)
|
|
|
+ return const_iterator(value_.map_->begin());
|
|
|
+ break;
|
|
|
+ default:
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ return {};
|
|
|
+ }
|
|
|
+
|
|
|
+ Value::const_iterator Value::end() const {
|
|
|
+ switch (type()) {
|
|
|
+ case arrayValue:
|
|
|
+ case objectValue:
|
|
|
+ if (value_.map_)
|
|
|
+ return const_iterator(value_.map_->end());
|
|
|
+ break;
|
|
|
+ default:
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ return {};
|
|
|
+ }
|
|
|
+
|
|
|
+ Value::iterator Value::begin() {
|
|
|
+ switch (type()) {
|
|
|
+ case arrayValue:
|
|
|
+ case objectValue:
|
|
|
+ if (value_.map_)
|
|
|
+ return iterator(value_.map_->begin());
|
|
|
+ break;
|
|
|
+ default:
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ return iterator();
|
|
|
+ }
|
|
|
+
|
|
|
+ Value::iterator Value::end() {
|
|
|
+ switch (type()) {
|
|
|
+ case arrayValue:
|
|
|
+ case objectValue:
|
|
|
+ if (value_.map_)
|
|
|
+ return iterator(value_.map_->end());
|
|
|
+ break;
|
|
|
+ default:
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ return iterator();
|
|
|
+ }
|
|
|
+
|
|
|
+// class PathArgument
|
|
|
+// //////////////////////////////////////////////////////////////////
|
|
|
+
|
|
|
+ PathArgument::PathArgument() = default;
|
|
|
+
|
|
|
+ PathArgument::PathArgument(ArrayIndex index)
|
|
|
+ : index_(index), kind_(kindIndex) {}
|
|
|
+
|
|
|
+ PathArgument::PathArgument(const char *key) : key_(key), kind_(kindKey) {}
|
|
|
+
|
|
|
+ PathArgument::PathArgument(String key) : key_(std::move(key)), kind_(kindKey) {}
|
|
|
+
|
|
|
+// class Path
|
|
|
+// //////////////////////////////////////////////////////////////////
|
|
|
+
|
|
|
+ Path::Path(const String &path, const PathArgument &a1, const PathArgument &a2,
|
|
|
+ const PathArgument &a3, const PathArgument &a4,
|
|
|
+ const PathArgument &a5) {
|
|
|
+ InArgs in;
|
|
|
+ in.reserve(5);
|
|
|
+ in.push_back(&a1);
|
|
|
+ in.push_back(&a2);
|
|
|
+ in.push_back(&a3);
|
|
|
+ in.push_back(&a4);
|
|
|
+ in.push_back(&a5);
|
|
|
+ makePath(path, in);
|
|
|
+ }
|
|
|
+
|
|
|
+ void Path::makePath(const String &path, const InArgs &in) {
|
|
|
+ const char *current = path.c_str();
|
|
|
+ const char *end = current + path.length();
|
|
|
+ auto itInArg = in.begin();
|
|
|
+ while (current != end) {
|
|
|
+ if (*current == '[') {
|
|
|
+ ++current;
|
|
|
+ if (*current == '%')
|
|
|
+ addPathInArg(path, in, itInArg, PathArgument::kindIndex);
|
|
|
+ else {
|
|
|
+ ArrayIndex index = 0;
|
|
|
+ for (; current != end && *current >= '0' && *current <= '9'; ++current)
|
|
|
+ index = index * 10 + ArrayIndex(*current - '0');
|
|
|
+ args_.push_back(index);
|
|
|
+ }
|
|
|
+ if (current == end || *++current != ']')
|
|
|
+ invalidPath(path, int(current - path.c_str()));
|
|
|
+ } else if (*current == '%') {
|
|
|
+ addPathInArg(path, in, itInArg, PathArgument::kindKey);
|
|
|
+ ++current;
|
|
|
+ } else if (*current == '.' || *current == ']') {
|
|
|
+ ++current;
|
|
|
+ } else {
|
|
|
+ const char *beginName = current;
|
|
|
+ while (current != end && !strchr("[.", *current))
|
|
|
+ ++current;
|
|
|
+ args_.push_back(String(beginName, current));
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ void Path::addPathInArg(const String & /*path*/, const InArgs &in,
|
|
|
+ InArgs::const_iterator &itInArg,
|
|
|
+ PathArgument::Kind kind) {
|
|
|
+ if (itInArg == in.end()) {
|
|
|
+ // Error: missing argument %d
|
|
|
+ } else if ((*itInArg)->kind_ != kind) {
|
|
|
+ // Error: bad argument type
|
|
|
+ } else {
|
|
|
+ args_.push_back(**itInArg++);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ void Path::invalidPath(const String & /*path*/, int /*location*/) {
|
|
|
+ // Error: invalid path.
|
|
|
+ }
|
|
|
+
|
|
|
+ const Value &Path::resolve(const Value &root) const {
|
|
|
+ const Value *node = &root;
|
|
|
+ for (const auto &arg : args_) {
|
|
|
+ if (arg.kind_ == PathArgument::kindIndex) {
|
|
|
+ if (!node->isArray() || !node->isValidIndex(arg.index_)) {
|
|
|
+ // Error: unable to resolve path (array value expected at position... )
|
|
|
+ return Value::nullSingleton();
|
|
|
+ }
|
|
|
+ node = &((*node)[arg.index_]);
|
|
|
+ } else if (arg.kind_ == PathArgument::kindKey) {
|
|
|
+ if (!node->isObject()) {
|
|
|
+ // Error: unable to resolve path (object value expected at position...)
|
|
|
+ return Value::nullSingleton();
|
|
|
+ }
|
|
|
+ node = &((*node)[arg.key_]);
|
|
|
+ if (node == &Value::nullSingleton()) {
|
|
|
+ // Error: unable to resolve path (object has no member named '' at
|
|
|
+ // position...)
|
|
|
+ return Value::nullSingleton();
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ return *node;
|
|
|
+ }
|
|
|
+
|
|
|
+ Value Path::resolve(const Value &root, const Value &defaultValue) const {
|
|
|
+ const Value *node = &root;
|
|
|
+ for (const auto &arg : args_) {
|
|
|
+ if (arg.kind_ == PathArgument::kindIndex) {
|
|
|
+ if (!node->isArray() || !node->isValidIndex(arg.index_))
|
|
|
+ return defaultValue;
|
|
|
+ node = &((*node)[arg.index_]);
|
|
|
+ } else if (arg.kind_ == PathArgument::kindKey) {
|
|
|
+ if (!node->isObject())
|
|
|
+ return defaultValue;
|
|
|
+ node = &((*node)[arg.key_]);
|
|
|
+ if (node == &Value::nullSingleton())
|
|
|
+ return defaultValue;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ return *node;
|
|
|
+ }
|
|
|
+
|
|
|
+ Value &Path::make(Value &root) const {
|
|
|
+ Value *node = &root;
|
|
|
+ for (const auto &arg : args_) {
|
|
|
+ if (arg.kind_ == PathArgument::kindIndex) {
|
|
|
+ if (!node->isArray()) {
|
|
|
+ // Error: node is not an array at position ...
|
|
|
+ }
|
|
|
+ node = &((*node)[arg.index_]);
|
|
|
+ } else if (arg.kind_ == PathArgument::kindKey) {
|
|
|
+ if (!node->isObject()) {
|
|
|
+ // Error: node is not an object at position...
|
|
|
+ }
|
|
|
+ node = &((*node)[arg.key_]);
|
|
|
+ }
|
|
|
+ }
|
|
|
+ return *node;
|
|
|
+ }
|
|
|
+
|
|
|
+} // namespace Json
|
|
|
+
|
|
|
+// //////////////////////////////////////////////////////////////////////
|
|
|
+// End of content of file: src/lib_json/json_value.cpp
|
|
|
+// //////////////////////////////////////////////////////////////////////
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+// //////////////////////////////////////////////////////////////////////
|
|
|
+// Beginning of content of file: src/lib_json/json_writer.cpp
|
|
|
+// //////////////////////////////////////////////////////////////////////
|
|
|
+
|
|
|
+// Copyright 2011 Baptiste Lepilleur and The JsonCpp Authors
|
|
|
+// Distributed under MIT license, or public domain if desired and
|
|
|
+// recognized in your jurisdiction.
|
|
|
+// See file LICENSE for detail or copy at http://jsoncpp.sourceforge.net/LICENSE
|
|
|
+
|
|
|
+#if !defined(JSON_IS_AMALGAMATION)
|
|
|
+ #include "json_tool.h"
|
|
|
+#include <json/writer.h>
|
|
|
+#endif // if !defined(JSON_IS_AMALGAMATION)
|
|
|
+
|
|
|
+#include <algorithm>
|
|
|
+#include <cassert>
|
|
|
+#include <cctype>
|
|
|
+#include <cstring>
|
|
|
+#include <iomanip>
|
|
|
+#include <memory>
|
|
|
+#include <set>
|
|
|
+#include <sstream>
|
|
|
+#include <utility>
|
|
|
+
|
|
|
+#if __cplusplus >= 201103L
|
|
|
+
|
|
|
+#include <cmath>
|
|
|
+#include <cstdio>
|
|
|
+
|
|
|
+#if !defined(isnan)
|
|
|
+#define isnan std::isnan
|
|
|
+#endif
|
|
|
+
|
|
|
+#if !defined(isfinite)
|
|
|
+#define isfinite std::isfinite
|
|
|
+#endif
|
|
|
+
|
|
|
+#else
|
|
|
+ #include <cmath>
|
|
|
+#include <cstdio>
|
|
|
+
|
|
|
+#if defined(_MSC_VER)
|
|
|
+#if !defined(isnan)
|
|
|
+#include <float.h>
|
|
|
+#define isnan _isnan
|
|
|
+#endif
|
|
|
+
|
|
|
+#if !defined(isfinite)
|
|
|
+#include <float.h>
|
|
|
+#define isfinite _finite
|
|
|
+#endif
|
|
|
+
|
|
|
+#if !defined(_CRT_SECURE_CPP_OVERLOAD_STANDARD_NAMES)
|
|
|
+#define _CRT_SECURE_CPP_OVERLOAD_STANDARD_NAMES 1
|
|
|
+#endif //_CRT_SECURE_CPP_OVERLOAD_STANDARD_NAMES
|
|
|
+
|
|
|
+#endif //_MSC_VER
|
|
|
+
|
|
|
+#if defined(__sun) && defined(__SVR4) // Solaris
|
|
|
+#if !defined(isfinite)
|
|
|
+#include <ieeefp.h>
|
|
|
+#define isfinite finite
|
|
|
+#endif
|
|
|
+#endif
|
|
|
+
|
|
|
+#if defined(__hpux)
|
|
|
+#if !defined(isfinite)
|
|
|
+#if defined(__ia64) && !defined(finite)
|
|
|
+#define isfinite(x) \
|
|
|
+ ((sizeof(x) == sizeof(float) ? _Isfinitef(x) : _IsFinite(x)))
|
|
|
+#endif
|
|
|
+#endif
|
|
|
+#endif
|
|
|
+
|
|
|
+#if !defined(isnan)
|
|
|
+// IEEE standard states that NaN values will not compare to themselves
|
|
|
+#define isnan(x) (x != x)
|
|
|
+#endif
|
|
|
+
|
|
|
+#if !defined(__APPLE__)
|
|
|
+#if !defined(isfinite)
|
|
|
+#define isfinite finite
|
|
|
+#endif
|
|
|
+#endif
|
|
|
+#endif
|
|
|
+
|
|
|
+#if defined(_MSC_VER)
|
|
|
+ // Disable warning about strdup being deprecated.
|
|
|
+#pragma warning(disable : 4996)
|
|
|
+#endif
|
|
|
+
|
|
|
+namespace Json {
|
|
|
+
|
|
|
+#if __cplusplus >= 201103L || (defined(_CPPLIB_VER) && _CPPLIB_VER >= 520)
|
|
|
+ using StreamWriterPtr = std::unique_ptr<StreamWriter>;
|
|
|
+#else
|
|
|
+ using StreamWriterPtr = std::auto_ptr<StreamWriter>;
|
|
|
+#endif
|
|
|
+
|
|
|
+ String valueToString(LargestInt value) {
|
|
|
+ UIntToStringBuffer buffer;
|
|
|
+ char *current = buffer + sizeof(buffer);
|
|
|
+ if (value == Value::minLargestInt) {
|
|
|
+ uintToString(LargestUInt(Value::maxLargestInt) + 1, current);
|
|
|
+ *--current = '-';
|
|
|
+ } else if (value < 0) {
|
|
|
+ uintToString(LargestUInt(-value), current);
|
|
|
+ *--current = '-';
|
|
|
+ } else {
|
|
|
+ uintToString(LargestUInt(value), current);
|
|
|
+ }
|
|
|
+ assert(current >= buffer);
|
|
|
+ return current;
|
|
|
+ }
|
|
|
+
|
|
|
+ String valueToString(LargestUInt value) {
|
|
|
+ UIntToStringBuffer buffer;
|
|
|
+ char *current = buffer + sizeof(buffer);
|
|
|
+ uintToString(value, current);
|
|
|
+ assert(current >= buffer);
|
|
|
+ return current;
|
|
|
+ }
|
|
|
+
|
|
|
+#if defined(JSON_HAS_INT64)
|
|
|
+
|
|
|
+ String valueToString(Int value) { return valueToString(LargestInt(value)); }
|
|
|
+
|
|
|
+ String valueToString(UInt value) { return valueToString(LargestUInt(value)); }
|
|
|
+
|
|
|
+#endif // # if defined(JSON_HAS_INT64)
|
|
|
+
|
|
|
+ namespace {
|
|
|
+ String valueToString(double value, bool useSpecialFloats,
|
|
|
+ unsigned int precision, PrecisionType precisionType) {
|
|
|
+ // Print into the buffer. We need not request the alternative representation
|
|
|
+ // that always has a decimal point because JSON doesn't distinguish the
|
|
|
+ // concepts of reals and integers.
|
|
|
+ if (!isfinite(value)) {
|
|
|
+ static const char *const reps[2][3] = {{"NaN", "-Infinity", "Infinity"},
|
|
|
+ {"null", "-1e+9999", "1e+9999"}};
|
|
|
+ return reps[useSpecialFloats ? 0 : 1]
|
|
|
+ [isnan(value) ? 0 : (value < 0) ? 1 : 2];
|
|
|
+ }
|
|
|
+
|
|
|
+ String buffer(size_t(36), '\0');
|
|
|
+ while (true) {
|
|
|
+ int len = jsoncpp_snprintf(
|
|
|
+ &*buffer.begin(), buffer.size(),
|
|
|
+ (precisionType == PrecisionType::significantDigits) ? "%.*g" : "%.*f",
|
|
|
+ precision, value);
|
|
|
+ assert(len >= 0);
|
|
|
+ auto wouldPrint = static_cast<size_t>(len);
|
|
|
+ if (wouldPrint >= buffer.size()) {
|
|
|
+ buffer.resize(wouldPrint + 1);
|
|
|
+ continue;
|
|
|
+ }
|
|
|
+ buffer.resize(wouldPrint);
|
|
|
+ break;
|
|
|
+ }
|
|
|
+
|
|
|
+ buffer.erase(fixNumericLocale(buffer.begin(), buffer.end()), buffer.end());
|
|
|
+
|
|
|
+ // strip the zero padding from the right
|
|
|
+ if (precisionType == PrecisionType::decimalPlaces) {
|
|
|
+ buffer.erase(fixZerosInTheEnd(buffer.begin(), buffer.end()), buffer.end());
|
|
|
+ }
|
|
|
+
|
|
|
+ // try to ensure we preserve the fact that this was given to us as a double on
|
|
|
+ // input
|
|
|
+ if (buffer.find('.') == buffer.npos && buffer.find('e') == buffer.npos) {
|
|
|
+ buffer += ".0";
|
|
|
+ }
|
|
|
+ return buffer;
|
|
|
+ }
|
|
|
+ } // namespace
|
|
|
+
|
|
|
+ String valueToString(double value, unsigned int precision,
|
|
|
+ PrecisionType precisionType) {
|
|
|
+ return valueToString(value, false, precision, precisionType);
|
|
|
+ }
|
|
|
+
|
|
|
+ String valueToString(bool value) { return value ? "true" : "false"; }
|
|
|
+
|
|
|
+ static bool doesAnyCharRequireEscaping(char const *s, size_t n) {
|
|
|
+ assert(s || !n);
|
|
|
+
|
|
|
+ return std::any_of(s, s + n, [](unsigned char c) {
|
|
|
+ return c == '\\' || c == '"' || c < 0x20 || c > 0x7F;
|
|
|
+ });
|
|
|
+ }
|
|
|
+
|
|
|
+ static unsigned int utf8ToCodepoint(const char *&s, const char *e) {
|
|
|
+ const unsigned int REPLACEMENT_CHARACTER = 0xFFFD;
|
|
|
+
|
|
|
+ unsigned int firstByte = static_cast<unsigned char>(*s);
|
|
|
+
|
|
|
+ if (firstByte < 0x80)
|
|
|
+ return firstByte;
|
|
|
+
|
|
|
+ if (firstByte < 0xE0) {
|
|
|
+ if (e - s < 2)
|
|
|
+ return REPLACEMENT_CHARACTER;
|
|
|
+
|
|
|
+ unsigned int calculated =
|
|
|
+ ((firstByte & 0x1F) << 6) | (static_cast<unsigned int>(s[1]) & 0x3F);
|
|
|
+ s += 1;
|
|
|
+ // oversized encoded characters are invalid
|
|
|
+ return calculated < 0x80 ? REPLACEMENT_CHARACTER : calculated;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (firstByte < 0xF0) {
|
|
|
+ if (e - s < 3)
|
|
|
+ return REPLACEMENT_CHARACTER;
|
|
|
+
|
|
|
+ unsigned int calculated = ((firstByte & 0x0F) << 12) |
|
|
|
+ ((static_cast<unsigned int>(s[1]) & 0x3F) << 6) |
|
|
|
+ (static_cast<unsigned int>(s[2]) & 0x3F);
|
|
|
+ s += 2;
|
|
|
+ // surrogates aren't valid codepoints itself
|
|
|
+ // shouldn't be UTF-8 encoded
|
|
|
+ if (calculated >= 0xD800 && calculated <= 0xDFFF)
|
|
|
+ return REPLACEMENT_CHARACTER;
|
|
|
+ // oversized encoded characters are invalid
|
|
|
+ return calculated < 0x800 ? REPLACEMENT_CHARACTER : calculated;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (firstByte < 0xF8) {
|
|
|
+ if (e - s < 4)
|
|
|
+ return REPLACEMENT_CHARACTER;
|
|
|
+
|
|
|
+ unsigned int calculated = ((firstByte & 0x07) << 18) |
|
|
|
+ ((static_cast<unsigned int>(s[1]) & 0x3F) << 12) |
|
|
|
+ ((static_cast<unsigned int>(s[2]) & 0x3F) << 6) |
|
|
|
+ (static_cast<unsigned int>(s[3]) & 0x3F);
|
|
|
+ s += 3;
|
|
|
+ // oversized encoded characters are invalid
|
|
|
+ return calculated < 0x10000 ? REPLACEMENT_CHARACTER : calculated;
|
|
|
+ }
|
|
|
+
|
|
|
+ return REPLACEMENT_CHARACTER;
|
|
|
+ }
|
|
|
+
|
|
|
+ static const char hex2[] = "000102030405060708090a0b0c0d0e0f"
|
|
|
+ "101112131415161718191a1b1c1d1e1f"
|
|
|
+ "202122232425262728292a2b2c2d2e2f"
|
|
|
+ "303132333435363738393a3b3c3d3e3f"
|
|
|
+ "404142434445464748494a4b4c4d4e4f"
|
|
|
+ "505152535455565758595a5b5c5d5e5f"
|
|
|
+ "606162636465666768696a6b6c6d6e6f"
|
|
|
+ "707172737475767778797a7b7c7d7e7f"
|
|
|
+ "808182838485868788898a8b8c8d8e8f"
|
|
|
+ "909192939495969798999a9b9c9d9e9f"
|
|
|
+ "a0a1a2a3a4a5a6a7a8a9aaabacadaeaf"
|
|
|
+ "b0b1b2b3b4b5b6b7b8b9babbbcbdbebf"
|
|
|
+ "c0c1c2c3c4c5c6c7c8c9cacbcccdcecf"
|
|
|
+ "d0d1d2d3d4d5d6d7d8d9dadbdcdddedf"
|
|
|
+ "e0e1e2e3e4e5e6e7e8e9eaebecedeeef"
|
|
|
+ "f0f1f2f3f4f5f6f7f8f9fafbfcfdfeff";
|
|
|
+
|
|
|
+ static String toHex16Bit(unsigned int x) {
|
|
|
+ const unsigned int hi = (x >> 8) & 0xff;
|
|
|
+ const unsigned int lo = x & 0xff;
|
|
|
+ String result(4, ' ');
|
|
|
+ result[0] = hex2[2 * hi];
|
|
|
+ result[1] = hex2[2 * hi + 1];
|
|
|
+ result[2] = hex2[2 * lo];
|
|
|
+ result[3] = hex2[2 * lo + 1];
|
|
|
+ return result;
|
|
|
+ }
|
|
|
+
|
|
|
+ static void appendRaw(String &result, unsigned ch) {
|
|
|
+ result += static_cast<char>(ch);
|
|
|
+ }
|
|
|
+
|
|
|
+ static void appendHex(String &result, unsigned ch) {
|
|
|
+ result.append("\\u").append(toHex16Bit(ch));
|
|
|
+ }
|
|
|
+
|
|
|
+ static String valueToQuotedStringN(const char *value, unsigned length,
|
|
|
+ bool emitUTF8 = false) {
|
|
|
+ if (value == nullptr)
|
|
|
+ return "";
|
|
|
+
|
|
|
+ if (!doesAnyCharRequireEscaping(value, length))
|
|
|
+ return String("\"") + value + "\"";
|
|
|
+ // We have to walk value and escape any special characters.
|
|
|
+ // Appending to String is not efficient, but this should be rare.
|
|
|
+ // (Note: forward slashes are *not* rare, but I am not escaping them.)
|
|
|
+ String::size_type maxsize = length * 2 + 3; // allescaped+quotes+NULL
|
|
|
+ String result;
|
|
|
+ result.reserve(maxsize); // to avoid lots of mallocs
|
|
|
+ result += "\"";
|
|
|
+ char const *end = value + length;
|
|
|
+ for (const char *c = value; c != end; ++c) {
|
|
|
+ switch (*c) {
|
|
|
+ case '\"':
|
|
|
+ result += "\\\"";
|
|
|
+ break;
|
|
|
+ case '\\':
|
|
|
+ result += "\\\\";
|
|
|
+ break;
|
|
|
+ case '\b':
|
|
|
+ result += "\\b";
|
|
|
+ break;
|
|
|
+ case '\f':
|
|
|
+ result += "\\f";
|
|
|
+ break;
|
|
|
+ case '\n':
|
|
|
+ result += "\\n";
|
|
|
+ break;
|
|
|
+ case '\r':
|
|
|
+ result += "\\r";
|
|
|
+ break;
|
|
|
+ case '\t':
|
|
|
+ result += "\\t";
|
|
|
+ break;
|
|
|
+ // case '/':
|
|
|
+ // Even though \/ is considered a legal escape in JSON, a bare
|
|
|
+ // slash is also legal, so I see no reason to escape it.
|
|
|
+ // (I hope I am not misunderstanding something.)
|
|
|
+ // blep notes: actually escaping \/ may be useful in javascript to avoid </
|
|
|
+ // sequence.
|
|
|
+ // Should add a flag to allow this compatibility mode and prevent this
|
|
|
+ // sequence from occurring.
|
|
|
+ default: {
|
|
|
+ if (emitUTF8) {
|
|
|
+ unsigned codepoint = static_cast<unsigned char>(*c);
|
|
|
+ if (codepoint < 0x20) {
|
|
|
+ appendHex(result, codepoint);
|
|
|
+ } else {
|
|
|
+ appendRaw(result, codepoint);
|
|
|
+ }
|
|
|
+ } else {
|
|
|
+ unsigned codepoint = utf8ToCodepoint(c, end); // modifies `c`
|
|
|
+ if (codepoint < 0x20) {
|
|
|
+ appendHex(result, codepoint);
|
|
|
+ } else if (codepoint < 0x80) {
|
|
|
+ appendRaw(result, codepoint);
|
|
|
+ } else if (codepoint < 0x10000) {
|
|
|
+ // Basic Multilingual Plane
|
|
|
+ appendHex(result, codepoint);
|
|
|
+ } else {
|
|
|
+ // Extended Unicode. Encode 20 bits as a surrogate pair.
|
|
|
+ codepoint -= 0x10000;
|
|
|
+ appendHex(result, 0xd800 + ((codepoint >> 10) & 0x3ff));
|
|
|
+ appendHex(result, 0xdc00 + (codepoint & 0x3ff));
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ result += "\"";
|
|
|
+ return result;
|
|
|
+ }
|
|
|
+
|
|
|
+ String valueToQuotedString(const char *value) {
|
|
|
+ return valueToQuotedStringN(value, static_cast<unsigned int>(strlen(value)));
|
|
|
+ }
|
|
|
+
|
|
|
+// Class Writer
|
|
|
+// //////////////////////////////////////////////////////////////////
|
|
|
+ Writer::~Writer() = default;
|
|
|
+
|
|
|
+// Class FastWriter
|
|
|
+// //////////////////////////////////////////////////////////////////
|
|
|
+
|
|
|
+ FastWriter::FastWriter()
|
|
|
+
|
|
|
+ = default;
|
|
|
+
|
|
|
+ void FastWriter::enableYAMLCompatibility() { yamlCompatibilityEnabled_ = true; }
|
|
|
+
|
|
|
+ void FastWriter::dropNullPlaceholders() { dropNullPlaceholders_ = true; }
|
|
|
+
|
|
|
+ void FastWriter::omitEndingLineFeed() { omitEndingLineFeed_ = true; }
|
|
|
+
|
|
|
+ String FastWriter::write(const Value &root) {
|
|
|
+ document_.clear();
|
|
|
+ writeValue(root);
|
|
|
+ if (!omitEndingLineFeed_)
|
|
|
+ document_ += '\n';
|
|
|
+ return document_;
|
|
|
+ }
|
|
|
+
|
|
|
+ void FastWriter::writeValue(const Value &value) {
|
|
|
+ switch (value.type()) {
|
|
|
+ case nullValue:
|
|
|
+ if (!dropNullPlaceholders_)
|
|
|
+ document_ += "null";
|
|
|
+ break;
|
|
|
+ case intValue:
|
|
|
+ document_ += valueToString(value.asLargestInt());
|
|
|
+ break;
|
|
|
+ case uintValue:
|
|
|
+ document_ += valueToString(value.asLargestUInt());
|
|
|
+ break;
|
|
|
+ case realValue:
|
|
|
+ document_ += valueToString(value.asDouble());
|
|
|
+ break;
|
|
|
+ case stringValue: {
|
|
|
+ // Is NULL possible for value.string_? No.
|
|
|
+ char const *str;
|
|
|
+ char const *end;
|
|
|
+ bool ok = value.getString(&str, &end);
|
|
|
+ if (ok)
|
|
|
+ document_ += valueToQuotedStringN(str, static_cast<unsigned>(end - str));
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ case booleanValue:
|
|
|
+ document_ += valueToString(value.asBool());
|
|
|
+ break;
|
|
|
+ case arrayValue: {
|
|
|
+ document_ += '[';
|
|
|
+ ArrayIndex size = value.size();
|
|
|
+ for (ArrayIndex index = 0; index < size; ++index) {
|
|
|
+ if (index > 0)
|
|
|
+ document_ += ',';
|
|
|
+ writeValue(value[index]);
|
|
|
+ }
|
|
|
+ document_ += ']';
|
|
|
+ }
|
|
|
+ break;
|
|
|
+ case objectValue: {
|
|
|
+ Value::Members members(value.getMemberNames());
|
|
|
+ document_ += '{';
|
|
|
+ for (auto it = members.begin(); it != members.end(); ++it) {
|
|
|
+ const String &name = *it;
|
|
|
+ if (it != members.begin())
|
|
|
+ document_ += ',';
|
|
|
+ document_ += valueToQuotedStringN(name.data(),
|
|
|
+ static_cast<unsigned>(name.length()));
|
|
|
+ document_ += yamlCompatibilityEnabled_ ? ": " : ":";
|
|
|
+ writeValue(value[name]);
|
|
|
+ }
|
|
|
+ document_ += '}';
|
|
|
+ }
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+// Class StyledWriter
|
|
|
+// //////////////////////////////////////////////////////////////////
|
|
|
+
|
|
|
+ StyledWriter::StyledWriter() = default;
|
|
|
+
|
|
|
+ String StyledWriter::write(const Value &root) {
|
|
|
+ document_.clear();
|
|
|
+ addChildValues_ = false;
|
|
|
+ indentString_.clear();
|
|
|
+ writeCommentBeforeValue(root);
|
|
|
+ writeValue(root);
|
|
|
+ writeCommentAfterValueOnSameLine(root);
|
|
|
+ document_ += '\n';
|
|
|
+ return document_;
|
|
|
+ }
|
|
|
+
|
|
|
+ void StyledWriter::writeValue(const Value &value) {
|
|
|
+ switch (value.type()) {
|
|
|
+ case nullValue:
|
|
|
+ pushValue("null");
|
|
|
+ break;
|
|
|
+ case intValue:
|
|
|
+ pushValue(valueToString(value.asLargestInt()));
|
|
|
+ break;
|
|
|
+ case uintValue:
|
|
|
+ pushValue(valueToString(value.asLargestUInt()));
|
|
|
+ break;
|
|
|
+ case realValue:
|
|
|
+ pushValue(valueToString(value.asDouble()));
|
|
|
+ break;
|
|
|
+ case stringValue: {
|
|
|
+ // Is NULL possible for value.string_? No.
|
|
|
+ char const *str;
|
|
|
+ char const *end;
|
|
|
+ bool ok = value.getString(&str, &end);
|
|
|
+ if (ok)
|
|
|
+ pushValue(valueToQuotedStringN(str, static_cast<unsigned>(end - str)));
|
|
|
+ else
|
|
|
+ pushValue("");
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ case booleanValue:
|
|
|
+ pushValue(valueToString(value.asBool()));
|
|
|
+ break;
|
|
|
+ case arrayValue:
|
|
|
+ writeArrayValue(value);
|
|
|
+ break;
|
|
|
+ case objectValue: {
|
|
|
+ Value::Members members(value.getMemberNames());
|
|
|
+ if (members.empty())
|
|
|
+ pushValue("{}");
|
|
|
+ else {
|
|
|
+ writeWithIndent("{");
|
|
|
+ indent();
|
|
|
+ auto it = members.begin();
|
|
|
+ for (;;) {
|
|
|
+ const String &name = *it;
|
|
|
+ const Value &childValue = value[name];
|
|
|
+ writeCommentBeforeValue(childValue);
|
|
|
+ writeWithIndent(valueToQuotedString(name.c_str()));
|
|
|
+ document_ += " : ";
|
|
|
+ writeValue(childValue);
|
|
|
+ if (++it == members.end()) {
|
|
|
+ writeCommentAfterValueOnSameLine(childValue);
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ document_ += ',';
|
|
|
+ writeCommentAfterValueOnSameLine(childValue);
|
|
|
+ }
|
|
|
+ unindent();
|
|
|
+ writeWithIndent("}");
|
|
|
+ }
|
|
|
+ }
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ void StyledWriter::writeArrayValue(const Value &value) {
|
|
|
+ unsigned size = value.size();
|
|
|
+ if (size == 0)
|
|
|
+ pushValue("[]");
|
|
|
+ else {
|
|
|
+ bool isArrayMultiLine = isMultilineArray(value);
|
|
|
+ if (isArrayMultiLine) {
|
|
|
+ writeWithIndent("[");
|
|
|
+ indent();
|
|
|
+ bool hasChildValue = !childValues_.empty();
|
|
|
+ unsigned index = 0;
|
|
|
+ for (;;) {
|
|
|
+ const Value &childValue = value[index];
|
|
|
+ writeCommentBeforeValue(childValue);
|
|
|
+ if (hasChildValue)
|
|
|
+ writeWithIndent(childValues_[index]);
|
|
|
+ else {
|
|
|
+ writeIndent();
|
|
|
+ writeValue(childValue);
|
|
|
+ }
|
|
|
+ if (++index == size) {
|
|
|
+ writeCommentAfterValueOnSameLine(childValue);
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ document_ += ',';
|
|
|
+ writeCommentAfterValueOnSameLine(childValue);
|
|
|
+ }
|
|
|
+ unindent();
|
|
|
+ writeWithIndent("]");
|
|
|
+ } else // output on a single line
|
|
|
+ {
|
|
|
+ assert(childValues_.size() == size);
|
|
|
+ document_ += "[ ";
|
|
|
+ for (unsigned index = 0; index < size; ++index) {
|
|
|
+ if (index > 0)
|
|
|
+ document_ += ", ";
|
|
|
+ document_ += childValues_[index];
|
|
|
+ }
|
|
|
+ document_ += " ]";
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ bool StyledWriter::isMultilineArray(const Value &value) {
|
|
|
+ ArrayIndex const size = value.size();
|
|
|
+ bool isMultiLine = size * 3 >= rightMargin_;
|
|
|
+ childValues_.clear();
|
|
|
+ for (ArrayIndex index = 0; index < size && !isMultiLine; ++index) {
|
|
|
+ const Value &childValue = value[index];
|
|
|
+ isMultiLine = ((childValue.isArray() || childValue.isObject()) &&
|
|
|
+ !childValue.empty());
|
|
|
+ }
|
|
|
+ if (!isMultiLine) // check if line length > max line length
|
|
|
+ {
|
|
|
+ childValues_.reserve(size);
|
|
|
+ addChildValues_ = true;
|
|
|
+ ArrayIndex lineLength = 4 + (size - 1) * 2; // '[ ' + ', '*n + ' ]'
|
|
|
+ for (ArrayIndex index = 0; index < size; ++index) {
|
|
|
+ if (hasCommentForValue(value[index])) {
|
|
|
+ isMultiLine = true;
|
|
|
+ }
|
|
|
+ writeValue(value[index]);
|
|
|
+ lineLength += static_cast<ArrayIndex>(childValues_[index].length());
|
|
|
+ }
|
|
|
+ addChildValues_ = false;
|
|
|
+ isMultiLine = isMultiLine || lineLength >= rightMargin_;
|
|
|
+ }
|
|
|
+ return isMultiLine;
|
|
|
+ }
|
|
|
+
|
|
|
+ void StyledWriter::pushValue(const String &value) {
|
|
|
+ if (addChildValues_)
|
|
|
+ childValues_.push_back(value);
|
|
|
+ else
|
|
|
+ document_ += value;
|
|
|
+ }
|
|
|
+
|
|
|
+ void StyledWriter::writeIndent() {
|
|
|
+ if (!document_.empty()) {
|
|
|
+ char last = document_[document_.length() - 1];
|
|
|
+ if (last == ' ') // already indented
|
|
|
+ return;
|
|
|
+ if (last != '\n') // Comments may add new-line
|
|
|
+ document_ += '\n';
|
|
|
+ }
|
|
|
+ document_ += indentString_;
|
|
|
+ }
|
|
|
+
|
|
|
+ void StyledWriter::writeWithIndent(const String &value) {
|
|
|
+ writeIndent();
|
|
|
+ document_ += value;
|
|
|
+ }
|
|
|
+
|
|
|
+ void StyledWriter::indent() { indentString_ += String(indentSize_, ' '); }
|
|
|
+
|
|
|
+ void StyledWriter::unindent() {
|
|
|
+ assert(indentString_.size() >= indentSize_);
|
|
|
+ indentString_.resize(indentString_.size() - indentSize_);
|
|
|
+ }
|
|
|
+
|
|
|
+ void StyledWriter::writeCommentBeforeValue(const Value &root) {
|
|
|
+ if (!root.hasComment(commentBefore))
|
|
|
+ return;
|
|
|
+
|
|
|
+ document_ += '\n';
|
|
|
+ writeIndent();
|
|
|
+ const String &comment = root.getComment(commentBefore);
|
|
|
+ String::const_iterator iter = comment.begin();
|
|
|
+ while (iter != comment.end()) {
|
|
|
+ document_ += *iter;
|
|
|
+ if (*iter == '\n' && ((iter + 1) != comment.end() && *(iter + 1) == '/'))
|
|
|
+ writeIndent();
|
|
|
+ ++iter;
|
|
|
+ }
|
|
|
+
|
|
|
+ // Comments are stripped of trailing newlines, so add one here
|
|
|
+ document_ += '\n';
|
|
|
+ }
|
|
|
+
|
|
|
+ void StyledWriter::writeCommentAfterValueOnSameLine(const Value &root) {
|
|
|
+ if (root.hasComment(commentAfterOnSameLine))
|
|
|
+ document_ += " " + root.getComment(commentAfterOnSameLine);
|
|
|
+
|
|
|
+ if (root.hasComment(commentAfter)) {
|
|
|
+ document_ += '\n';
|
|
|
+ document_ += root.getComment(commentAfter);
|
|
|
+ document_ += '\n';
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ bool StyledWriter::hasCommentForValue(const Value &value) {
|
|
|
+ return value.hasComment(commentBefore) ||
|
|
|
+ value.hasComment(commentAfterOnSameLine) ||
|
|
|
+ value.hasComment(commentAfter);
|
|
|
+ }
|
|
|
+
|
|
|
+// Class StyledStreamWriter
|
|
|
+// //////////////////////////////////////////////////////////////////
|
|
|
+
|
|
|
+ StyledStreamWriter::StyledStreamWriter(String indentation)
|
|
|
+ : document_(nullptr), indentation_(std::move(indentation)),
|
|
|
+ addChildValues_(), indented_(false) {}
|
|
|
+
|
|
|
+ void StyledStreamWriter::write(OStream &out, const Value &root) {
|
|
|
+ document_ = &out;
|
|
|
+ addChildValues_ = false;
|
|
|
+ indentString_.clear();
|
|
|
+ indented_ = true;
|
|
|
+ writeCommentBeforeValue(root);
|
|
|
+ if (!indented_)
|
|
|
+ writeIndent();
|
|
|
+ indented_ = true;
|
|
|
+ writeValue(root);
|
|
|
+ writeCommentAfterValueOnSameLine(root);
|
|
|
+ *document_ << "\n";
|
|
|
+ document_ = nullptr; // Forget the stream, for safety.
|
|
|
+ }
|
|
|
+
|
|
|
+ void StyledStreamWriter::writeValue(const Value &value) {
|
|
|
+ switch (value.type()) {
|
|
|
+ case nullValue:
|
|
|
+ pushValue("null");
|
|
|
+ break;
|
|
|
+ case intValue:
|
|
|
+ pushValue(valueToString(value.asLargestInt()));
|
|
|
+ break;
|
|
|
+ case uintValue:
|
|
|
+ pushValue(valueToString(value.asLargestUInt()));
|
|
|
+ break;
|
|
|
+ case realValue:
|
|
|
+ pushValue(valueToString(value.asDouble()));
|
|
|
+ break;
|
|
|
+ case stringValue: {
|
|
|
+ // Is NULL possible for value.string_? No.
|
|
|
+ char const *str;
|
|
|
+ char const *end;
|
|
|
+ bool ok = value.getString(&str, &end);
|
|
|
+ if (ok)
|
|
|
+ pushValue(valueToQuotedStringN(str, static_cast<unsigned>(end - str)));
|
|
|
+ else
|
|
|
+ pushValue("");
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ case booleanValue:
|
|
|
+ pushValue(valueToString(value.asBool()));
|
|
|
+ break;
|
|
|
+ case arrayValue:
|
|
|
+ writeArrayValue(value);
|
|
|
+ break;
|
|
|
+ case objectValue: {
|
|
|
+ Value::Members members(value.getMemberNames());
|
|
|
+ if (members.empty())
|
|
|
+ pushValue("{}");
|
|
|
+ else {
|
|
|
+ writeWithIndent("{");
|
|
|
+ indent();
|
|
|
+ auto it = members.begin();
|
|
|
+ for (;;) {
|
|
|
+ const String &name = *it;
|
|
|
+ const Value &childValue = value[name];
|
|
|
+ writeCommentBeforeValue(childValue);
|
|
|
+ writeWithIndent(valueToQuotedString(name.c_str()));
|
|
|
+ *document_ << " : ";
|
|
|
+ writeValue(childValue);
|
|
|
+ if (++it == members.end()) {
|
|
|
+ writeCommentAfterValueOnSameLine(childValue);
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ *document_ << ",";
|
|
|
+ writeCommentAfterValueOnSameLine(childValue);
|
|
|
+ }
|
|
|
+ unindent();
|
|
|
+ writeWithIndent("}");
|
|
|
+ }
|
|
|
+ }
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ void StyledStreamWriter::writeArrayValue(const Value &value) {
|
|
|
+ unsigned size = value.size();
|
|
|
+ if (size == 0)
|
|
|
+ pushValue("[]");
|
|
|
+ else {
|
|
|
+ bool isArrayMultiLine = isMultilineArray(value);
|
|
|
+ if (isArrayMultiLine) {
|
|
|
+ writeWithIndent("[");
|
|
|
+ indent();
|
|
|
+ bool hasChildValue = !childValues_.empty();
|
|
|
+ unsigned index = 0;
|
|
|
+ for (;;) {
|
|
|
+ const Value &childValue = value[index];
|
|
|
+ writeCommentBeforeValue(childValue);
|
|
|
+ if (hasChildValue)
|
|
|
+ writeWithIndent(childValues_[index]);
|
|
|
+ else {
|
|
|
+ if (!indented_)
|
|
|
+ writeIndent();
|
|
|
+ indented_ = true;
|
|
|
+ writeValue(childValue);
|
|
|
+ indented_ = false;
|
|
|
+ }
|
|
|
+ if (++index == size) {
|
|
|
+ writeCommentAfterValueOnSameLine(childValue);
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ *document_ << ",";
|
|
|
+ writeCommentAfterValueOnSameLine(childValue);
|
|
|
+ }
|
|
|
+ unindent();
|
|
|
+ writeWithIndent("]");
|
|
|
+ } else // output on a single line
|
|
|
+ {
|
|
|
+ assert(childValues_.size() == size);
|
|
|
+ *document_ << "[ ";
|
|
|
+ for (unsigned index = 0; index < size; ++index) {
|
|
|
+ if (index > 0)
|
|
|
+ *document_ << ", ";
|
|
|
+ *document_ << childValues_[index];
|
|
|
+ }
|
|
|
+ *document_ << " ]";
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ bool StyledStreamWriter::isMultilineArray(const Value &value) {
|
|
|
+ ArrayIndex const size = value.size();
|
|
|
+ bool isMultiLine = size * 3 >= rightMargin_;
|
|
|
+ childValues_.clear();
|
|
|
+ for (ArrayIndex index = 0; index < size && !isMultiLine; ++index) {
|
|
|
+ const Value &childValue = value[index];
|
|
|
+ isMultiLine = ((childValue.isArray() || childValue.isObject()) &&
|
|
|
+ !childValue.empty());
|
|
|
+ }
|
|
|
+ if (!isMultiLine) // check if line length > max line length
|
|
|
+ {
|
|
|
+ childValues_.reserve(size);
|
|
|
+ addChildValues_ = true;
|
|
|
+ ArrayIndex lineLength = 4 + (size - 1) * 2; // '[ ' + ', '*n + ' ]'
|
|
|
+ for (ArrayIndex index = 0; index < size; ++index) {
|
|
|
+ if (hasCommentForValue(value[index])) {
|
|
|
+ isMultiLine = true;
|
|
|
+ }
|
|
|
+ writeValue(value[index]);
|
|
|
+ lineLength += static_cast<ArrayIndex>(childValues_[index].length());
|
|
|
+ }
|
|
|
+ addChildValues_ = false;
|
|
|
+ isMultiLine = isMultiLine || lineLength >= rightMargin_;
|
|
|
+ }
|
|
|
+ return isMultiLine;
|
|
|
+ }
|
|
|
+
|
|
|
+ void StyledStreamWriter::pushValue(const String &value) {
|
|
|
+ if (addChildValues_)
|
|
|
+ childValues_.push_back(value);
|
|
|
+ else
|
|
|
+ *document_ << value;
|
|
|
+ }
|
|
|
+
|
|
|
+ void StyledStreamWriter::writeIndent() {
|
|
|
+ // blep intended this to look at the so-far-written string
|
|
|
+ // to determine whether we are already indented, but
|
|
|
+ // with a stream we cannot do that. So we rely on some saved state.
|
|
|
+ // The caller checks indented_.
|
|
|
+ *document_ << '\n' << indentString_;
|
|
|
+ }
|
|
|
+
|
|
|
+ void StyledStreamWriter::writeWithIndent(const String &value) {
|
|
|
+ if (!indented_)
|
|
|
+ writeIndent();
|
|
|
+ *document_ << value;
|
|
|
+ indented_ = false;
|
|
|
+ }
|
|
|
+
|
|
|
+ void StyledStreamWriter::indent() { indentString_ += indentation_; }
|
|
|
+
|
|
|
+ void StyledStreamWriter::unindent() {
|
|
|
+ assert(indentString_.size() >= indentation_.size());
|
|
|
+ indentString_.resize(indentString_.size() - indentation_.size());
|
|
|
+ }
|
|
|
+
|
|
|
+ void StyledStreamWriter::writeCommentBeforeValue(const Value &root) {
|
|
|
+ if (!root.hasComment(commentBefore))
|
|
|
+ return;
|
|
|
+
|
|
|
+ if (!indented_)
|
|
|
+ writeIndent();
|
|
|
+ const String &comment = root.getComment(commentBefore);
|
|
|
+ String::const_iterator iter = comment.begin();
|
|
|
+ while (iter != comment.end()) {
|
|
|
+ *document_ << *iter;
|
|
|
+ if (*iter == '\n' && ((iter + 1) != comment.end() && *(iter + 1) == '/'))
|
|
|
+ // writeIndent(); // would include newline
|
|
|
+ *document_ << indentString_;
|
|
|
+ ++iter;
|
|
|
+ }
|
|
|
+ indented_ = false;
|
|
|
+ }
|
|
|
+
|
|
|
+ void StyledStreamWriter::writeCommentAfterValueOnSameLine(const Value &root) {
|
|
|
+ if (root.hasComment(commentAfterOnSameLine))
|
|
|
+ *document_ << ' ' << root.getComment(commentAfterOnSameLine);
|
|
|
+
|
|
|
+ if (root.hasComment(commentAfter)) {
|
|
|
+ writeIndent();
|
|
|
+ *document_ << root.getComment(commentAfter);
|
|
|
+ }
|
|
|
+ indented_ = false;
|
|
|
+ }
|
|
|
+
|
|
|
+ bool StyledStreamWriter::hasCommentForValue(const Value &value) {
|
|
|
+ return value.hasComment(commentBefore) ||
|
|
|
+ value.hasComment(commentAfterOnSameLine) ||
|
|
|
+ value.hasComment(commentAfter);
|
|
|
+ }
|
|
|
+
|
|
|
+//////////////////////////
|
|
|
+// BuiltStyledStreamWriter
|
|
|
+
|
|
|
+/// Scoped enums are not available until C++11.
|
|
|
+ struct CommentStyle {
|
|
|
+ /// Decide whether to write comments.
|
|
|
+ enum Enum {
|
|
|
+ None, ///< Drop all comments.
|
|
|
+ Most, ///< Recover odd behavior of previous versions (not implemented yet).
|
|
|
+ All ///< Keep all comments.
|
|
|
+ };
|
|
|
+ };
|
|
|
+
|
|
|
+ struct BuiltStyledStreamWriter : public StreamWriter {
|
|
|
+ BuiltStyledStreamWriter(String indentation, CommentStyle::Enum cs,
|
|
|
+ String colonSymbol, String nullSymbol,
|
|
|
+ String endingLineFeedSymbol, bool useSpecialFloats,
|
|
|
+ bool emitUTF8, unsigned int precision,
|
|
|
+ PrecisionType precisionType);
|
|
|
+
|
|
|
+ int write(Value const &root, OStream *sout) override;
|
|
|
+
|
|
|
+ private:
|
|
|
+ void writeValue(Value const &value);
|
|
|
+
|
|
|
+ void writeArrayValue(Value const &value);
|
|
|
+
|
|
|
+ bool isMultilineArray(Value const &value);
|
|
|
+
|
|
|
+ void pushValue(String const &value);
|
|
|
+
|
|
|
+ void writeIndent();
|
|
|
+
|
|
|
+ void writeWithIndent(String const &value);
|
|
|
+
|
|
|
+ void indent();
|
|
|
+
|
|
|
+ void unindent();
|
|
|
+
|
|
|
+ void writeCommentBeforeValue(Value const &root);
|
|
|
+
|
|
|
+ void writeCommentAfterValueOnSameLine(Value const &root);
|
|
|
+
|
|
|
+ static bool hasCommentForValue(const Value &value);
|
|
|
+
|
|
|
+ using ChildValues = std::vector<String>;
|
|
|
+
|
|
|
+ ChildValues childValues_;
|
|
|
+ String indentString_;
|
|
|
+ unsigned int rightMargin_;
|
|
|
+ String indentation_;
|
|
|
+ CommentStyle::Enum cs_;
|
|
|
+ String colonSymbol_;
|
|
|
+ String nullSymbol_;
|
|
|
+ String endingLineFeedSymbol_;
|
|
|
+ bool addChildValues_: 1;
|
|
|
+ bool indented_: 1;
|
|
|
+ bool useSpecialFloats_: 1;
|
|
|
+ bool emitUTF8_: 1;
|
|
|
+ unsigned int precision_;
|
|
|
+ PrecisionType precisionType_;
|
|
|
+ };
|
|
|
+
|
|
|
+ BuiltStyledStreamWriter::BuiltStyledStreamWriter(
|
|
|
+ String indentation, CommentStyle::Enum cs, String colonSymbol,
|
|
|
+ String nullSymbol, String endingLineFeedSymbol, bool useSpecialFloats,
|
|
|
+ bool emitUTF8, unsigned int precision, PrecisionType precisionType)
|
|
|
+ : rightMargin_(74), indentation_(std::move(indentation)), cs_(cs),
|
|
|
+ colonSymbol_(std::move(colonSymbol)), nullSymbol_(std::move(nullSymbol)),
|
|
|
+ endingLineFeedSymbol_(std::move(endingLineFeedSymbol)),
|
|
|
+ addChildValues_(false), indented_(false),
|
|
|
+ useSpecialFloats_(useSpecialFloats), emitUTF8_(emitUTF8),
|
|
|
+ precision_(precision), precisionType_(precisionType) {}
|
|
|
+
|
|
|
+ int BuiltStyledStreamWriter::write(Value const &root, OStream *sout) {
|
|
|
+ sout_ = sout;
|
|
|
+ addChildValues_ = false;
|
|
|
+ indented_ = true;
|
|
|
+ indentString_.clear();
|
|
|
+ writeCommentBeforeValue(root);
|
|
|
+ if (!indented_)
|
|
|
+ writeIndent();
|
|
|
+ indented_ = true;
|
|
|
+ writeValue(root);
|
|
|
+ writeCommentAfterValueOnSameLine(root);
|
|
|
+ *sout_ << endingLineFeedSymbol_;
|
|
|
+ sout_ = nullptr;
|
|
|
+ return 0;
|
|
|
+ }
|
|
|
+
|
|
|
+ void BuiltStyledStreamWriter::writeValue(Value const &value) {
|
|
|
+ switch (value.type()) {
|
|
|
+ case nullValue:
|
|
|
+ pushValue(nullSymbol_);
|
|
|
+ break;
|
|
|
+ case intValue:
|
|
|
+ pushValue(valueToString(value.asLargestInt()));
|
|
|
+ break;
|
|
|
+ case uintValue:
|
|
|
+ pushValue(valueToString(value.asLargestUInt()));
|
|
|
+ break;
|
|
|
+ case realValue:
|
|
|
+ pushValue(valueToString(value.asDouble(), useSpecialFloats_, precision_,
|
|
|
+ precisionType_));
|
|
|
+ break;
|
|
|
+ case stringValue: {
|
|
|
+ // Is NULL is possible for value.string_? No.
|
|
|
+ char const *str;
|
|
|
+ char const *end;
|
|
|
+ bool ok = value.getString(&str, &end);
|
|
|
+ if (ok)
|
|
|
+ pushValue(valueToQuotedStringN(str, static_cast<unsigned>(end - str),
|
|
|
+ emitUTF8_));
|
|
|
+ else
|
|
|
+ pushValue("");
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ case booleanValue:
|
|
|
+ pushValue(valueToString(value.asBool()));
|
|
|
+ break;
|
|
|
+ case arrayValue:
|
|
|
+ writeArrayValue(value);
|
|
|
+ break;
|
|
|
+ case objectValue: {
|
|
|
+ Value::Members members(value.getMemberNames());
|
|
|
+ if (members.empty())
|
|
|
+ pushValue("{}");
|
|
|
+ else {
|
|
|
+ writeWithIndent("{");
|
|
|
+ indent();
|
|
|
+ auto it = members.begin();
|
|
|
+ for (;;) {
|
|
|
+ String const &name = *it;
|
|
|
+ Value const &childValue = value[name];
|
|
|
+ writeCommentBeforeValue(childValue);
|
|
|
+ writeWithIndent(valueToQuotedStringN(
|
|
|
+ name.data(), static_cast<unsigned>(name.length()), emitUTF8_));
|
|
|
+ *sout_ << colonSymbol_;
|
|
|
+ writeValue(childValue);
|
|
|
+ if (++it == members.end()) {
|
|
|
+ writeCommentAfterValueOnSameLine(childValue);
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ *sout_ << ",";
|
|
|
+ writeCommentAfterValueOnSameLine(childValue);
|
|
|
+ }
|
|
|
+ unindent();
|
|
|
+ writeWithIndent("}");
|
|
|
+ }
|
|
|
+ }
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ void BuiltStyledStreamWriter::writeArrayValue(Value const &value) {
|
|
|
+ unsigned size = value.size();
|
|
|
+ if (size == 0)
|
|
|
+ pushValue("[]");
|
|
|
+ else {
|
|
|
+ bool isMultiLine = (cs_ == CommentStyle::All) || isMultilineArray(value);
|
|
|
+ if (isMultiLine) {
|
|
|
+ writeWithIndent("[");
|
|
|
+ indent();
|
|
|
+ bool hasChildValue = !childValues_.empty();
|
|
|
+ unsigned index = 0;
|
|
|
+ for (;;) {
|
|
|
+ Value const &childValue = value[index];
|
|
|
+ writeCommentBeforeValue(childValue);
|
|
|
+ if (hasChildValue)
|
|
|
+ writeWithIndent(childValues_[index]);
|
|
|
+ else {
|
|
|
+ if (!indented_)
|
|
|
+ writeIndent();
|
|
|
+ indented_ = true;
|
|
|
+ writeValue(childValue);
|
|
|
+ indented_ = false;
|
|
|
+ }
|
|
|
+ if (++index == size) {
|
|
|
+ writeCommentAfterValueOnSameLine(childValue);
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ *sout_ << ",";
|
|
|
+ writeCommentAfterValueOnSameLine(childValue);
|
|
|
+ }
|
|
|
+ unindent();
|
|
|
+ writeWithIndent("]");
|
|
|
+ } else // output on a single line
|
|
|
+ {
|
|
|
+ assert(childValues_.size() == size);
|
|
|
+ *sout_ << "[";
|
|
|
+ if (!indentation_.empty())
|
|
|
+ *sout_ << " ";
|
|
|
+ for (unsigned index = 0; index < size; ++index) {
|
|
|
+ if (index > 0)
|
|
|
+ *sout_ << ((!indentation_.empty()) ? ", " : ",");
|
|
|
+ *sout_ << childValues_[index];
|
|
|
+ }
|
|
|
+ if (!indentation_.empty())
|
|
|
+ *sout_ << " ";
|
|
|
+ *sout_ << "]";
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ bool BuiltStyledStreamWriter::isMultilineArray(Value const &value) {
|
|
|
+ ArrayIndex const size = value.size();
|
|
|
+ bool isMultiLine = size * 3 >= rightMargin_;
|
|
|
+ childValues_.clear();
|
|
|
+ for (ArrayIndex index = 0; index < size && !isMultiLine; ++index) {
|
|
|
+ Value const &childValue = value[index];
|
|
|
+ isMultiLine = ((childValue.isArray() || childValue.isObject()) &&
|
|
|
+ !childValue.empty());
|
|
|
+ }
|
|
|
+ if (!isMultiLine) // check if line length > max line length
|
|
|
+ {
|
|
|
+ childValues_.reserve(size);
|
|
|
+ addChildValues_ = true;
|
|
|
+ ArrayIndex lineLength = 4 + (size - 1) * 2; // '[ ' + ', '*n + ' ]'
|
|
|
+ for (ArrayIndex index = 0; index < size; ++index) {
|
|
|
+ if (hasCommentForValue(value[index])) {
|
|
|
+ isMultiLine = true;
|
|
|
+ }
|
|
|
+ writeValue(value[index]);
|
|
|
+ lineLength += static_cast<ArrayIndex>(childValues_[index].length());
|
|
|
+ }
|
|
|
+ addChildValues_ = false;
|
|
|
+ isMultiLine = isMultiLine || lineLength >= rightMargin_;
|
|
|
+ }
|
|
|
+ return isMultiLine;
|
|
|
+ }
|
|
|
+
|
|
|
+ void BuiltStyledStreamWriter::pushValue(String const &value) {
|
|
|
+ if (addChildValues_)
|
|
|
+ childValues_.push_back(value);
|
|
|
+ else
|
|
|
+ *sout_ << value;
|
|
|
+ }
|
|
|
+
|
|
|
+ void BuiltStyledStreamWriter::writeIndent() {
|
|
|
+ // blep intended this to look at the so-far-written string
|
|
|
+ // to determine whether we are already indented, but
|
|
|
+ // with a stream we cannot do that. So we rely on some saved state.
|
|
|
+ // The caller checks indented_.
|
|
|
+
|
|
|
+ if (!indentation_.empty()) {
|
|
|
+ // In this case, drop newlines too.
|
|
|
+ *sout_ << '\n' << indentString_;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ void BuiltStyledStreamWriter::writeWithIndent(String const &value) {
|
|
|
+ if (!indented_)
|
|
|
+ writeIndent();
|
|
|
+ *sout_ << value;
|
|
|
+ indented_ = false;
|
|
|
+ }
|
|
|
+
|
|
|
+ void BuiltStyledStreamWriter::indent() { indentString_ += indentation_; }
|
|
|
+
|
|
|
+ void BuiltStyledStreamWriter::unindent() {
|
|
|
+ assert(indentString_.size() >= indentation_.size());
|
|
|
+ indentString_.resize(indentString_.size() - indentation_.size());
|
|
|
+ }
|
|
|
+
|
|
|
+ void BuiltStyledStreamWriter::writeCommentBeforeValue(Value const &root) {
|
|
|
+ if (cs_ == CommentStyle::None)
|
|
|
+ return;
|
|
|
+ if (!root.hasComment(commentBefore))
|
|
|
+ return;
|
|
|
+
|
|
|
+ if (!indented_)
|
|
|
+ writeIndent();
|
|
|
+ const String &comment = root.getComment(commentBefore);
|
|
|
+ String::const_iterator iter = comment.begin();
|
|
|
+ while (iter != comment.end()) {
|
|
|
+ *sout_ << *iter;
|
|
|
+ if (*iter == '\n' && ((iter + 1) != comment.end() && *(iter + 1) == '/'))
|
|
|
+ // writeIndent(); // would write extra newline
|
|
|
+ *sout_ << indentString_;
|
|
|
+ ++iter;
|
|
|
+ }
|
|
|
+ indented_ = false;
|
|
|
+ }
|
|
|
+
|
|
|
+ void BuiltStyledStreamWriter::writeCommentAfterValueOnSameLine(
|
|
|
+ Value const &root) {
|
|
|
+ if (cs_ == CommentStyle::None)
|
|
|
+ return;
|
|
|
+ if (root.hasComment(commentAfterOnSameLine))
|
|
|
+ *sout_ << " " + root.getComment(commentAfterOnSameLine);
|
|
|
+
|
|
|
+ if (root.hasComment(commentAfter)) {
|
|
|
+ writeIndent();
|
|
|
+ *sout_ << root.getComment(commentAfter);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+// static
|
|
|
+ bool BuiltStyledStreamWriter::hasCommentForValue(const Value &value) {
|
|
|
+ return value.hasComment(commentBefore) ||
|
|
|
+ value.hasComment(commentAfterOnSameLine) ||
|
|
|
+ value.hasComment(commentAfter);
|
|
|
+ }
|
|
|
+
|
|
|
+///////////////
|
|
|
+// StreamWriter
|
|
|
+
|
|
|
+ StreamWriter::StreamWriter() : sout_(nullptr) {}
|
|
|
+
|
|
|
+ StreamWriter::~StreamWriter() = default;
|
|
|
+
|
|
|
+ StreamWriter::Factory::~Factory() = default;
|
|
|
+
|
|
|
+ StreamWriterBuilder::StreamWriterBuilder() { setDefaults(&settings_); }
|
|
|
+
|
|
|
+ StreamWriterBuilder::~StreamWriterBuilder() = default;
|
|
|
+
|
|
|
+ StreamWriter *StreamWriterBuilder::newStreamWriter() const {
|
|
|
+ const String indentation = settings_["indentation"].asString();
|
|
|
+ const String cs_str = settings_["commentStyle"].asString();
|
|
|
+ const String pt_str = settings_["precisionType"].asString();
|
|
|
+ const bool eyc = settings_["enableYAMLCompatibility"].asBool();
|
|
|
+ const bool dnp = settings_["dropNullPlaceholders"].asBool();
|
|
|
+ const bool usf = settings_["useSpecialFloats"].asBool();
|
|
|
+ const bool emitUTF8 = settings_["emitUTF8"].asBool();
|
|
|
+ unsigned int pre = settings_["precision"].asUInt();
|
|
|
+ CommentStyle::Enum cs = CommentStyle::All;
|
|
|
+ if (cs_str == "All") {
|
|
|
+ cs = CommentStyle::All;
|
|
|
+ } else if (cs_str == "None") {
|
|
|
+ cs = CommentStyle::None;
|
|
|
+ } else {
|
|
|
+ throwRuntimeError("commentStyle must be 'All' or 'None'");
|
|
|
+ }
|
|
|
+ PrecisionType precisionType(significantDigits);
|
|
|
+ if (pt_str == "significant") {
|
|
|
+ precisionType = PrecisionType::significantDigits;
|
|
|
+ } else if (pt_str == "decimal") {
|
|
|
+ precisionType = PrecisionType::decimalPlaces;
|
|
|
+ } else {
|
|
|
+ throwRuntimeError("precisionType must be 'significant' or 'decimal'");
|
|
|
+ }
|
|
|
+ String colonSymbol = " : ";
|
|
|
+ if (eyc) {
|
|
|
+ colonSymbol = ": ";
|
|
|
+ } else if (indentation.empty()) {
|
|
|
+ colonSymbol = ":";
|
|
|
+ }
|
|
|
+ String nullSymbol = "null";
|
|
|
+ if (dnp) {
|
|
|
+ nullSymbol.clear();
|
|
|
+ }
|
|
|
+ if (pre > 17)
|
|
|
+ pre = 17;
|
|
|
+ String endingLineFeedSymbol;
|
|
|
+ return new BuiltStyledStreamWriter(indentation, cs, colonSymbol, nullSymbol,
|
|
|
+ endingLineFeedSymbol, usf, emitUTF8, pre,
|
|
|
+ precisionType);
|
|
|
+ }
|
|
|
+
|
|
|
+ bool StreamWriterBuilder::validate(Json::Value *invalid) const {
|
|
|
+ static const auto &valid_keys = *new std::set<String>{
|
|
|
+ "indentation",
|
|
|
+ "commentStyle",
|
|
|
+ "enableYAMLCompatibility",
|
|
|
+ "dropNullPlaceholders",
|
|
|
+ "useSpecialFloats",
|
|
|
+ "emitUTF8",
|
|
|
+ "precision",
|
|
|
+ "precisionType",
|
|
|
+ };
|
|
|
+ for (auto si = settings_.begin(); si != settings_.end(); ++si) {
|
|
|
+ auto key = si.name();
|
|
|
+ if (valid_keys.count(key))
|
|
|
+ continue;
|
|
|
+ if (invalid)
|
|
|
+ (*invalid)[std::move(key)] = *si;
|
|
|
+ else
|
|
|
+ return false;
|
|
|
+ }
|
|
|
+ return invalid ? invalid->empty() : true;
|
|
|
+ }
|
|
|
+
|
|
|
+ Value &StreamWriterBuilder::operator[](const String &key) {
|
|
|
+ return settings_[key];
|
|
|
+ }
|
|
|
+
|
|
|
+// static
|
|
|
+ void StreamWriterBuilder::setDefaults(Json::Value *settings) {
|
|
|
+ //! [StreamWriterBuilderDefaults]
|
|
|
+ (*settings)["commentStyle"] = "All";
|
|
|
+ (*settings)["indentation"] = "\t";
|
|
|
+ (*settings)["enableYAMLCompatibility"] = false;
|
|
|
+ (*settings)["dropNullPlaceholders"] = false;
|
|
|
+ (*settings)["useSpecialFloats"] = false;
|
|
|
+ (*settings)["emitUTF8"] = false;
|
|
|
+ (*settings)["precision"] = 17;
|
|
|
+ (*settings)["precisionType"] = "significant";
|
|
|
+ //! [StreamWriterBuilderDefaults]
|
|
|
+ }
|
|
|
+
|
|
|
+ String writeString(StreamWriter::Factory const &factory, Value const &root) {
|
|
|
+ OStringStream sout;
|
|
|
+ StreamWriterPtr const writer(factory.newStreamWriter());
|
|
|
+ writer->write(root, &sout);
|
|
|
+ return sout.str();
|
|
|
+ }
|
|
|
+
|
|
|
+ OStream &operator<<(OStream &sout, Value const &root) {
|
|
|
+ StreamWriterBuilder builder;
|
|
|
+ StreamWriterPtr const writer(builder.newStreamWriter());
|
|
|
+ writer->write(root, &sout);
|
|
|
+ return sout;
|
|
|
+ }
|
|
|
+
|
|
|
+} // namespace Json
|
|
|
+
|
|
|
+// //////////////////////////////////////////////////////////////////////
|
|
|
+// End of content of file: src/lib_json/json_writer.cpp
|
|
|
+// //////////////////////////////////////////////////////////////////////
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|