| /// 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/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*& current) { |
| *--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, unsigned int precision) { |
| for (; begin != end; --end) { |
| if (*(end - 1) != '0') { |
| return end; |
| } |
| // Don't delete the last zero before the decimal point. |
| if (begin != (end - 1) && begin != (end - 2) && *(end - 2) == '.') { |
| if (precision) { |
| return end; |
| } |
| return end - 2; |
| } |
| } |
| 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 <cmath> |
| #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::istreambuf_iterator<char>(is), {}); |
| 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 set |
| 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 set |
| 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)) { |
| if (value == std::numeric_limits<double>::max()) |
| value = std::numeric_limits<double>::infinity(); |
| else if (value == std::numeric_limits<double>::lowest()) |
| value = -std::numeric_limits<double>::infinity(); |
| else if (!std::isinf(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& current, |
| 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& current, |
| 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& current, Location end, |
| unsigned int& unicode); |
| bool decodeUnicodeEscapeSequence(Token& token, Location& current, |
| 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& currentValue(); |
| 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 set |
| 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 set |
| 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, meaning 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)) { |
| if (value == std::numeric_limits<double>::max()) |
| value = std::numeric_limits<double>::infinity(); |
| else if (value == std::numeric_limits<double>::lowest()) |
| value = -std::numeric_limits<double>::infinity(); |
| else if (!std::isinf(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& current, |
| 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& current, |
| 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)[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& current) |
| : 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& current) |
| : 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& current) |
| : 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) noexcept |
| : 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) noexcept { |
| 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) noexcept { |
| 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) noexcept { |
| 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) |
| for (ArrayIndex i = oldSize; i < newSize; ++i) |
| (*this)[i]; |
| 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) noexcept |
| : 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) noexcept { |
| 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 (slot >= CommentPlacement::numberOfCommentPlacement) |
| return; |
| if (!ptr_) |
| ptr_ = std::unique_ptr<Array>(new Array()); |
| (*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()); |
| |
| // 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"; |
| } |
| |
| // strip the zero padding from the right |
| if (precisionType == PrecisionType::decimalPlaces) { |
| buffer.erase(fixZerosInTheEnd(buffer.begin(), buffer.end(), precision), |
| buffer.end()); |
| } |
| |
| 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, size_t 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, 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<size_t>(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(), 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<size_t>(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) { |
| size_t size = value.size(); |
| if (size == 0) |
| pushValue("[]"); |
| else { |
| bool isArrayMultiLine = isMultilineArray(value); |
| if (isArrayMultiLine) { |
| writeWithIndent("["); |
| indent(); |
| bool hasChildValue = !childValues_.empty(); |
| ArrayIndex 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 (size_t 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<size_t>(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<size_t>(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(), 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)[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 |
| // ////////////////////////////////////////////////////////////////////// |
| |
| |
| |
| |
| |