From 9ff0cc0f0212cb1e85b83ce5d2907ba9531b6cae Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sat, 25 Feb 2017 16:43:15 +0100 Subject: [PATCH 001/530] :memo: updated documentation --- README.md | 2 +- doc/examples/README.link | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/README.md b/README.md index 1334180b..8dea44aa 100644 --- a/README.md +++ b/README.md @@ -5,7 +5,7 @@ [![Coverage Status](https://img.shields.io/coveralls/nlohmann/json.svg)](https://coveralls.io/r/nlohmann/json) [![Coverity Scan Build Status](https://scan.coverity.com/projects/5550/badge.svg)](https://scan.coverity.com/projects/nlohmann-json) [![Codacy Badge](https://api.codacy.com/project/badge/Grade/f3732b3327e34358a0e9d1fe9f661f08)](https://www.codacy.com/app/nlohmann/json?utm_source=github.com&utm_medium=referral&utm_content=nlohmann/json&utm_campaign=Badge_Grade) -[![Try online](https://img.shields.io/badge/try-online-blue.svg)](http://melpon.org/wandbox/permlink/4NEU6ZZMoM9lpIex) +[![Try online](https://img.shields.io/badge/try-online-blue.svg)](http://melpon.org/wandbox/permlink/nv9fOg0XVVhWmFFy) [![Documentation](https://img.shields.io/badge/docs-doxygen-blue.svg)](http://nlohmann.github.io/json) [![GitHub license](https://img.shields.io/badge/license-MIT-blue.svg)](https://raw.githubusercontent.com/nlohmann/json/master/LICENSE.MIT) [![Github Releases](https://img.shields.io/github/release/nlohmann/json.svg)](https://github.com/nlohmann/json/releases) diff --git a/doc/examples/README.link b/doc/examples/README.link index d0168aab..a774c8ff 100644 --- a/doc/examples/README.link +++ b/doc/examples/README.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file From f1cd15ce7e66518007101e73bb91c87b27c13512 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 26 Feb 2017 11:22:18 +0100 Subject: [PATCH 002/530] :zap: avoid copying a string --- src/json.hpp | 2 +- src/json.hpp.re2c | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 6dfc1831..6be0a2bf 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -12477,7 +12477,7 @@ basic_json_parser_74: // the valid JSON Patch operations enum class patch_operations {add, remove, replace, move, copy, test, invalid}; - const auto get_op = [](const std::string op) + const auto get_op = [](const std::string & op) { if (op == "add") { diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index a42daba6..0d6e9dab 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -11511,7 +11511,7 @@ class basic_json // the valid JSON Patch operations enum class patch_operations {add, remove, replace, move, copy, test, invalid}; - const auto get_op = [](const std::string op) + const auto get_op = [](const std::string & op) { if (op == "add") { From d1b30250d6c75cc5ad521a7ddb2bf18563b4721c Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 26 Feb 2017 11:50:52 +0100 Subject: [PATCH 003/530] :white_check_mark: added missing tests --- test/src/unit-constructor1.cpp | 9 +++++++++ test/src/unit-conversions.cpp | 4 ++++ test/src/unit-regression.cpp | 5 ++++- 3 files changed, 17 insertions(+), 1 deletion(-) diff --git a/test/src/unit-constructor1.cpp b/test/src/unit-constructor1.cpp index 6bba7019..b3354b58 100644 --- a/test/src/unit-constructor1.cpp +++ b/test/src/unit-constructor1.cpp @@ -699,6 +699,15 @@ TEST_CASE("constructors") json j(n); CHECK(j.type() == json::value_t::number_float); } + + SECTION("infinity") + { + // infinity is stored as null + // should change in the future: https://github.com/nlohmann/json/issues/388 + json::number_float_t n(std::numeric_limits::infinity()); + json j(n); + CHECK(j.type() == json::value_t::null); + } } SECTION("create a floating-point number (implicit)") diff --git a/test/src/unit-conversions.cpp b/test/src/unit-conversions.cpp index 106e37de..e4545fbe 100644 --- a/test/src/unit-conversions.cpp +++ b/test/src/unit-conversions.cpp @@ -182,6 +182,10 @@ TEST_CASE("value conversion") std::vector v; CHECK_THROWS_AS(nlohmann::from_json(j, v), std::logic_error); CHECK(v.capacity() == j.size()); + + // make sure all values are properly copied + std::vector v2 = json({1, 2, 3, 4, 5, 6, 7, 8, 9, 10}); + CHECK(v2.size() == 10); } #endif } diff --git a/test/src/unit-regression.cpp b/test/src/unit-regression.cpp index 83bb371a..7980371b 100644 --- a/test/src/unit-regression.cpp +++ b/test/src/unit-regression.cpp @@ -63,7 +63,7 @@ TEST_CASE("regression tests") SECTION("pull request #71 - handle enum type") { - enum { t = 0, u = 1}; + enum { t = 0, u = 102}; json j = json::array(); j.push_back(t); @@ -73,6 +73,9 @@ TEST_CASE("regression tests") auto anon_enum_value = j2.get(); CHECK(u == anon_enum_value); + // check if the actual value was stored + CHECK(j2 == 102); + static_assert(std::is_same::value, ""); j.push_back(json::object( From bf4d744d1a31b83eaaf7ae8ffa6778400d8d3340 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 26 Feb 2017 14:34:58 +0100 Subject: [PATCH 004/530] :white_check_mark: more tests for meta() call --- test/src/unit-meta.cpp | 10 +++++++--- 1 file changed, 7 insertions(+), 3 deletions(-) diff --git a/test/src/unit-meta.cpp b/test/src/unit-meta.cpp index 8c614a6c..2e4cce06 100644 --- a/test/src/unit-meta.cpp +++ b/test/src/unit-meta.cpp @@ -33,10 +33,14 @@ using nlohmann::json; TEST_CASE("version information") { - SECTION("version()") + SECTION("meta()") { - CHECK(json::meta()["name"] == "JSON for Modern C++"); - CHECK(json::meta()["version"] == json( + json j = json::meta(); + + CHECK(j["name"] == "JSON for Modern C++"); + CHECK(j["copyright"] == "(C) 2013-2017 Niels Lohmann"); + CHECK(j["url"] == "https://github.com/nlohmann/json"); + CHECK(j["version"] == json( { {"string", "2.1.1"}, {"major", 2}, From ae155c47347d8412c11b5888727a76b4b24fcd3a Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 26 Feb 2017 14:45:41 +0100 Subject: [PATCH 005/530] :lipstick: cleanup --- src/json.hpp | 5 +---- src/json.hpp.re2c | 5 +---- 2 files changed, 2 insertions(+), 8 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 6be0a2bf..7d2433d4 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -1136,10 +1136,7 @@ class basic_json result["url"] = "https://github.com/nlohmann/json"; result["version"] = { - {"string", "2.1.1"}, - {"major", 2}, - {"minor", 1}, - {"patch", 1} + {"string", "2.1.1"}, {"major", 2}, {"minor", 1}, {"patch", 1} }; #ifdef _WIN32 diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 0d6e9dab..5fa876d7 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -1136,10 +1136,7 @@ class basic_json result["url"] = "https://github.com/nlohmann/json"; result["version"] = { - {"string", "2.1.1"}, - {"major", 2}, - {"minor", 1}, - {"patch", 1} + {"string", "2.1.1"}, {"major", 2}, {"minor", 1}, {"patch", 1} }; #ifdef _WIN32 From bd0326cbc108326f7ebc5f0af049a60aa5a1c87d Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 26 Feb 2017 16:55:54 +0100 Subject: [PATCH 006/530] :zap: micro-optimizations for dump() Added separate code paths for normal output and pritty-printed output. This allowed to remove most of the ifs along the way. Benchmarks and cachegrind suggest a 10% performance improvement. --- src/json.hpp | 96 ++++++++++++++++++++++++++--------------------- src/json.hpp.re2c | 96 ++++++++++++++++++++++++++--------------------- 2 files changed, 108 insertions(+), 84 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 7d2433d4..3c41b41a 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -8414,9 +8414,6 @@ class basic_json const unsigned int indent_step, const unsigned int current_indent = 0) const { - // variable to hold indentation for recursive calls - unsigned int new_indent = current_indent; - switch (m_type) { case value_t::object: @@ -8427,35 +8424,43 @@ class basic_json return; } - o << "{"; - - // increase indentation if (pretty_print) { - new_indent += indent_step; - o << "\n"; - } + o << "{\n"; - for (auto i = m_value.object->cbegin(); i != m_value.object->cend(); ++i) - { - if (i != m_value.object->cbegin()) + // variable to hold indentation for recursive calls + const auto new_indent = current_indent + indent_step; + const std::string indent_string = string_t(new_indent, ' '); + + for (auto i = m_value.object->cbegin(); i != m_value.object->cend(); ++i) { - o << (pretty_print ? ",\n" : ","); + if (i != m_value.object->cbegin()) + { + o << ",\n"; + } + o << indent_string << '\"' << escape_string(i->first) << "\": "; + i->second.dump(o, true, indent_step, new_indent); } - o << string_t(new_indent, ' ') << "\"" - << escape_string(i->first) << "\":" - << (pretty_print ? " " : ""); - i->second.dump(o, pretty_print, indent_step, new_indent); - } - // decrease indentation - if (pretty_print) + o << '\n' << string_t(current_indent, ' ') + '}'; + } + else { - new_indent -= indent_step; - o << "\n"; + o << '{'; + + for (auto i = m_value.object->cbegin(); i != m_value.object->cend(); ++i) + { + if (i != m_value.object->cbegin()) + { + o << ','; + } + o << '\"' << escape_string(i->first) << "\":"; + i->second.dump(o, false, indent_step, current_indent); + } + + o << '}'; } - o << string_t(new_indent, ' ') + "}"; return; } @@ -8467,39 +8472,46 @@ class basic_json return; } - o << "["; - - // increase indentation if (pretty_print) { - new_indent += indent_step; - o << "\n"; - } + o << "[\n"; - for (auto i = m_value.array->cbegin(); i != m_value.array->cend(); ++i) - { - if (i != m_value.array->cbegin()) + // variable to hold indentation for recursive calls + const auto new_indent = current_indent + indent_step; + const std::string indent_string = string_t(new_indent, ' '); + + for (auto i = m_value.array->cbegin(); i != m_value.array->cend() - 1; ++i) { - o << (pretty_print ? ",\n" : ","); + o << indent_string; + i->dump(o, true, indent_step, new_indent); + o << ",\n"; } - o << string_t(new_indent, ' '); - i->dump(o, pretty_print, indent_step, new_indent); - } - // decrease indentation - if (pretty_print) + o << indent_string; + assert(not m_value.array->empty()); + m_value.array->back().dump(o, true, indent_step, new_indent); + + o << '\n' << string_t(current_indent, ' ') << ']'; + } + else { - new_indent -= indent_step; - o << "\n"; + o << '['; + for (auto i = m_value.array->cbegin(); i != m_value.array->cend() - 1; ++i) + { + i->dump(o, false, indent_step, current_indent); + o << ','; + } + assert(not m_value.array->empty()); + m_value.array->back().dump(o, false, indent_step, current_indent); + o << ']'; } - o << string_t(new_indent, ' ') << "]"; return; } case value_t::string: { - o << string_t("\"") << escape_string(*m_value.string) << "\""; + o << '\"' << escape_string(*m_value.string) << '\"'; return; } diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 5fa876d7..6412fe8b 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -8414,9 +8414,6 @@ class basic_json const unsigned int indent_step, const unsigned int current_indent = 0) const { - // variable to hold indentation for recursive calls - unsigned int new_indent = current_indent; - switch (m_type) { case value_t::object: @@ -8427,35 +8424,43 @@ class basic_json return; } - o << "{"; - - // increase indentation if (pretty_print) { - new_indent += indent_step; - o << "\n"; - } + o << "{\n"; - for (auto i = m_value.object->cbegin(); i != m_value.object->cend(); ++i) - { - if (i != m_value.object->cbegin()) + // variable to hold indentation for recursive calls + const auto new_indent = current_indent + indent_step; + const std::string indent_string = string_t(new_indent, ' '); + + for (auto i = m_value.object->cbegin(); i != m_value.object->cend(); ++i) { - o << (pretty_print ? ",\n" : ","); + if (i != m_value.object->cbegin()) + { + o << ",\n"; + } + o << indent_string << '\"' << escape_string(i->first) << "\": "; + i->second.dump(o, true, indent_step, new_indent); } - o << string_t(new_indent, ' ') << "\"" - << escape_string(i->first) << "\":" - << (pretty_print ? " " : ""); - i->second.dump(o, pretty_print, indent_step, new_indent); - } - // decrease indentation - if (pretty_print) + o << '\n' << string_t(current_indent, ' ') + '}'; + } + else { - new_indent -= indent_step; - o << "\n"; + o << '{'; + + for (auto i = m_value.object->cbegin(); i != m_value.object->cend(); ++i) + { + if (i != m_value.object->cbegin()) + { + o << ','; + } + o << '\"' << escape_string(i->first) << "\":"; + i->second.dump(o, false, indent_step, current_indent); + } + + o << '}'; } - o << string_t(new_indent, ' ') + "}"; return; } @@ -8467,39 +8472,46 @@ class basic_json return; } - o << "["; - - // increase indentation if (pretty_print) { - new_indent += indent_step; - o << "\n"; - } + o << "[\n"; - for (auto i = m_value.array->cbegin(); i != m_value.array->cend(); ++i) - { - if (i != m_value.array->cbegin()) + // variable to hold indentation for recursive calls + const auto new_indent = current_indent + indent_step; + const std::string indent_string = string_t(new_indent, ' '); + + for (auto i = m_value.array->cbegin(); i != m_value.array->cend() - 1; ++i) { - o << (pretty_print ? ",\n" : ","); + o << indent_string; + i->dump(o, true, indent_step, new_indent); + o << ",\n"; } - o << string_t(new_indent, ' '); - i->dump(o, pretty_print, indent_step, new_indent); - } - // decrease indentation - if (pretty_print) + o << indent_string; + assert(not m_value.array->empty()); + m_value.array->back().dump(o, true, indent_step, new_indent); + + o << '\n' << string_t(current_indent, ' ') << ']'; + } + else { - new_indent -= indent_step; - o << "\n"; + o << '['; + for (auto i = m_value.array->cbegin(); i != m_value.array->cend() - 1; ++i) + { + i->dump(o, false, indent_step, current_indent); + o << ','; + } + assert(not m_value.array->empty()); + m_value.array->back().dump(o, false, indent_step, current_indent); + o << ']'; } - o << string_t(new_indent, ' ') << "]"; return; } case value_t::string: { - o << string_t("\"") << escape_string(*m_value.string) << "\""; + o << '\"' << escape_string(*m_value.string) << '\"'; return; } From b1441f3485355e774f178cce7e2b645fd3f20943 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 26 Feb 2017 20:58:00 +0100 Subject: [PATCH 007/530] :zap: micro-optimizations for dump() Indentation string is recycled to avoid allocations. Comma-separation in objects does not need an if any more. Cachegrind measures 1% performance improvement. --- src/json.hpp | 31 +++++++++++++++++-------------- src/json.hpp.re2c | 31 +++++++++++++++++-------------- 2 files changed, 34 insertions(+), 28 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 3c41b41a..3a638d76 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -8430,34 +8430,36 @@ class basic_json // variable to hold indentation for recursive calls const auto new_indent = current_indent + indent_step; - const std::string indent_string = string_t(new_indent, ' '); + string_t indent_string = string_t(new_indent, ' '); - for (auto i = m_value.object->cbegin(); i != m_value.object->cend(); ++i) + auto i = m_value.object->cbegin(); + for (size_t cnt = 0; cnt < m_value.object->size() - 1; ++cnt, ++i) { - if (i != m_value.object->cbegin()) - { - o << ",\n"; - } o << indent_string << '\"' << escape_string(i->first) << "\": "; i->second.dump(o, true, indent_step, new_indent); + o << ",\n"; } - o << '\n' << string_t(current_indent, ' ') + '}'; + o << indent_string << '\"' << escape_string(i->first) << "\": "; + i->second.dump(o, true, indent_step, new_indent); + + indent_string.resize(current_indent); + o << '\n' << indent_string << '}'; } else { o << '{'; - for (auto i = m_value.object->cbegin(); i != m_value.object->cend(); ++i) + auto i = m_value.object->cbegin(); + for (size_t cnt = 0; cnt < m_value.object->size() - 1; ++cnt, ++i) { - if (i != m_value.object->cbegin()) - { - o << ','; - } o << '\"' << escape_string(i->first) << "\":"; i->second.dump(o, false, indent_step, current_indent); + o << ','; } + o << '\"' << escape_string(i->first) << "\":"; + i->second.dump(o, false, indent_step, current_indent); o << '}'; } @@ -8478,7 +8480,7 @@ class basic_json // variable to hold indentation for recursive calls const auto new_indent = current_indent + indent_step; - const std::string indent_string = string_t(new_indent, ' '); + string_t indent_string = string_t(new_indent, ' '); for (auto i = m_value.array->cbegin(); i != m_value.array->cend() - 1; ++i) { @@ -8491,7 +8493,8 @@ class basic_json assert(not m_value.array->empty()); m_value.array->back().dump(o, true, indent_step, new_indent); - o << '\n' << string_t(current_indent, ' ') << ']'; + indent_string.resize(current_indent); + o << '\n' << indent_string << ']'; } else { diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 6412fe8b..796f8c56 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -8430,34 +8430,36 @@ class basic_json // variable to hold indentation for recursive calls const auto new_indent = current_indent + indent_step; - const std::string indent_string = string_t(new_indent, ' '); + string_t indent_string = string_t(new_indent, ' '); - for (auto i = m_value.object->cbegin(); i != m_value.object->cend(); ++i) + auto i = m_value.object->cbegin(); + for (size_t cnt = 0; cnt < m_value.object->size() - 1; ++cnt, ++i) { - if (i != m_value.object->cbegin()) - { - o << ",\n"; - } o << indent_string << '\"' << escape_string(i->first) << "\": "; i->second.dump(o, true, indent_step, new_indent); + o << ",\n"; } - o << '\n' << string_t(current_indent, ' ') + '}'; + o << indent_string << '\"' << escape_string(i->first) << "\": "; + i->second.dump(o, true, indent_step, new_indent); + + indent_string.resize(current_indent); + o << '\n' << indent_string << '}'; } else { o << '{'; - for (auto i = m_value.object->cbegin(); i != m_value.object->cend(); ++i) + auto i = m_value.object->cbegin(); + for (size_t cnt = 0; cnt < m_value.object->size() - 1; ++cnt, ++i) { - if (i != m_value.object->cbegin()) - { - o << ','; - } o << '\"' << escape_string(i->first) << "\":"; i->second.dump(o, false, indent_step, current_indent); + o << ','; } + o << '\"' << escape_string(i->first) << "\":"; + i->second.dump(o, false, indent_step, current_indent); o << '}'; } @@ -8478,7 +8480,7 @@ class basic_json // variable to hold indentation for recursive calls const auto new_indent = current_indent + indent_step; - const std::string indent_string = string_t(new_indent, ' '); + string_t indent_string = string_t(new_indent, ' '); for (auto i = m_value.array->cbegin(); i != m_value.array->cend() - 1; ++i) { @@ -8491,7 +8493,8 @@ class basic_json assert(not m_value.array->empty()); m_value.array->back().dump(o, true, indent_step, new_indent); - o << '\n' << string_t(current_indent, ' ') << ']'; + indent_string.resize(current_indent); + o << '\n' << indent_string << ']'; } else { From 0f04e42dd5ee53af466e25a82667fe4c3b039775 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Mon, 27 Feb 2017 01:22:24 +0100 Subject: [PATCH 008/530] :zap: micro-optimizations for dump() MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit All ‘<<‘ calls have been replaced by write()/put() calls. The indentation strings needs not to be resized. Cachegrind measures 1% performance improvement. --- src/json.hpp | 92 +++++++++++++++++++++++++++++++++-------------- src/json.hpp.re2c | 92 +++++++++++++++++++++++++++++++++-------------- 2 files changed, 132 insertions(+), 52 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 3a638d76..1094aade 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -8420,47 +8420,69 @@ class basic_json { if (m_value.object->empty()) { - o << "{}"; + o.write("{}", 2); return; } if (pretty_print) { - o << "{\n"; + o.write("{\n", 2); // variable to hold indentation for recursive calls const auto new_indent = current_indent + indent_step; string_t indent_string = string_t(new_indent, ' '); + // first n-1 elements auto i = m_value.object->cbegin(); for (size_t cnt = 0; cnt < m_value.object->size() - 1; ++cnt, ++i) { - o << indent_string << '\"' << escape_string(i->first) << "\": "; + o.write(indent_string.c_str(), new_indent); + o.put('\"'); + const auto s = escape_string(i->first); + o.write(s.c_str(), static_cast(s.size())); + o.write("\": ", 3); i->second.dump(o, true, indent_step, new_indent); - o << ",\n"; + o.write(",\n", 2); } - o << indent_string << '\"' << escape_string(i->first) << "\": "; + // last element + assert(i != m_value.object->cend()); + o.write(indent_string.c_str(), new_indent); + o.put('\"'); + const auto s = escape_string(i->first); + o.write(s.c_str(), static_cast(s.size())); + o.write("\": ", 3); i->second.dump(o, true, indent_step, new_indent); - indent_string.resize(current_indent); - o << '\n' << indent_string << '}'; + o.put('\n'); + o.write(indent_string.c_str(), current_indent); + o.put('}'); } else { - o << '{'; + o.put('{'); + // first n-1 elements auto i = m_value.object->cbegin(); for (size_t cnt = 0; cnt < m_value.object->size() - 1; ++cnt, ++i) { - o << '\"' << escape_string(i->first) << "\":"; + o.put('\"'); + const auto s = escape_string(i->first); + o.write(s.c_str(), static_cast(s.size())); + o.write("\":", 2); i->second.dump(o, false, indent_step, current_indent); - o << ','; + o.put(','); } - o << '\"' << escape_string(i->first) << "\":"; + // last element + assert(i != m_value.object->cend()); + o.put('\"'); + const auto s = escape_string(i->first); + o.write(s.c_str(), static_cast(s.size())); + o.write("\":", 2); i->second.dump(o, false, indent_step, current_indent); - o << '}'; + + o.put('}'); } return; @@ -8470,43 +8492,51 @@ class basic_json { if (m_value.array->empty()) { - o << "[]"; + o.write("[]", 2); return; } if (pretty_print) { - o << "[\n"; + o.write("[\n", 2); // variable to hold indentation for recursive calls const auto new_indent = current_indent + indent_step; string_t indent_string = string_t(new_indent, ' '); + // first n-1 elements for (auto i = m_value.array->cbegin(); i != m_value.array->cend() - 1; ++i) { - o << indent_string; + o.write(indent_string.c_str(), new_indent); i->dump(o, true, indent_step, new_indent); - o << ",\n"; + o.write(",\n", 2); } - o << indent_string; + // last element assert(not m_value.array->empty()); + o.write(indent_string.c_str(), new_indent); m_value.array->back().dump(o, true, indent_step, new_indent); - indent_string.resize(current_indent); - o << '\n' << indent_string << ']'; + o.put('\n'); + o.write(indent_string.c_str(), current_indent); + o.put(']'); } else { - o << '['; + o.put('['); + + // first n-1 elements for (auto i = m_value.array->cbegin(); i != m_value.array->cend() - 1; ++i) { i->dump(o, false, indent_step, current_indent); - o << ','; + o.put(','); } + + // last element assert(not m_value.array->empty()); m_value.array->back().dump(o, false, indent_step, current_indent); - o << ']'; + + o.put(']'); } return; @@ -8514,13 +8544,23 @@ class basic_json case value_t::string: { - o << '\"' << escape_string(*m_value.string) << '\"'; + o.put('\"'); + const auto s = escape_string(*m_value.string); + o.write(s.c_str(), static_cast(s.size())); + o.put('\"'); return; } case value_t::boolean: { - o << (m_value.boolean ? "true" : "false"); + if (m_value.boolean) + { + o.write("true", 4); + } + else + { + o.write("false", 5); + } return; } @@ -8544,13 +8584,13 @@ class basic_json case value_t::discarded: { - o << ""; + o.write("", 11); return; } case value_t::null: { - o << "null"; + o.write("null", 4); return; } } diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 796f8c56..a686a1a7 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -8420,47 +8420,69 @@ class basic_json { if (m_value.object->empty()) { - o << "{}"; + o.write("{}", 2); return; } if (pretty_print) { - o << "{\n"; + o.write("{\n", 2); // variable to hold indentation for recursive calls const auto new_indent = current_indent + indent_step; string_t indent_string = string_t(new_indent, ' '); + // first n-1 elements auto i = m_value.object->cbegin(); for (size_t cnt = 0; cnt < m_value.object->size() - 1; ++cnt, ++i) { - o << indent_string << '\"' << escape_string(i->first) << "\": "; + o.write(indent_string.c_str(), new_indent); + o.put('\"'); + const auto s = escape_string(i->first); + o.write(s.c_str(), static_cast(s.size())); + o.write("\": ", 3); i->second.dump(o, true, indent_step, new_indent); - o << ",\n"; + o.write(",\n", 2); } - o << indent_string << '\"' << escape_string(i->first) << "\": "; + // last element + assert(i != m_value.object->cend()); + o.write(indent_string.c_str(), new_indent); + o.put('\"'); + const auto s = escape_string(i->first); + o.write(s.c_str(), static_cast(s.size())); + o.write("\": ", 3); i->second.dump(o, true, indent_step, new_indent); - indent_string.resize(current_indent); - o << '\n' << indent_string << '}'; + o.put('\n'); + o.write(indent_string.c_str(), current_indent); + o.put('}'); } else { - o << '{'; + o.put('{'); + // first n-1 elements auto i = m_value.object->cbegin(); for (size_t cnt = 0; cnt < m_value.object->size() - 1; ++cnt, ++i) { - o << '\"' << escape_string(i->first) << "\":"; + o.put('\"'); + const auto s = escape_string(i->first); + o.write(s.c_str(), static_cast(s.size())); + o.write("\":", 2); i->second.dump(o, false, indent_step, current_indent); - o << ','; + o.put(','); } - o << '\"' << escape_string(i->first) << "\":"; + // last element + assert(i != m_value.object->cend()); + o.put('\"'); + const auto s = escape_string(i->first); + o.write(s.c_str(), static_cast(s.size())); + o.write("\":", 2); i->second.dump(o, false, indent_step, current_indent); - o << '}'; + + o.put('}'); } return; @@ -8470,43 +8492,51 @@ class basic_json { if (m_value.array->empty()) { - o << "[]"; + o.write("[]", 2); return; } if (pretty_print) { - o << "[\n"; + o.write("[\n", 2); // variable to hold indentation for recursive calls const auto new_indent = current_indent + indent_step; string_t indent_string = string_t(new_indent, ' '); + // first n-1 elements for (auto i = m_value.array->cbegin(); i != m_value.array->cend() - 1; ++i) { - o << indent_string; + o.write(indent_string.c_str(), new_indent); i->dump(o, true, indent_step, new_indent); - o << ",\n"; + o.write(",\n", 2); } - o << indent_string; + // last element assert(not m_value.array->empty()); + o.write(indent_string.c_str(), new_indent); m_value.array->back().dump(o, true, indent_step, new_indent); - indent_string.resize(current_indent); - o << '\n' << indent_string << ']'; + o.put('\n'); + o.write(indent_string.c_str(), current_indent); + o.put(']'); } else { - o << '['; + o.put('['); + + // first n-1 elements for (auto i = m_value.array->cbegin(); i != m_value.array->cend() - 1; ++i) { i->dump(o, false, indent_step, current_indent); - o << ','; + o.put(','); } + + // last element assert(not m_value.array->empty()); m_value.array->back().dump(o, false, indent_step, current_indent); - o << ']'; + + o.put(']'); } return; @@ -8514,13 +8544,23 @@ class basic_json case value_t::string: { - o << '\"' << escape_string(*m_value.string) << '\"'; + o.put('\"'); + const auto s = escape_string(*m_value.string); + o.write(s.c_str(), static_cast(s.size())); + o.put('\"'); return; } case value_t::boolean: { - o << (m_value.boolean ? "true" : "false"); + if (m_value.boolean) + { + o.write("true", 4); + } + else + { + o.write("false", 5); + } return; } @@ -8544,13 +8584,13 @@ class basic_json case value_t::discarded: { - o << ""; + o.write("", 11); return; } case value_t::null: { - o << "null"; + o.write("null", 4); return; } } From 909b439b03dd1aa048eeaef82c5abfce85c6a56a Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Mon, 27 Feb 2017 14:58:10 +0100 Subject: [PATCH 009/530] :zap: micro-optimizations for dump() numtostr now directly writes to a stream. Return value of snprintf is reused to avoid finding end of string. Cachegrind suggests a 1% performance increase. --- src/json.hpp | 72 +++++++++++++++++------------------------------ src/json.hpp.re2c | 72 +++++++++++++++++------------------------------ 2 files changed, 52 insertions(+), 92 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 1094aade..0f452706 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -8252,14 +8252,9 @@ class basic_json { public: template - numtostr(NumberType value) + numtostr(NumberType value, std::ostream& o) { - x_write(value, std::is_integral()); - } - - const char* c_str() const - { - return m_buf.data(); + x_write(value, std::is_integral(), o); } private: @@ -8267,12 +8262,12 @@ class basic_json std::array < char, 64 > m_buf{{}}; template - void x_write(NumberType x, /*is_integral=*/std::true_type) + void x_write(NumberType x, /*is_integral=*/std::true_type, std::ostream& o) { // special case for "0" if (x == 0) { - m_buf[0] = '0'; + o.put('0'); return; } @@ -8298,30 +8293,31 @@ class basic_json } std::reverse(m_buf.begin(), m_buf.begin() + i); + o.write(m_buf.data(), static_cast(i)); } template - void x_write(NumberType x, /*is_integral=*/std::false_type) + void x_write(NumberType x, /*is_integral=*/std::false_type, std::ostream& o) { // special case for 0.0 and -0.0 if (x == 0) { - size_t i = 0; if (std::signbit(x)) { - m_buf[i++] = '-'; + o.write("-0.0", 4); + } + else + { + o.write("0.0", 3); } - m_buf[i++] = '0'; - m_buf[i++] = '.'; - m_buf[i] = '0'; return; } // get number of digits for a text -> float -> text round-trip static constexpr auto d = std::numeric_limits::digits10; - // the actual conversion - const auto written_bytes = snprintf(m_buf.data(), m_buf.size(), "%.*g", d, x); + // the actual conversion; store how many bytes have been written + auto written_bytes = snprintf(m_buf.data(), m_buf.size(), "%.*g", d, x); // negative value indicates an error assert(written_bytes > 0); @@ -8342,6 +8338,7 @@ class basic_json { const auto end = std::remove(m_buf.begin(), m_buf.begin() + written_bytes, thousands_sep); std::fill(end, m_buf.end(), '\0'); + written_bytes -= m_buf.end() - end; } // convert decimal point to '.' @@ -8357,36 +8354,19 @@ class basic_json } } + o.write(m_buf.data(), static_cast(written_bytes)); + // determine if need to append ".0" - size_t i = 0; - bool value_is_int_like = true; - for (i = 0; i < m_buf.size(); ++i) + const bool value_is_int_like = std::all_of(m_buf.begin(), + m_buf.begin() + written_bytes + 1, + [](char c) { - // break when end of number is reached - if (m_buf[i] == '\0') - { - break; - } - - // check if we find non-int character - value_is_int_like = value_is_int_like and m_buf[i] != '.' and - m_buf[i] != 'e' and m_buf[i] != 'E'; - } - + // we use %g above, so there cannot be an 'E' character + return c != '.' and c != 'e'; + }); if (value_is_int_like) { - // there must be 2 bytes left for ".0" - assert((i + 2) < m_buf.size()); - // we write to the end of the number - assert(m_buf[i] == '\0'); - assert(m_buf[i - 1] != '\0'); - - // add ".0" - m_buf[i] = '.'; - m_buf[i + 1] = '0'; - - // the resulting string is properly terminated - assert(m_buf[i + 2] == '\0'); + o.write(".0", 2); } } }; @@ -8566,19 +8546,19 @@ class basic_json case value_t::number_integer: { - o << numtostr(m_value.number_integer).c_str(); + numtostr(m_value.number_integer, o); return; } case value_t::number_unsigned: { - o << numtostr(m_value.number_unsigned).c_str(); + numtostr(m_value.number_unsigned, o); return; } case value_t::number_float: { - o << numtostr(m_value.number_float).c_str(); + numtostr(m_value.number_float, o); return; } diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index a686a1a7..b4b9e3d2 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -8252,14 +8252,9 @@ class basic_json { public: template - numtostr(NumberType value) + numtostr(NumberType value, std::ostream& o) { - x_write(value, std::is_integral()); - } - - const char* c_str() const - { - return m_buf.data(); + x_write(value, std::is_integral(), o); } private: @@ -8267,12 +8262,12 @@ class basic_json std::array < char, 64 > m_buf{{}}; template - void x_write(NumberType x, /*is_integral=*/std::true_type) + void x_write(NumberType x, /*is_integral=*/std::true_type, std::ostream& o) { // special case for "0" if (x == 0) { - m_buf[0] = '0'; + o.put('0'); return; } @@ -8298,30 +8293,31 @@ class basic_json } std::reverse(m_buf.begin(), m_buf.begin() + i); + o.write(m_buf.data(), static_cast(i)); } template - void x_write(NumberType x, /*is_integral=*/std::false_type) + void x_write(NumberType x, /*is_integral=*/std::false_type, std::ostream& o) { // special case for 0.0 and -0.0 if (x == 0) { - size_t i = 0; if (std::signbit(x)) { - m_buf[i++] = '-'; + o.write("-0.0", 4); + } + else + { + o.write("0.0", 3); } - m_buf[i++] = '0'; - m_buf[i++] = '.'; - m_buf[i] = '0'; return; } // get number of digits for a text -> float -> text round-trip static constexpr auto d = std::numeric_limits::digits10; - // the actual conversion - const auto written_bytes = snprintf(m_buf.data(), m_buf.size(), "%.*g", d, x); + // the actual conversion; store how many bytes have been written + auto written_bytes = snprintf(m_buf.data(), m_buf.size(), "%.*g", d, x); // negative value indicates an error assert(written_bytes > 0); @@ -8342,6 +8338,7 @@ class basic_json { const auto end = std::remove(m_buf.begin(), m_buf.begin() + written_bytes, thousands_sep); std::fill(end, m_buf.end(), '\0'); + written_bytes -= m_buf.end() - end; } // convert decimal point to '.' @@ -8357,36 +8354,19 @@ class basic_json } } + o.write(m_buf.data(), static_cast(written_bytes)); + // determine if need to append ".0" - size_t i = 0; - bool value_is_int_like = true; - for (i = 0; i < m_buf.size(); ++i) + const bool value_is_int_like = std::all_of(m_buf.begin(), + m_buf.begin() + written_bytes + 1, + [](char c) { - // break when end of number is reached - if (m_buf[i] == '\0') - { - break; - } - - // check if we find non-int character - value_is_int_like = value_is_int_like and m_buf[i] != '.' and - m_buf[i] != 'e' and m_buf[i] != 'E'; - } - + // we use %g above, so there cannot be an 'E' character + return c != '.' and c != 'e'; + }); if (value_is_int_like) { - // there must be 2 bytes left for ".0" - assert((i + 2) < m_buf.size()); - // we write to the end of the number - assert(m_buf[i] == '\0'); - assert(m_buf[i - 1] != '\0'); - - // add ".0" - m_buf[i] = '.'; - m_buf[i + 1] = '0'; - - // the resulting string is properly terminated - assert(m_buf[i + 2] == '\0'); + o.write(".0", 2); } } }; @@ -8566,19 +8546,19 @@ class basic_json case value_t::number_integer: { - o << numtostr(m_value.number_integer).c_str(); + numtostr(m_value.number_integer, o); return; } case value_t::number_unsigned: { - o << numtostr(m_value.number_unsigned).c_str(); + numtostr(m_value.number_unsigned, o); return; } case value_t::number_float: { - o << numtostr(m_value.number_float).c_str(); + numtostr(m_value.number_float, o); return; } From 9c4919ff34a78dfd0dd4307518716e60399ad37a Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Mon, 27 Feb 2017 16:19:07 +0100 Subject: [PATCH 010/530] :rewind: ":zap: micro-optimizations for dump()" This reverts commit 909b439b03dd1aa048eeaef82c5abfce85c6a56a. For some strange reason, the test suite crashes when compiled with GCC. --- src/json.hpp | 72 ++++++++++++++++++++++++++++++----------------- src/json.hpp.re2c | 72 ++++++++++++++++++++++++++++++----------------- 2 files changed, 92 insertions(+), 52 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 0f452706..1094aade 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -8252,9 +8252,14 @@ class basic_json { public: template - numtostr(NumberType value, std::ostream& o) + numtostr(NumberType value) { - x_write(value, std::is_integral(), o); + x_write(value, std::is_integral()); + } + + const char* c_str() const + { + return m_buf.data(); } private: @@ -8262,12 +8267,12 @@ class basic_json std::array < char, 64 > m_buf{{}}; template - void x_write(NumberType x, /*is_integral=*/std::true_type, std::ostream& o) + void x_write(NumberType x, /*is_integral=*/std::true_type) { // special case for "0" if (x == 0) { - o.put('0'); + m_buf[0] = '0'; return; } @@ -8293,31 +8298,30 @@ class basic_json } std::reverse(m_buf.begin(), m_buf.begin() + i); - o.write(m_buf.data(), static_cast(i)); } template - void x_write(NumberType x, /*is_integral=*/std::false_type, std::ostream& o) + void x_write(NumberType x, /*is_integral=*/std::false_type) { // special case for 0.0 and -0.0 if (x == 0) { + size_t i = 0; if (std::signbit(x)) { - o.write("-0.0", 4); - } - else - { - o.write("0.0", 3); + m_buf[i++] = '-'; } + m_buf[i++] = '0'; + m_buf[i++] = '.'; + m_buf[i] = '0'; return; } // get number of digits for a text -> float -> text round-trip static constexpr auto d = std::numeric_limits::digits10; - // the actual conversion; store how many bytes have been written - auto written_bytes = snprintf(m_buf.data(), m_buf.size(), "%.*g", d, x); + // the actual conversion + const auto written_bytes = snprintf(m_buf.data(), m_buf.size(), "%.*g", d, x); // negative value indicates an error assert(written_bytes > 0); @@ -8338,7 +8342,6 @@ class basic_json { const auto end = std::remove(m_buf.begin(), m_buf.begin() + written_bytes, thousands_sep); std::fill(end, m_buf.end(), '\0'); - written_bytes -= m_buf.end() - end; } // convert decimal point to '.' @@ -8354,19 +8357,36 @@ class basic_json } } - o.write(m_buf.data(), static_cast(written_bytes)); - // determine if need to append ".0" - const bool value_is_int_like = std::all_of(m_buf.begin(), - m_buf.begin() + written_bytes + 1, - [](char c) + size_t i = 0; + bool value_is_int_like = true; + for (i = 0; i < m_buf.size(); ++i) { - // we use %g above, so there cannot be an 'E' character - return c != '.' and c != 'e'; - }); + // break when end of number is reached + if (m_buf[i] == '\0') + { + break; + } + + // check if we find non-int character + value_is_int_like = value_is_int_like and m_buf[i] != '.' and + m_buf[i] != 'e' and m_buf[i] != 'E'; + } + if (value_is_int_like) { - o.write(".0", 2); + // there must be 2 bytes left for ".0" + assert((i + 2) < m_buf.size()); + // we write to the end of the number + assert(m_buf[i] == '\0'); + assert(m_buf[i - 1] != '\0'); + + // add ".0" + m_buf[i] = '.'; + m_buf[i + 1] = '0'; + + // the resulting string is properly terminated + assert(m_buf[i + 2] == '\0'); } } }; @@ -8546,19 +8566,19 @@ class basic_json case value_t::number_integer: { - numtostr(m_value.number_integer, o); + o << numtostr(m_value.number_integer).c_str(); return; } case value_t::number_unsigned: { - numtostr(m_value.number_unsigned, o); + o << numtostr(m_value.number_unsigned).c_str(); return; } case value_t::number_float: { - numtostr(m_value.number_float, o); + o << numtostr(m_value.number_float).c_str(); return; } diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index b4b9e3d2..a686a1a7 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -8252,9 +8252,14 @@ class basic_json { public: template - numtostr(NumberType value, std::ostream& o) + numtostr(NumberType value) { - x_write(value, std::is_integral(), o); + x_write(value, std::is_integral()); + } + + const char* c_str() const + { + return m_buf.data(); } private: @@ -8262,12 +8267,12 @@ class basic_json std::array < char, 64 > m_buf{{}}; template - void x_write(NumberType x, /*is_integral=*/std::true_type, std::ostream& o) + void x_write(NumberType x, /*is_integral=*/std::true_type) { // special case for "0" if (x == 0) { - o.put('0'); + m_buf[0] = '0'; return; } @@ -8293,31 +8298,30 @@ class basic_json } std::reverse(m_buf.begin(), m_buf.begin() + i); - o.write(m_buf.data(), static_cast(i)); } template - void x_write(NumberType x, /*is_integral=*/std::false_type, std::ostream& o) + void x_write(NumberType x, /*is_integral=*/std::false_type) { // special case for 0.0 and -0.0 if (x == 0) { + size_t i = 0; if (std::signbit(x)) { - o.write("-0.0", 4); - } - else - { - o.write("0.0", 3); + m_buf[i++] = '-'; } + m_buf[i++] = '0'; + m_buf[i++] = '.'; + m_buf[i] = '0'; return; } // get number of digits for a text -> float -> text round-trip static constexpr auto d = std::numeric_limits::digits10; - // the actual conversion; store how many bytes have been written - auto written_bytes = snprintf(m_buf.data(), m_buf.size(), "%.*g", d, x); + // the actual conversion + const auto written_bytes = snprintf(m_buf.data(), m_buf.size(), "%.*g", d, x); // negative value indicates an error assert(written_bytes > 0); @@ -8338,7 +8342,6 @@ class basic_json { const auto end = std::remove(m_buf.begin(), m_buf.begin() + written_bytes, thousands_sep); std::fill(end, m_buf.end(), '\0'); - written_bytes -= m_buf.end() - end; } // convert decimal point to '.' @@ -8354,19 +8357,36 @@ class basic_json } } - o.write(m_buf.data(), static_cast(written_bytes)); - // determine if need to append ".0" - const bool value_is_int_like = std::all_of(m_buf.begin(), - m_buf.begin() + written_bytes + 1, - [](char c) + size_t i = 0; + bool value_is_int_like = true; + for (i = 0; i < m_buf.size(); ++i) { - // we use %g above, so there cannot be an 'E' character - return c != '.' and c != 'e'; - }); + // break when end of number is reached + if (m_buf[i] == '\0') + { + break; + } + + // check if we find non-int character + value_is_int_like = value_is_int_like and m_buf[i] != '.' and + m_buf[i] != 'e' and m_buf[i] != 'E'; + } + if (value_is_int_like) { - o.write(".0", 2); + // there must be 2 bytes left for ".0" + assert((i + 2) < m_buf.size()); + // we write to the end of the number + assert(m_buf[i] == '\0'); + assert(m_buf[i - 1] != '\0'); + + // add ".0" + m_buf[i] = '.'; + m_buf[i + 1] = '0'; + + // the resulting string is properly terminated + assert(m_buf[i + 2] == '\0'); } } }; @@ -8546,19 +8566,19 @@ class basic_json case value_t::number_integer: { - numtostr(m_value.number_integer, o); + o << numtostr(m_value.number_integer).c_str(); return; } case value_t::number_unsigned: { - numtostr(m_value.number_unsigned, o); + o << numtostr(m_value.number_unsigned).c_str(); return; } case value_t::number_float: { - numtostr(m_value.number_float, o); + o << numtostr(m_value.number_float).c_str(); return; } From 54ef5f7b47276fc56637afbf083ee9e8a9b1ef44 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Mon, 27 Feb 2017 21:22:39 +0100 Subject: [PATCH 011/530] :hammer: moved serialization functions to serializer class The class is currently just a wrapper for an std::ostream and collects all functions related to serialization. The next step should be recycling of variables to avoid repetitive initialization for each recursive dump call. --- src/json.hpp | 1046 ++++++++++++++++---------------- src/json.hpp.re2c | 1050 +++++++++++++++++---------------- test/src/unit-convenience.cpp | 76 +-- 3 files changed, 1101 insertions(+), 1071 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 1094aade..a73b5048 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -2644,14 +2644,15 @@ class basic_json string_t dump(const int indent = -1) const { std::stringstream ss; + serializer s(ss); if (indent >= 0) { - dump(ss, true, static_cast(indent)); + s.dump(*this, true, static_cast(indent)); } else { - dump(ss, false, 0); + s.dump(*this, false, 0); } return ss.str(); @@ -6194,6 +6195,531 @@ class basic_json /// @name serialization /// @{ + private: + class serializer + { + public: + serializer(std::ostream& s) + : o(s) + {} + + /*! + @brief internal implementation of the serialization function + + This function is called by the public member function dump and organizes + the serialization internally. The indentation level is propagated as + additional parameter. In case of arrays and objects, the function is + called recursively. Note that + + - strings and object keys are escaped using `escape_string()` + - integer numbers are converted implicitly via `operator<<` + - floating-point numbers are converted to a string using `"%g"` format + + @param[in] val value to serialize + @param[in] pretty_print whether the output shall be pretty-printed + @param[in] indent_step the indent level + @param[in] current_indent the current indent level (only used internally) + */ + void dump(const basic_json& val, + const bool pretty_print, + const unsigned int indent_step, + const unsigned int current_indent = 0) const + { + switch (val.m_type) + { + case value_t::object: + { + if (val.m_value.object->empty()) + { + o.write("{}", 2); + return; + } + + if (pretty_print) + { + o.write("{\n", 2); + + // variable to hold indentation for recursive calls + const auto new_indent = current_indent + indent_step; + string_t indent_string = string_t(new_indent, ' '); + + // first n-1 elements + auto i = val.m_value.object->cbegin(); + for (size_t cnt = 0; cnt < val.m_value.object->size() - 1; ++cnt, ++i) + { + o.write(indent_string.c_str(), new_indent); + o.put('\"'); + const auto s = escape_string(i->first); + o.write(s.c_str(), static_cast(s.size())); + o.write("\": ", 3); + dump(i->second, true, indent_step, new_indent); + o.write(",\n", 2); + } + + // last element + assert(i != val.m_value.object->cend()); + o.write(indent_string.c_str(), new_indent); + o.put('\"'); + const auto s = escape_string(i->first); + o.write(s.c_str(), static_cast(s.size())); + o.write("\": ", 3); + dump(i->second, true, indent_step, new_indent); + + o.put('\n'); + o.write(indent_string.c_str(), current_indent); + o.put('}'); + } + else + { + o.put('{'); + + // first n-1 elements + auto i = val.m_value.object->cbegin(); + for (size_t cnt = 0; cnt < val.m_value.object->size() - 1; ++cnt, ++i) + { + o.put('\"'); + const auto s = escape_string(i->first); + o.write(s.c_str(), static_cast(s.size())); + o.write("\":", 2); + dump(i->second, false, indent_step, current_indent); + o.put(','); + } + + // last element + assert(i != val.m_value.object->cend()); + o.put('\"'); + const auto s = escape_string(i->first); + o.write(s.c_str(), static_cast(s.size())); + o.write("\":", 2); + dump(i->second, false, indent_step, current_indent); + + o.put('}'); + } + + return; + } + + case value_t::array: + { + if (val.m_value.array->empty()) + { + o.write("[]", 2); + return; + } + + if (pretty_print) + { + o.write("[\n", 2); + + // variable to hold indentation for recursive calls + const auto new_indent = current_indent + indent_step; + string_t indent_string = string_t(new_indent, ' '); + + // first n-1 elements + for (auto i = val.m_value.array->cbegin(); i != val.m_value.array->cend() - 1; ++i) + { + o.write(indent_string.c_str(), new_indent); + dump(*i, true, indent_step, new_indent); + o.write(",\n", 2); + } + + // last element + assert(not val.m_value.array->empty()); + o.write(indent_string.c_str(), new_indent); + dump(val.m_value.array->back(), true, indent_step, new_indent); + + o.put('\n'); + o.write(indent_string.c_str(), current_indent); + o.put(']'); + } + else + { + o.put('['); + + // first n-1 elements + for (auto i = val.m_value.array->cbegin(); i != val.m_value.array->cend() - 1; ++i) + { + dump(*i, false, indent_step, current_indent); + o.put(','); + } + + // last element + assert(not val.m_value.array->empty()); + dump(val.m_value.array->back(), false, indent_step, current_indent); + + o.put(']'); + } + + return; + } + + case value_t::string: + { + o.put('\"'); + const auto s = escape_string(*val.m_value.string); + o.write(s.c_str(), static_cast(s.size())); + o.put('\"'); + return; + } + + case value_t::boolean: + { + if (val.m_value.boolean) + { + o.write("true", 4); + } + else + { + o.write("false", 5); + } + return; + } + + case value_t::number_integer: + { + o << numtostr(val.m_value.number_integer).c_str(); + return; + } + + case value_t::number_unsigned: + { + o << numtostr(val.m_value.number_unsigned).c_str(); + return; + } + + case value_t::number_float: + { + o << numtostr(val.m_value.number_float).c_str(); + return; + } + + case value_t::discarded: + { + o.write("", 11); + return; + } + + case value_t::null: + { + o.write("null", 4); + return; + } + } + } + + private: + /*! + @brief calculates the extra space to escape a JSON string + + @param[in] s the string to escape + @return the number of characters required to escape string @a s + + @complexity Linear in the length of string @a s. + */ + static std::size_t extra_space(const string_t& s) noexcept + { + return std::accumulate(s.begin(), s.end(), size_t{}, + [](size_t res, typename string_t::value_type c) + { + switch (c) + { + case '"': + case '\\': + case '\b': + case '\f': + case '\n': + case '\r': + case '\t': + { + // from c (1 byte) to \x (2 bytes) + return res + 1; + } + + default: + { + if (c >= 0x00 and c <= 0x1f) + { + // from c (1 byte) to \uxxxx (6 bytes) + return res + 5; + } + + return res; + } + } + }); + } + + /*! + @brief escape a string + + Escape a string by replacing certain special characters by a sequence of + an escape character (backslash) and another character and other control + characters by a sequence of "\u" followed by a four-digit hex + representation. + + @param[in] s the string to escape + @return the escaped string + + @complexity Linear in the length of string @a s. + */ + static string_t escape_string(const string_t& s) + { + const auto space = extra_space(s); + if (space == 0) + { + return s; + } + + // create a result string of necessary size + string_t result(s.size() + space, '\\'); + std::size_t pos = 0; + + for (const auto& c : s) + { + switch (c) + { + // quotation mark (0x22) + case '"': + { + result[pos + 1] = '"'; + pos += 2; + break; + } + + // reverse solidus (0x5c) + case '\\': + { + // nothing to change + pos += 2; + break; + } + + // backspace (0x08) + case '\b': + { + result[pos + 1] = 'b'; + pos += 2; + break; + } + + // formfeed (0x0c) + case '\f': + { + result[pos + 1] = 'f'; + pos += 2; + break; + } + + // newline (0x0a) + case '\n': + { + result[pos + 1] = 'n'; + pos += 2; + break; + } + + // carriage return (0x0d) + case '\r': + { + result[pos + 1] = 'r'; + pos += 2; + break; + } + + // horizontal tab (0x09) + case '\t': + { + result[pos + 1] = 't'; + pos += 2; + break; + } + + default: + { + if (c >= 0x00 and c <= 0x1f) + { + // convert a number 0..15 to its hex representation + // (0..f) + static const char hexify[16] = + { + '0', '1', '2', '3', '4', '5', '6', '7', + '8', '9', 'a', 'b', 'c', 'd', 'e', 'f' + }; + + // print character c as \uxxxx + for (const char m : + { 'u', '0', '0', hexify[c >> 4], hexify[c & 0x0f] + }) + { + result[++pos] = m; + } + + ++pos; + } + else + { + // all other characters are added as-is + result[pos++] = c; + } + break; + } + } + } + + return result; + } + + /*! + @brief locale-independent serialization for built-in arithmetic types + */ + struct numtostr + { + public: + template + numtostr(NumberType value) + { + x_write(value, std::is_integral()); + } + + const char* c_str() const + { + return m_buf.data(); + } + + private: + /// a (hopefully) large enough character buffer + std::array < char, 64 > m_buf{{}}; + + template + void x_write(NumberType x, /*is_integral=*/std::true_type) + { + // special case for "0" + if (x == 0) + { + m_buf[0] = '0'; + return; + } + + const bool is_negative = x < 0; + size_t i = 0; + + // spare 1 byte for '\0' + while (x != 0 and i < m_buf.size() - 1) + { + const auto digit = std::labs(static_cast(x % 10)); + m_buf[i++] = static_cast('0' + digit); + x /= 10; + } + + // make sure the number has been processed completely + assert(x == 0); + + if (is_negative) + { + // make sure there is capacity for the '-' + assert(i < m_buf.size() - 2); + m_buf[i++] = '-'; + } + + std::reverse(m_buf.begin(), m_buf.begin() + i); + } + + template + void x_write(NumberType x, /*is_integral=*/std::false_type) + { + // special case for 0.0 and -0.0 + if (x == 0) + { + size_t i = 0; + if (std::signbit(x)) + { + m_buf[i++] = '-'; + } + m_buf[i++] = '0'; + m_buf[i++] = '.'; + m_buf[i] = '0'; + return; + } + + // get number of digits for a text -> float -> text round-trip + static constexpr auto d = std::numeric_limits::digits10; + + // the actual conversion + const auto written_bytes = snprintf(m_buf.data(), m_buf.size(), "%.*g", d, x); + + // negative value indicates an error + assert(written_bytes > 0); + // check if buffer was large enough + assert(static_cast(written_bytes) < m_buf.size()); + + // read information from locale + const auto loc = localeconv(); + assert(loc != nullptr); + const char thousands_sep = !loc->thousands_sep ? '\0' + : loc->thousands_sep[0]; + + const char decimal_point = !loc->decimal_point ? '\0' + : loc->decimal_point[0]; + + // erase thousands separator + if (thousands_sep != '\0') + { + const auto end = std::remove(m_buf.begin(), m_buf.begin() + written_bytes, thousands_sep); + std::fill(end, m_buf.end(), '\0'); + } + + // convert decimal point to '.' + if (decimal_point != '\0' and decimal_point != '.') + { + for (auto& c : m_buf) + { + if (c == decimal_point) + { + c = '.'; + break; + } + } + } + + // determine if need to append ".0" + size_t i = 0; + bool value_is_int_like = true; + for (i = 0; i < m_buf.size(); ++i) + { + // break when end of number is reached + if (m_buf[i] == '\0') + { + break; + } + + // check if we find non-int character + value_is_int_like = value_is_int_like and m_buf[i] != '.' and + m_buf[i] != 'e' and m_buf[i] != 'E'; + } + + if (value_is_int_like) + { + // there must be 2 bytes left for ".0" + assert((i + 2) < m_buf.size()); + // we write to the end of the number + assert(m_buf[i] == '\0'); + assert(m_buf[i - 1] != '\0'); + + // add ".0" + m_buf[i] = '.'; + m_buf[i + 1] = '0'; + + // the resulting string is properly terminated + assert(m_buf[i + 2] == '\0'); + } + } + }; + + private: + std::ostream& o; + }; + + public: /*! @brief serialize to stream @@ -6226,7 +6752,8 @@ class basic_json o.width(0); // do the actual serialization - j.dump(o, pretty_print, static_cast(indentation)); + serializer s(o); + s.dump(j, pretty_print, static_cast(indentation)); return o; } @@ -8082,519 +8609,6 @@ class basic_json } } - private: - /*! - @brief calculates the extra space to escape a JSON string - - @param[in] s the string to escape - @return the number of characters required to escape string @a s - - @complexity Linear in the length of string @a s. - */ - static std::size_t extra_space(const string_t& s) noexcept - { - return std::accumulate(s.begin(), s.end(), size_t{}, - [](size_t res, typename string_t::value_type c) - { - switch (c) - { - case '"': - case '\\': - case '\b': - case '\f': - case '\n': - case '\r': - case '\t': - { - // from c (1 byte) to \x (2 bytes) - return res + 1; - } - - default: - { - if (c >= 0x00 and c <= 0x1f) - { - // from c (1 byte) to \uxxxx (6 bytes) - return res + 5; - } - - return res; - } - } - }); - } - - /*! - @brief escape a string - - Escape a string by replacing certain special characters by a sequence of - an escape character (backslash) and another character and other control - characters by a sequence of "\u" followed by a four-digit hex - representation. - - @param[in] s the string to escape - @return the escaped string - - @complexity Linear in the length of string @a s. - */ - static string_t escape_string(const string_t& s) - { - const auto space = extra_space(s); - if (space == 0) - { - return s; - } - - // create a result string of necessary size - string_t result(s.size() + space, '\\'); - std::size_t pos = 0; - - for (const auto& c : s) - { - switch (c) - { - // quotation mark (0x22) - case '"': - { - result[pos + 1] = '"'; - pos += 2; - break; - } - - // reverse solidus (0x5c) - case '\\': - { - // nothing to change - pos += 2; - break; - } - - // backspace (0x08) - case '\b': - { - result[pos + 1] = 'b'; - pos += 2; - break; - } - - // formfeed (0x0c) - case '\f': - { - result[pos + 1] = 'f'; - pos += 2; - break; - } - - // newline (0x0a) - case '\n': - { - result[pos + 1] = 'n'; - pos += 2; - break; - } - - // carriage return (0x0d) - case '\r': - { - result[pos + 1] = 'r'; - pos += 2; - break; - } - - // horizontal tab (0x09) - case '\t': - { - result[pos + 1] = 't'; - pos += 2; - break; - } - - default: - { - if (c >= 0x00 and c <= 0x1f) - { - // convert a number 0..15 to its hex representation - // (0..f) - static const char hexify[16] = - { - '0', '1', '2', '3', '4', '5', '6', '7', - '8', '9', 'a', 'b', 'c', 'd', 'e', 'f' - }; - - // print character c as \uxxxx - for (const char m : - { 'u', '0', '0', hexify[c >> 4], hexify[c & 0x0f] - }) - { - result[++pos] = m; - } - - ++pos; - } - else - { - // all other characters are added as-is - result[pos++] = c; - } - break; - } - } - } - - return result; - } - - - /*! - @brief locale-independent serialization for built-in arithmetic types - */ - struct numtostr - { - public: - template - numtostr(NumberType value) - { - x_write(value, std::is_integral()); - } - - const char* c_str() const - { - return m_buf.data(); - } - - private: - /// a (hopefully) large enough character buffer - std::array < char, 64 > m_buf{{}}; - - template - void x_write(NumberType x, /*is_integral=*/std::true_type) - { - // special case for "0" - if (x == 0) - { - m_buf[0] = '0'; - return; - } - - const bool is_negative = x < 0; - size_t i = 0; - - // spare 1 byte for '\0' - while (x != 0 and i < m_buf.size() - 1) - { - const auto digit = std::labs(static_cast(x % 10)); - m_buf[i++] = static_cast('0' + digit); - x /= 10; - } - - // make sure the number has been processed completely - assert(x == 0); - - if (is_negative) - { - // make sure there is capacity for the '-' - assert(i < m_buf.size() - 2); - m_buf[i++] = '-'; - } - - std::reverse(m_buf.begin(), m_buf.begin() + i); - } - - template - void x_write(NumberType x, /*is_integral=*/std::false_type) - { - // special case for 0.0 and -0.0 - if (x == 0) - { - size_t i = 0; - if (std::signbit(x)) - { - m_buf[i++] = '-'; - } - m_buf[i++] = '0'; - m_buf[i++] = '.'; - m_buf[i] = '0'; - return; - } - - // get number of digits for a text -> float -> text round-trip - static constexpr auto d = std::numeric_limits::digits10; - - // the actual conversion - const auto written_bytes = snprintf(m_buf.data(), m_buf.size(), "%.*g", d, x); - - // negative value indicates an error - assert(written_bytes > 0); - // check if buffer was large enough - assert(static_cast(written_bytes) < m_buf.size()); - - // read information from locale - const auto loc = localeconv(); - assert(loc != nullptr); - const char thousands_sep = !loc->thousands_sep ? '\0' - : loc->thousands_sep[0]; - - const char decimal_point = !loc->decimal_point ? '\0' - : loc->decimal_point[0]; - - // erase thousands separator - if (thousands_sep != '\0') - { - const auto end = std::remove(m_buf.begin(), m_buf.begin() + written_bytes, thousands_sep); - std::fill(end, m_buf.end(), '\0'); - } - - // convert decimal point to '.' - if (decimal_point != '\0' and decimal_point != '.') - { - for (auto& c : m_buf) - { - if (c == decimal_point) - { - c = '.'; - break; - } - } - } - - // determine if need to append ".0" - size_t i = 0; - bool value_is_int_like = true; - for (i = 0; i < m_buf.size(); ++i) - { - // break when end of number is reached - if (m_buf[i] == '\0') - { - break; - } - - // check if we find non-int character - value_is_int_like = value_is_int_like and m_buf[i] != '.' and - m_buf[i] != 'e' and m_buf[i] != 'E'; - } - - if (value_is_int_like) - { - // there must be 2 bytes left for ".0" - assert((i + 2) < m_buf.size()); - // we write to the end of the number - assert(m_buf[i] == '\0'); - assert(m_buf[i - 1] != '\0'); - - // add ".0" - m_buf[i] = '.'; - m_buf[i + 1] = '0'; - - // the resulting string is properly terminated - assert(m_buf[i + 2] == '\0'); - } - } - }; - - - /*! - @brief internal implementation of the serialization function - - This function is called by the public member function dump and organizes - the serialization internally. The indentation level is propagated as - additional parameter. In case of arrays and objects, the function is - called recursively. Note that - - - strings and object keys are escaped using `escape_string()` - - integer numbers are converted implicitly via `operator<<` - - floating-point numbers are converted to a string using `"%g"` format - - @param[out] o stream to write to - @param[in] pretty_print whether the output shall be pretty-printed - @param[in] indent_step the indent level - @param[in] current_indent the current indent level (only used internally) - */ - void dump(std::ostream& o, - const bool pretty_print, - const unsigned int indent_step, - const unsigned int current_indent = 0) const - { - switch (m_type) - { - case value_t::object: - { - if (m_value.object->empty()) - { - o.write("{}", 2); - return; - } - - if (pretty_print) - { - o.write("{\n", 2); - - // variable to hold indentation for recursive calls - const auto new_indent = current_indent + indent_step; - string_t indent_string = string_t(new_indent, ' '); - - // first n-1 elements - auto i = m_value.object->cbegin(); - for (size_t cnt = 0; cnt < m_value.object->size() - 1; ++cnt, ++i) - { - o.write(indent_string.c_str(), new_indent); - o.put('\"'); - const auto s = escape_string(i->first); - o.write(s.c_str(), static_cast(s.size())); - o.write("\": ", 3); - i->second.dump(o, true, indent_step, new_indent); - o.write(",\n", 2); - } - - // last element - assert(i != m_value.object->cend()); - o.write(indent_string.c_str(), new_indent); - o.put('\"'); - const auto s = escape_string(i->first); - o.write(s.c_str(), static_cast(s.size())); - o.write("\": ", 3); - i->second.dump(o, true, indent_step, new_indent); - - o.put('\n'); - o.write(indent_string.c_str(), current_indent); - o.put('}'); - } - else - { - o.put('{'); - - // first n-1 elements - auto i = m_value.object->cbegin(); - for (size_t cnt = 0; cnt < m_value.object->size() - 1; ++cnt, ++i) - { - o.put('\"'); - const auto s = escape_string(i->first); - o.write(s.c_str(), static_cast(s.size())); - o.write("\":", 2); - i->second.dump(o, false, indent_step, current_indent); - o.put(','); - } - - // last element - assert(i != m_value.object->cend()); - o.put('\"'); - const auto s = escape_string(i->first); - o.write(s.c_str(), static_cast(s.size())); - o.write("\":", 2); - i->second.dump(o, false, indent_step, current_indent); - - o.put('}'); - } - - return; - } - - case value_t::array: - { - if (m_value.array->empty()) - { - o.write("[]", 2); - return; - } - - if (pretty_print) - { - o.write("[\n", 2); - - // variable to hold indentation for recursive calls - const auto new_indent = current_indent + indent_step; - string_t indent_string = string_t(new_indent, ' '); - - // first n-1 elements - for (auto i = m_value.array->cbegin(); i != m_value.array->cend() - 1; ++i) - { - o.write(indent_string.c_str(), new_indent); - i->dump(o, true, indent_step, new_indent); - o.write(",\n", 2); - } - - // last element - assert(not m_value.array->empty()); - o.write(indent_string.c_str(), new_indent); - m_value.array->back().dump(o, true, indent_step, new_indent); - - o.put('\n'); - o.write(indent_string.c_str(), current_indent); - o.put(']'); - } - else - { - o.put('['); - - // first n-1 elements - for (auto i = m_value.array->cbegin(); i != m_value.array->cend() - 1; ++i) - { - i->dump(o, false, indent_step, current_indent); - o.put(','); - } - - // last element - assert(not m_value.array->empty()); - m_value.array->back().dump(o, false, indent_step, current_indent); - - o.put(']'); - } - - return; - } - - case value_t::string: - { - o.put('\"'); - const auto s = escape_string(*m_value.string); - o.write(s.c_str(), static_cast(s.size())); - o.put('\"'); - return; - } - - case value_t::boolean: - { - if (m_value.boolean) - { - o.write("true", 4); - } - else - { - o.write("false", 5); - } - return; - } - - case value_t::number_integer: - { - o << numtostr(m_value.number_integer).c_str(); - return; - } - - case value_t::number_unsigned: - { - o << numtostr(m_value.number_unsigned).c_str(); - return; - } - - case value_t::number_float: - { - o << numtostr(m_value.number_float).c_str(); - return; - } - - case value_t::discarded: - { - o.write("", 11); - return; - } - - case value_t::null: - { - o.write("null", 4); - return; - } - } - } private: ////////////////////// diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index a686a1a7..01f6bbfe 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -2644,14 +2644,15 @@ class basic_json string_t dump(const int indent = -1) const { std::stringstream ss; + serializer s(ss); if (indent >= 0) { - dump(ss, true, static_cast(indent)); + s.dump(*this, true, static_cast(indent)); } else { - dump(ss, false, 0); + s.dump(*this, false, 0); } return ss.str(); @@ -6194,6 +6195,534 @@ class basic_json /// @name serialization /// @{ + private: + /*! + @brief wrapper around the serialization functions + */ + class serializer + { + public: + serializer(std::ostream& s) + : o(s) + {} + + /*! + @brief internal implementation of the serialization function + + This function is called by the public member function dump and organizes + the serialization internally. The indentation level is propagated as + additional parameter. In case of arrays and objects, the function is + called recursively. Note that + + - strings and object keys are escaped using `escape_string()` + - integer numbers are converted implicitly via `operator<<` + - floating-point numbers are converted to a string using `"%g"` format + + @param[in] val value to serialize + @param[in] pretty_print whether the output shall be pretty-printed + @param[in] indent_step the indent level + @param[in] current_indent the current indent level (only used internally) + */ + void dump(const basic_json& val, + const bool pretty_print, + const unsigned int indent_step, + const unsigned int current_indent = 0) const + { + switch (val.m_type) + { + case value_t::object: + { + if (val.m_value.object->empty()) + { + o.write("{}", 2); + return; + } + + if (pretty_print) + { + o.write("{\n", 2); + + // variable to hold indentation for recursive calls + const auto new_indent = current_indent + indent_step; + string_t indent_string = string_t(new_indent, ' '); + + // first n-1 elements + auto i = val.m_value.object->cbegin(); + for (size_t cnt = 0; cnt < val.m_value.object->size() - 1; ++cnt, ++i) + { + o.write(indent_string.c_str(), new_indent); + o.put('\"'); + const auto s = escape_string(i->first); + o.write(s.c_str(), static_cast(s.size())); + o.write("\": ", 3); + dump(i->second, true, indent_step, new_indent); + o.write(",\n", 2); + } + + // last element + assert(i != val.m_value.object->cend()); + o.write(indent_string.c_str(), new_indent); + o.put('\"'); + const auto s = escape_string(i->first); + o.write(s.c_str(), static_cast(s.size())); + o.write("\": ", 3); + dump(i->second, true, indent_step, new_indent); + + o.put('\n'); + o.write(indent_string.c_str(), current_indent); + o.put('}'); + } + else + { + o.put('{'); + + // first n-1 elements + auto i = val.m_value.object->cbegin(); + for (size_t cnt = 0; cnt < val.m_value.object->size() - 1; ++cnt, ++i) + { + o.put('\"'); + const auto s = escape_string(i->first); + o.write(s.c_str(), static_cast(s.size())); + o.write("\":", 2); + dump(i->second, false, indent_step, current_indent); + o.put(','); + } + + // last element + assert(i != val.m_value.object->cend()); + o.put('\"'); + const auto s = escape_string(i->first); + o.write(s.c_str(), static_cast(s.size())); + o.write("\":", 2); + dump(i->second, false, indent_step, current_indent); + + o.put('}'); + } + + return; + } + + case value_t::array: + { + if (val.m_value.array->empty()) + { + o.write("[]", 2); + return; + } + + if (pretty_print) + { + o.write("[\n", 2); + + // variable to hold indentation for recursive calls + const auto new_indent = current_indent + indent_step; + string_t indent_string = string_t(new_indent, ' '); + + // first n-1 elements + for (auto i = val.m_value.array->cbegin(); i != val.m_value.array->cend() - 1; ++i) + { + o.write(indent_string.c_str(), new_indent); + dump(*i, true, indent_step, new_indent); + o.write(",\n", 2); + } + + // last element + assert(not val.m_value.array->empty()); + o.write(indent_string.c_str(), new_indent); + dump(val.m_value.array->back(), true, indent_step, new_indent); + + o.put('\n'); + o.write(indent_string.c_str(), current_indent); + o.put(']'); + } + else + { + o.put('['); + + // first n-1 elements + for (auto i = val.m_value.array->cbegin(); i != val.m_value.array->cend() - 1; ++i) + { + dump(*i, false, indent_step, current_indent); + o.put(','); + } + + // last element + assert(not val.m_value.array->empty()); + dump(val.m_value.array->back(), false, indent_step, current_indent); + + o.put(']'); + } + + return; + } + + case value_t::string: + { + o.put('\"'); + const auto s = escape_string(*val.m_value.string); + o.write(s.c_str(), static_cast(s.size())); + o.put('\"'); + return; + } + + case value_t::boolean: + { + if (val.m_value.boolean) + { + o.write("true", 4); + } + else + { + o.write("false", 5); + } + return; + } + + case value_t::number_integer: + { + o << numtostr(val.m_value.number_integer).c_str(); + return; + } + + case value_t::number_unsigned: + { + o << numtostr(val.m_value.number_unsigned).c_str(); + return; + } + + case value_t::number_float: + { + o << numtostr(val.m_value.number_float).c_str(); + return; + } + + case value_t::discarded: + { + o.write("", 11); + return; + } + + case value_t::null: + { + o.write("null", 4); + return; + } + } + } + + private: + /*! + @brief calculates the extra space to escape a JSON string + + @param[in] s the string to escape + @return the number of characters required to escape string @a s + + @complexity Linear in the length of string @a s. + */ + static std::size_t extra_space(const string_t& s) noexcept + { + return std::accumulate(s.begin(), s.end(), size_t{}, + [](size_t res, typename string_t::value_type c) + { + switch (c) + { + case '"': + case '\\': + case '\b': + case '\f': + case '\n': + case '\r': + case '\t': + { + // from c (1 byte) to \x (2 bytes) + return res + 1; + } + + default: + { + if (c >= 0x00 and c <= 0x1f) + { + // from c (1 byte) to \uxxxx (6 bytes) + return res + 5; + } + + return res; + } + } + }); + } + + /*! + @brief escape a string + + Escape a string by replacing certain special characters by a sequence of + an escape character (backslash) and another character and other control + characters by a sequence of "\u" followed by a four-digit hex + representation. + + @param[in] s the string to escape + @return the escaped string + + @complexity Linear in the length of string @a s. + */ + static string_t escape_string(const string_t& s) + { + const auto space = extra_space(s); + if (space == 0) + { + return s; + } + + // create a result string of necessary size + string_t result(s.size() + space, '\\'); + std::size_t pos = 0; + + for (const auto& c : s) + { + switch (c) + { + // quotation mark (0x22) + case '"': + { + result[pos + 1] = '"'; + pos += 2; + break; + } + + // reverse solidus (0x5c) + case '\\': + { + // nothing to change + pos += 2; + break; + } + + // backspace (0x08) + case '\b': + { + result[pos + 1] = 'b'; + pos += 2; + break; + } + + // formfeed (0x0c) + case '\f': + { + result[pos + 1] = 'f'; + pos += 2; + break; + } + + // newline (0x0a) + case '\n': + { + result[pos + 1] = 'n'; + pos += 2; + break; + } + + // carriage return (0x0d) + case '\r': + { + result[pos + 1] = 'r'; + pos += 2; + break; + } + + // horizontal tab (0x09) + case '\t': + { + result[pos + 1] = 't'; + pos += 2; + break; + } + + default: + { + if (c >= 0x00 and c <= 0x1f) + { + // convert a number 0..15 to its hex representation + // (0..f) + static const char hexify[16] = + { + '0', '1', '2', '3', '4', '5', '6', '7', + '8', '9', 'a', 'b', 'c', 'd', 'e', 'f' + }; + + // print character c as \uxxxx + for (const char m : + { 'u', '0', '0', hexify[c >> 4], hexify[c & 0x0f] + }) + { + result[++pos] = m; + } + + ++pos; + } + else + { + // all other characters are added as-is + result[pos++] = c; + } + break; + } + } + } + + return result; + } + + /*! + @brief locale-independent serialization for built-in arithmetic types + */ + struct numtostr + { + public: + template + numtostr(NumberType value) + { + x_write(value, std::is_integral()); + } + + const char* c_str() const + { + return m_buf.data(); + } + + private: + /// a (hopefully) large enough character buffer + std::array < char, 64 > m_buf{{}}; + + template + void x_write(NumberType x, /*is_integral=*/std::true_type) + { + // special case for "0" + if (x == 0) + { + m_buf[0] = '0'; + return; + } + + const bool is_negative = x < 0; + size_t i = 0; + + // spare 1 byte for '\0' + while (x != 0 and i < m_buf.size() - 1) + { + const auto digit = std::labs(static_cast(x % 10)); + m_buf[i++] = static_cast('0' + digit); + x /= 10; + } + + // make sure the number has been processed completely + assert(x == 0); + + if (is_negative) + { + // make sure there is capacity for the '-' + assert(i < m_buf.size() - 2); + m_buf[i++] = '-'; + } + + std::reverse(m_buf.begin(), m_buf.begin() + i); + } + + template + void x_write(NumberType x, /*is_integral=*/std::false_type) + { + // special case for 0.0 and -0.0 + if (x == 0) + { + size_t i = 0; + if (std::signbit(x)) + { + m_buf[i++] = '-'; + } + m_buf[i++] = '0'; + m_buf[i++] = '.'; + m_buf[i] = '0'; + return; + } + + // get number of digits for a text -> float -> text round-trip + static constexpr auto d = std::numeric_limits::digits10; + + // the actual conversion + const auto written_bytes = snprintf(m_buf.data(), m_buf.size(), "%.*g", d, x); + + // negative value indicates an error + assert(written_bytes > 0); + // check if buffer was large enough + assert(static_cast(written_bytes) < m_buf.size()); + + // read information from locale + const auto loc = localeconv(); + assert(loc != nullptr); + const char thousands_sep = !loc->thousands_sep ? '\0' + : loc->thousands_sep[0]; + + const char decimal_point = !loc->decimal_point ? '\0' + : loc->decimal_point[0]; + + // erase thousands separator + if (thousands_sep != '\0') + { + const auto end = std::remove(m_buf.begin(), m_buf.begin() + written_bytes, thousands_sep); + std::fill(end, m_buf.end(), '\0'); + } + + // convert decimal point to '.' + if (decimal_point != '\0' and decimal_point != '.') + { + for (auto& c : m_buf) + { + if (c == decimal_point) + { + c = '.'; + break; + } + } + } + + // determine if need to append ".0" + size_t i = 0; + bool value_is_int_like = true; + for (i = 0; i < m_buf.size(); ++i) + { + // break when end of number is reached + if (m_buf[i] == '\0') + { + break; + } + + // check if we find non-int character + value_is_int_like = value_is_int_like and m_buf[i] != '.' and + m_buf[i] != 'e' and m_buf[i] != 'E'; + } + + if (value_is_int_like) + { + // there must be 2 bytes left for ".0" + assert((i + 2) < m_buf.size()); + // we write to the end of the number + assert(m_buf[i] == '\0'); + assert(m_buf[i - 1] != '\0'); + + // add ".0" + m_buf[i] = '.'; + m_buf[i + 1] = '0'; + + // the resulting string is properly terminated + assert(m_buf[i + 2] == '\0'); + } + } + }; + + private: + std::ostream& o; + }; + + public: /*! @brief serialize to stream @@ -6226,8 +6755,8 @@ class basic_json o.width(0); // do the actual serialization - j.dump(o, pretty_print, static_cast(indentation)); - + serializer s(o); + s.dump(j, pretty_print, static_cast(indentation)); return o; } @@ -8082,519 +8611,6 @@ class basic_json } } - private: - /*! - @brief calculates the extra space to escape a JSON string - - @param[in] s the string to escape - @return the number of characters required to escape string @a s - - @complexity Linear in the length of string @a s. - */ - static std::size_t extra_space(const string_t& s) noexcept - { - return std::accumulate(s.begin(), s.end(), size_t{}, - [](size_t res, typename string_t::value_type c) - { - switch (c) - { - case '"': - case '\\': - case '\b': - case '\f': - case '\n': - case '\r': - case '\t': - { - // from c (1 byte) to \x (2 bytes) - return res + 1; - } - - default: - { - if (c >= 0x00 and c <= 0x1f) - { - // from c (1 byte) to \uxxxx (6 bytes) - return res + 5; - } - - return res; - } - } - }); - } - - /*! - @brief escape a string - - Escape a string by replacing certain special characters by a sequence of - an escape character (backslash) and another character and other control - characters by a sequence of "\u" followed by a four-digit hex - representation. - - @param[in] s the string to escape - @return the escaped string - - @complexity Linear in the length of string @a s. - */ - static string_t escape_string(const string_t& s) - { - const auto space = extra_space(s); - if (space == 0) - { - return s; - } - - // create a result string of necessary size - string_t result(s.size() + space, '\\'); - std::size_t pos = 0; - - for (const auto& c : s) - { - switch (c) - { - // quotation mark (0x22) - case '"': - { - result[pos + 1] = '"'; - pos += 2; - break; - } - - // reverse solidus (0x5c) - case '\\': - { - // nothing to change - pos += 2; - break; - } - - // backspace (0x08) - case '\b': - { - result[pos + 1] = 'b'; - pos += 2; - break; - } - - // formfeed (0x0c) - case '\f': - { - result[pos + 1] = 'f'; - pos += 2; - break; - } - - // newline (0x0a) - case '\n': - { - result[pos + 1] = 'n'; - pos += 2; - break; - } - - // carriage return (0x0d) - case '\r': - { - result[pos + 1] = 'r'; - pos += 2; - break; - } - - // horizontal tab (0x09) - case '\t': - { - result[pos + 1] = 't'; - pos += 2; - break; - } - - default: - { - if (c >= 0x00 and c <= 0x1f) - { - // convert a number 0..15 to its hex representation - // (0..f) - static const char hexify[16] = - { - '0', '1', '2', '3', '4', '5', '6', '7', - '8', '9', 'a', 'b', 'c', 'd', 'e', 'f' - }; - - // print character c as \uxxxx - for (const char m : - { 'u', '0', '0', hexify[c >> 4], hexify[c & 0x0f] - }) - { - result[++pos] = m; - } - - ++pos; - } - else - { - // all other characters are added as-is - result[pos++] = c; - } - break; - } - } - } - - return result; - } - - - /*! - @brief locale-independent serialization for built-in arithmetic types - */ - struct numtostr - { - public: - template - numtostr(NumberType value) - { - x_write(value, std::is_integral()); - } - - const char* c_str() const - { - return m_buf.data(); - } - - private: - /// a (hopefully) large enough character buffer - std::array < char, 64 > m_buf{{}}; - - template - void x_write(NumberType x, /*is_integral=*/std::true_type) - { - // special case for "0" - if (x == 0) - { - m_buf[0] = '0'; - return; - } - - const bool is_negative = x < 0; - size_t i = 0; - - // spare 1 byte for '\0' - while (x != 0 and i < m_buf.size() - 1) - { - const auto digit = std::labs(static_cast(x % 10)); - m_buf[i++] = static_cast('0' + digit); - x /= 10; - } - - // make sure the number has been processed completely - assert(x == 0); - - if (is_negative) - { - // make sure there is capacity for the '-' - assert(i < m_buf.size() - 2); - m_buf[i++] = '-'; - } - - std::reverse(m_buf.begin(), m_buf.begin() + i); - } - - template - void x_write(NumberType x, /*is_integral=*/std::false_type) - { - // special case for 0.0 and -0.0 - if (x == 0) - { - size_t i = 0; - if (std::signbit(x)) - { - m_buf[i++] = '-'; - } - m_buf[i++] = '0'; - m_buf[i++] = '.'; - m_buf[i] = '0'; - return; - } - - // get number of digits for a text -> float -> text round-trip - static constexpr auto d = std::numeric_limits::digits10; - - // the actual conversion - const auto written_bytes = snprintf(m_buf.data(), m_buf.size(), "%.*g", d, x); - - // negative value indicates an error - assert(written_bytes > 0); - // check if buffer was large enough - assert(static_cast(written_bytes) < m_buf.size()); - - // read information from locale - const auto loc = localeconv(); - assert(loc != nullptr); - const char thousands_sep = !loc->thousands_sep ? '\0' - : loc->thousands_sep[0]; - - const char decimal_point = !loc->decimal_point ? '\0' - : loc->decimal_point[0]; - - // erase thousands separator - if (thousands_sep != '\0') - { - const auto end = std::remove(m_buf.begin(), m_buf.begin() + written_bytes, thousands_sep); - std::fill(end, m_buf.end(), '\0'); - } - - // convert decimal point to '.' - if (decimal_point != '\0' and decimal_point != '.') - { - for (auto& c : m_buf) - { - if (c == decimal_point) - { - c = '.'; - break; - } - } - } - - // determine if need to append ".0" - size_t i = 0; - bool value_is_int_like = true; - for (i = 0; i < m_buf.size(); ++i) - { - // break when end of number is reached - if (m_buf[i] == '\0') - { - break; - } - - // check if we find non-int character - value_is_int_like = value_is_int_like and m_buf[i] != '.' and - m_buf[i] != 'e' and m_buf[i] != 'E'; - } - - if (value_is_int_like) - { - // there must be 2 bytes left for ".0" - assert((i + 2) < m_buf.size()); - // we write to the end of the number - assert(m_buf[i] == '\0'); - assert(m_buf[i - 1] != '\0'); - - // add ".0" - m_buf[i] = '.'; - m_buf[i + 1] = '0'; - - // the resulting string is properly terminated - assert(m_buf[i + 2] == '\0'); - } - } - }; - - - /*! - @brief internal implementation of the serialization function - - This function is called by the public member function dump and organizes - the serialization internally. The indentation level is propagated as - additional parameter. In case of arrays and objects, the function is - called recursively. Note that - - - strings and object keys are escaped using `escape_string()` - - integer numbers are converted implicitly via `operator<<` - - floating-point numbers are converted to a string using `"%g"` format - - @param[out] o stream to write to - @param[in] pretty_print whether the output shall be pretty-printed - @param[in] indent_step the indent level - @param[in] current_indent the current indent level (only used internally) - */ - void dump(std::ostream& o, - const bool pretty_print, - const unsigned int indent_step, - const unsigned int current_indent = 0) const - { - switch (m_type) - { - case value_t::object: - { - if (m_value.object->empty()) - { - o.write("{}", 2); - return; - } - - if (pretty_print) - { - o.write("{\n", 2); - - // variable to hold indentation for recursive calls - const auto new_indent = current_indent + indent_step; - string_t indent_string = string_t(new_indent, ' '); - - // first n-1 elements - auto i = m_value.object->cbegin(); - for (size_t cnt = 0; cnt < m_value.object->size() - 1; ++cnt, ++i) - { - o.write(indent_string.c_str(), new_indent); - o.put('\"'); - const auto s = escape_string(i->first); - o.write(s.c_str(), static_cast(s.size())); - o.write("\": ", 3); - i->second.dump(o, true, indent_step, new_indent); - o.write(",\n", 2); - } - - // last element - assert(i != m_value.object->cend()); - o.write(indent_string.c_str(), new_indent); - o.put('\"'); - const auto s = escape_string(i->first); - o.write(s.c_str(), static_cast(s.size())); - o.write("\": ", 3); - i->second.dump(o, true, indent_step, new_indent); - - o.put('\n'); - o.write(indent_string.c_str(), current_indent); - o.put('}'); - } - else - { - o.put('{'); - - // first n-1 elements - auto i = m_value.object->cbegin(); - for (size_t cnt = 0; cnt < m_value.object->size() - 1; ++cnt, ++i) - { - o.put('\"'); - const auto s = escape_string(i->first); - o.write(s.c_str(), static_cast(s.size())); - o.write("\":", 2); - i->second.dump(o, false, indent_step, current_indent); - o.put(','); - } - - // last element - assert(i != m_value.object->cend()); - o.put('\"'); - const auto s = escape_string(i->first); - o.write(s.c_str(), static_cast(s.size())); - o.write("\":", 2); - i->second.dump(o, false, indent_step, current_indent); - - o.put('}'); - } - - return; - } - - case value_t::array: - { - if (m_value.array->empty()) - { - o.write("[]", 2); - return; - } - - if (pretty_print) - { - o.write("[\n", 2); - - // variable to hold indentation for recursive calls - const auto new_indent = current_indent + indent_step; - string_t indent_string = string_t(new_indent, ' '); - - // first n-1 elements - for (auto i = m_value.array->cbegin(); i != m_value.array->cend() - 1; ++i) - { - o.write(indent_string.c_str(), new_indent); - i->dump(o, true, indent_step, new_indent); - o.write(",\n", 2); - } - - // last element - assert(not m_value.array->empty()); - o.write(indent_string.c_str(), new_indent); - m_value.array->back().dump(o, true, indent_step, new_indent); - - o.put('\n'); - o.write(indent_string.c_str(), current_indent); - o.put(']'); - } - else - { - o.put('['); - - // first n-1 elements - for (auto i = m_value.array->cbegin(); i != m_value.array->cend() - 1; ++i) - { - i->dump(o, false, indent_step, current_indent); - o.put(','); - } - - // last element - assert(not m_value.array->empty()); - m_value.array->back().dump(o, false, indent_step, current_indent); - - o.put(']'); - } - - return; - } - - case value_t::string: - { - o.put('\"'); - const auto s = escape_string(*m_value.string); - o.write(s.c_str(), static_cast(s.size())); - o.put('\"'); - return; - } - - case value_t::boolean: - { - if (m_value.boolean) - { - o.write("true", 4); - } - else - { - o.write("false", 5); - } - return; - } - - case value_t::number_integer: - { - o << numtostr(m_value.number_integer).c_str(); - return; - } - - case value_t::number_unsigned: - { - o << numtostr(m_value.number_unsigned).c_str(); - return; - } - - case value_t::number_float: - { - o << numtostr(m_value.number_float).c_str(); - return; - } - - case value_t::discarded: - { - o.write("", 11); - return; - } - - case value_t::null: - { - o.write("null", 4); - return; - } - } - } private: ////////////////////// diff --git a/test/src/unit-convenience.cpp b/test/src/unit-convenience.cpp index 891dbc14..45637033 100644 --- a/test/src/unit-convenience.cpp +++ b/test/src/unit-convenience.cpp @@ -49,44 +49,44 @@ TEST_CASE("convenience functions") SECTION("string escape") { - CHECK(json::escape_string("\"") == "\\\""); - CHECK(json::escape_string("\\") == "\\\\"); - CHECK(json::escape_string("\b") == "\\b"); - CHECK(json::escape_string("\f") == "\\f"); - CHECK(json::escape_string("\n") == "\\n"); - CHECK(json::escape_string("\r") == "\\r"); - CHECK(json::escape_string("\t") == "\\t"); + CHECK(json::serializer::escape_string("\"") == "\\\""); + CHECK(json::serializer::escape_string("\\") == "\\\\"); + CHECK(json::serializer::escape_string("\b") == "\\b"); + CHECK(json::serializer::escape_string("\f") == "\\f"); + CHECK(json::serializer::escape_string("\n") == "\\n"); + CHECK(json::serializer::escape_string("\r") == "\\r"); + CHECK(json::serializer::escape_string("\t") == "\\t"); - CHECK(json::escape_string("\x01") == "\\u0001"); - CHECK(json::escape_string("\x02") == "\\u0002"); - CHECK(json::escape_string("\x03") == "\\u0003"); - CHECK(json::escape_string("\x04") == "\\u0004"); - CHECK(json::escape_string("\x05") == "\\u0005"); - CHECK(json::escape_string("\x06") == "\\u0006"); - CHECK(json::escape_string("\x07") == "\\u0007"); - CHECK(json::escape_string("\x08") == "\\b"); - CHECK(json::escape_string("\x09") == "\\t"); - CHECK(json::escape_string("\x0a") == "\\n"); - CHECK(json::escape_string("\x0b") == "\\u000b"); - CHECK(json::escape_string("\x0c") == "\\f"); - CHECK(json::escape_string("\x0d") == "\\r"); - CHECK(json::escape_string("\x0e") == "\\u000e"); - CHECK(json::escape_string("\x0f") == "\\u000f"); - CHECK(json::escape_string("\x10") == "\\u0010"); - CHECK(json::escape_string("\x11") == "\\u0011"); - CHECK(json::escape_string("\x12") == "\\u0012"); - CHECK(json::escape_string("\x13") == "\\u0013"); - CHECK(json::escape_string("\x14") == "\\u0014"); - CHECK(json::escape_string("\x15") == "\\u0015"); - CHECK(json::escape_string("\x16") == "\\u0016"); - CHECK(json::escape_string("\x17") == "\\u0017"); - CHECK(json::escape_string("\x18") == "\\u0018"); - CHECK(json::escape_string("\x19") == "\\u0019"); - CHECK(json::escape_string("\x1a") == "\\u001a"); - CHECK(json::escape_string("\x1b") == "\\u001b"); - CHECK(json::escape_string("\x1c") == "\\u001c"); - CHECK(json::escape_string("\x1d") == "\\u001d"); - CHECK(json::escape_string("\x1e") == "\\u001e"); - CHECK(json::escape_string("\x1f") == "\\u001f"); + CHECK(json::serializer::escape_string("\x01") == "\\u0001"); + CHECK(json::serializer::escape_string("\x02") == "\\u0002"); + CHECK(json::serializer::escape_string("\x03") == "\\u0003"); + CHECK(json::serializer::escape_string("\x04") == "\\u0004"); + CHECK(json::serializer::escape_string("\x05") == "\\u0005"); + CHECK(json::serializer::escape_string("\x06") == "\\u0006"); + CHECK(json::serializer::escape_string("\x07") == "\\u0007"); + CHECK(json::serializer::escape_string("\x08") == "\\b"); + CHECK(json::serializer::escape_string("\x09") == "\\t"); + CHECK(json::serializer::escape_string("\x0a") == "\\n"); + CHECK(json::serializer::escape_string("\x0b") == "\\u000b"); + CHECK(json::serializer::escape_string("\x0c") == "\\f"); + CHECK(json::serializer::escape_string("\x0d") == "\\r"); + CHECK(json::serializer::escape_string("\x0e") == "\\u000e"); + CHECK(json::serializer::escape_string("\x0f") == "\\u000f"); + CHECK(json::serializer::escape_string("\x10") == "\\u0010"); + CHECK(json::serializer::escape_string("\x11") == "\\u0011"); + CHECK(json::serializer::escape_string("\x12") == "\\u0012"); + CHECK(json::serializer::escape_string("\x13") == "\\u0013"); + CHECK(json::serializer::escape_string("\x14") == "\\u0014"); + CHECK(json::serializer::escape_string("\x15") == "\\u0015"); + CHECK(json::serializer::escape_string("\x16") == "\\u0016"); + CHECK(json::serializer::escape_string("\x17") == "\\u0017"); + CHECK(json::serializer::escape_string("\x18") == "\\u0018"); + CHECK(json::serializer::escape_string("\x19") == "\\u0019"); + CHECK(json::serializer::escape_string("\x1a") == "\\u001a"); + CHECK(json::serializer::escape_string("\x1b") == "\\u001b"); + CHECK(json::serializer::escape_string("\x1c") == "\\u001c"); + CHECK(json::serializer::escape_string("\x1d") == "\\u001d"); + CHECK(json::serializer::escape_string("\x1e") == "\\u001e"); + CHECK(json::serializer::escape_string("\x1f") == "\\u001f"); } } From af070744aecf944598c8655f5460779a63e66b40 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Mon, 27 Feb 2017 22:10:57 +0100 Subject: [PATCH 012/530] :hammer: integrating numtostr into serializer class By merging numtostr into serializer, we can write directly to the output stream. As a consequence, all stream calls are now unformatted. --- src/json.hpp | 220 ++++++++++++++++++++-------------------------- src/json.hpp.re2c | 216 +++++++++++++++++++-------------------------- 2 files changed, 185 insertions(+), 251 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index a73b5048..3e5a45cf 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -6196,6 +6196,9 @@ class basic_json /// @{ private: + /*! + @brief wrapper around the serialization functions + */ class serializer { public: @@ -6223,7 +6226,7 @@ class basic_json void dump(const basic_json& val, const bool pretty_print, const unsigned int indent_step, - const unsigned int current_indent = 0) const + const unsigned int current_indent = 0) { switch (val.m_type) { @@ -6377,19 +6380,19 @@ class basic_json case value_t::number_integer: { - o << numtostr(val.m_value.number_integer).c_str(); + x_write(val.m_value.number_integer); return; } case value_t::number_unsigned: { - o << numtostr(val.m_value.number_unsigned).c_str(); + x_write(val.m_value.number_unsigned); return; } case value_t::number_float: { - o << numtostr(val.m_value.number_float).c_str(); + x_write(val.m_value.number_float); return; } @@ -6569,154 +6572,120 @@ class basic_json return result; } - /*! - @brief locale-independent serialization for built-in arithmetic types - */ - struct numtostr + template + void x_write(NumberType x) { - public: - template - numtostr(NumberType value) + // special case for "0" + if (x == 0) { - x_write(value, std::is_integral()); + o.put('0'); + return; } - const char* c_str() const + const bool is_negative = x < 0; + size_t i = 0; + + // spare 1 byte for '\0' + while (x != 0 and i < m_buf.size() - 1) { - return m_buf.data(); + const auto digit = std::labs(static_cast(x % 10)); + m_buf[i++] = static_cast('0' + digit); + x /= 10; } - private: - /// a (hopefully) large enough character buffer - std::array < char, 64 > m_buf{{}}; + // make sure the number has been processed completely + assert(x == 0); - template - void x_write(NumberType x, /*is_integral=*/std::true_type) + if (is_negative) { - // special case for "0" - if (x == 0) - { - m_buf[0] = '0'; - return; - } - - const bool is_negative = x < 0; - size_t i = 0; - - // spare 1 byte for '\0' - while (x != 0 and i < m_buf.size() - 1) - { - const auto digit = std::labs(static_cast(x % 10)); - m_buf[i++] = static_cast('0' + digit); - x /= 10; - } - - // make sure the number has been processed completely - assert(x == 0); - - if (is_negative) - { - // make sure there is capacity for the '-' - assert(i < m_buf.size() - 2); - m_buf[i++] = '-'; - } - - std::reverse(m_buf.begin(), m_buf.begin() + i); + // make sure there is capacity for the '-' + assert(i < m_buf.size() - 2); + m_buf[i++] = '-'; } - template - void x_write(NumberType x, /*is_integral=*/std::false_type) + std::reverse(m_buf.begin(), m_buf.begin() + i); + o.write(m_buf.data(), static_cast(i)); + } + + void x_write(number_float_t x) + { + // special case for 0.0 and -0.0 + if (x == 0) { - // special case for 0.0 and -0.0 - if (x == 0) + if (std::signbit(x)) { - size_t i = 0; - if (std::signbit(x)) - { - m_buf[i++] = '-'; - } - m_buf[i++] = '0'; - m_buf[i++] = '.'; - m_buf[i] = '0'; - return; + o.write("-0.0", 4); } - - // get number of digits for a text -> float -> text round-trip - static constexpr auto d = std::numeric_limits::digits10; - - // the actual conversion - const auto written_bytes = snprintf(m_buf.data(), m_buf.size(), "%.*g", d, x); - - // negative value indicates an error - assert(written_bytes > 0); - // check if buffer was large enough - assert(static_cast(written_bytes) < m_buf.size()); - - // read information from locale - const auto loc = localeconv(); - assert(loc != nullptr); - const char thousands_sep = !loc->thousands_sep ? '\0' - : loc->thousands_sep[0]; - - const char decimal_point = !loc->decimal_point ? '\0' - : loc->decimal_point[0]; - - // erase thousands separator - if (thousands_sep != '\0') + else { - const auto end = std::remove(m_buf.begin(), m_buf.begin() + written_bytes, thousands_sep); - std::fill(end, m_buf.end(), '\0'); + o.write("0.0", 3); } + return; + } - // convert decimal point to '.' - if (decimal_point != '\0' and decimal_point != '.') - { - for (auto& c : m_buf) - { - if (c == decimal_point) - { - c = '.'; - break; - } - } - } + // get number of digits for a text -> float -> text round-trip + static constexpr auto d = std::numeric_limits::digits10; - // determine if need to append ".0" - size_t i = 0; - bool value_is_int_like = true; - for (i = 0; i < m_buf.size(); ++i) + // the actual conversion + auto written_bytes = snprintf(m_buf.data(), m_buf.size(), "%.*g", d, x); + + // negative value indicates an error + assert(written_bytes > 0); + // check if buffer was large enough + assert(static_cast(written_bytes) < m_buf.size()); + + // read information from locale + const auto loc = localeconv(); + assert(loc != nullptr); + const char thousands_sep = !loc->thousands_sep ? '\0' + : loc->thousands_sep[0]; + + const char decimal_point = !loc->decimal_point ? '\0' + : loc->decimal_point[0]; + + // erase thousands separator + if (thousands_sep != '\0') + { + const auto end = std::remove(m_buf.begin(), m_buf.begin() + written_bytes, thousands_sep); + std::fill(end, m_buf.end(), '\0'); + written_bytes -= (m_buf.end() - end); + } + + // convert decimal point to '.' + if (decimal_point != '\0' and decimal_point != '.') + { + for (auto& c : m_buf) { - // break when end of number is reached - if (m_buf[i] == '\0') + if (c == decimal_point) { + c = '.'; break; } - - // check if we find non-int character - value_is_int_like = value_is_int_like and m_buf[i] != '.' and - m_buf[i] != 'e' and m_buf[i] != 'E'; - } - - if (value_is_int_like) - { - // there must be 2 bytes left for ".0" - assert((i + 2) < m_buf.size()); - // we write to the end of the number - assert(m_buf[i] == '\0'); - assert(m_buf[i - 1] != '\0'); - - // add ".0" - m_buf[i] = '.'; - m_buf[i + 1] = '0'; - - // the resulting string is properly terminated - assert(m_buf[i + 2] == '\0'); } } - }; + + // determine if need to append ".0" + bool value_is_int_like = true; + for (size_t i = 0; i < static_cast(written_bytes); ++i) + { + // check if we find non-int character + value_is_int_like = value_is_int_like and m_buf[i] != '.' and + m_buf[i] != 'e'; + } + + o.write(m_buf.data(), static_cast(written_bytes)); + + if (value_is_int_like) + { + o.write(".0", 2); + } + } private: std::ostream& o; + + /// a (hopefully) large enough character buffer + std::array < char, 64 > m_buf{{}}; }; public: @@ -6754,7 +6723,6 @@ class basic_json // do the actual serialization serializer s(o); s.dump(j, pretty_print, static_cast(indentation)); - return o; } diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 01f6bbfe..e648b41b 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -6226,7 +6226,7 @@ class basic_json void dump(const basic_json& val, const bool pretty_print, const unsigned int indent_step, - const unsigned int current_indent = 0) const + const unsigned int current_indent = 0) { switch (val.m_type) { @@ -6380,19 +6380,19 @@ class basic_json case value_t::number_integer: { - o << numtostr(val.m_value.number_integer).c_str(); + x_write(val.m_value.number_integer); return; } case value_t::number_unsigned: { - o << numtostr(val.m_value.number_unsigned).c_str(); + x_write(val.m_value.number_unsigned); return; } case value_t::number_float: { - o << numtostr(val.m_value.number_float).c_str(); + x_write(val.m_value.number_float); return; } @@ -6572,154 +6572,120 @@ class basic_json return result; } - /*! - @brief locale-independent serialization for built-in arithmetic types - */ - struct numtostr + template + void x_write(NumberType x) { - public: - template - numtostr(NumberType value) + // special case for "0" + if (x == 0) { - x_write(value, std::is_integral()); + o.put('0'); + return; } - const char* c_str() const + const bool is_negative = x < 0; + size_t i = 0; + + // spare 1 byte for '\0' + while (x != 0 and i < m_buf.size() - 1) { - return m_buf.data(); + const auto digit = std::labs(static_cast(x % 10)); + m_buf[i++] = static_cast('0' + digit); + x /= 10; } - private: - /// a (hopefully) large enough character buffer - std::array < char, 64 > m_buf{{}}; + // make sure the number has been processed completely + assert(x == 0); - template - void x_write(NumberType x, /*is_integral=*/std::true_type) + if (is_negative) { - // special case for "0" - if (x == 0) - { - m_buf[0] = '0'; - return; - } - - const bool is_negative = x < 0; - size_t i = 0; - - // spare 1 byte for '\0' - while (x != 0 and i < m_buf.size() - 1) - { - const auto digit = std::labs(static_cast(x % 10)); - m_buf[i++] = static_cast('0' + digit); - x /= 10; - } - - // make sure the number has been processed completely - assert(x == 0); - - if (is_negative) - { - // make sure there is capacity for the '-' - assert(i < m_buf.size() - 2); - m_buf[i++] = '-'; - } - - std::reverse(m_buf.begin(), m_buf.begin() + i); + // make sure there is capacity for the '-' + assert(i < m_buf.size() - 2); + m_buf[i++] = '-'; } - template - void x_write(NumberType x, /*is_integral=*/std::false_type) + std::reverse(m_buf.begin(), m_buf.begin() + i); + o.write(m_buf.data(), static_cast(i)); + } + + void x_write(number_float_t x) + { + // special case for 0.0 and -0.0 + if (x == 0) { - // special case for 0.0 and -0.0 - if (x == 0) + if (std::signbit(x)) { - size_t i = 0; - if (std::signbit(x)) - { - m_buf[i++] = '-'; - } - m_buf[i++] = '0'; - m_buf[i++] = '.'; - m_buf[i] = '0'; - return; + o.write("-0.0", 4); } - - // get number of digits for a text -> float -> text round-trip - static constexpr auto d = std::numeric_limits::digits10; - - // the actual conversion - const auto written_bytes = snprintf(m_buf.data(), m_buf.size(), "%.*g", d, x); - - // negative value indicates an error - assert(written_bytes > 0); - // check if buffer was large enough - assert(static_cast(written_bytes) < m_buf.size()); - - // read information from locale - const auto loc = localeconv(); - assert(loc != nullptr); - const char thousands_sep = !loc->thousands_sep ? '\0' - : loc->thousands_sep[0]; - - const char decimal_point = !loc->decimal_point ? '\0' - : loc->decimal_point[0]; - - // erase thousands separator - if (thousands_sep != '\0') + else { - const auto end = std::remove(m_buf.begin(), m_buf.begin() + written_bytes, thousands_sep); - std::fill(end, m_buf.end(), '\0'); + o.write("0.0", 3); } + return; + } - // convert decimal point to '.' - if (decimal_point != '\0' and decimal_point != '.') - { - for (auto& c : m_buf) - { - if (c == decimal_point) - { - c = '.'; - break; - } - } - } + // get number of digits for a text -> float -> text round-trip + static constexpr auto d = std::numeric_limits::digits10; - // determine if need to append ".0" - size_t i = 0; - bool value_is_int_like = true; - for (i = 0; i < m_buf.size(); ++i) + // the actual conversion + auto written_bytes = snprintf(m_buf.data(), m_buf.size(), "%.*g", d, x); + + // negative value indicates an error + assert(written_bytes > 0); + // check if buffer was large enough + assert(static_cast(written_bytes) < m_buf.size()); + + // read information from locale + const auto loc = localeconv(); + assert(loc != nullptr); + const char thousands_sep = !loc->thousands_sep ? '\0' + : loc->thousands_sep[0]; + + const char decimal_point = !loc->decimal_point ? '\0' + : loc->decimal_point[0]; + + // erase thousands separator + if (thousands_sep != '\0') + { + const auto end = std::remove(m_buf.begin(), m_buf.begin() + written_bytes, thousands_sep); + std::fill(end, m_buf.end(), '\0'); + written_bytes -= (m_buf.end() - end); + } + + // convert decimal point to '.' + if (decimal_point != '\0' and decimal_point != '.') + { + for (auto& c : m_buf) { - // break when end of number is reached - if (m_buf[i] == '\0') + if (c == decimal_point) { + c = '.'; break; } - - // check if we find non-int character - value_is_int_like = value_is_int_like and m_buf[i] != '.' and - m_buf[i] != 'e' and m_buf[i] != 'E'; - } - - if (value_is_int_like) - { - // there must be 2 bytes left for ".0" - assert((i + 2) < m_buf.size()); - // we write to the end of the number - assert(m_buf[i] == '\0'); - assert(m_buf[i - 1] != '\0'); - - // add ".0" - m_buf[i] = '.'; - m_buf[i + 1] = '0'; - - // the resulting string is properly terminated - assert(m_buf[i + 2] == '\0'); } } - }; + + // determine if need to append ".0" + bool value_is_int_like = true; + for (size_t i = 0; i < static_cast(written_bytes); ++i) + { + // check if we find non-int character + value_is_int_like = value_is_int_like and m_buf[i] != '.' and + m_buf[i] != 'e'; + } + + o.write(m_buf.data(), static_cast(written_bytes)); + + if (value_is_int_like) + { + o.write(".0", 2); + } + } private: std::ostream& o; + + /// a (hopefully) large enough character buffer + std::array < char, 64 > m_buf{{}}; }; public: From fc48b8ac2b595a8df5ad121b588d8820b3b6941d Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 28 Feb 2017 11:45:38 +0100 Subject: [PATCH 013/530] :bug: fixed a logical error Treated the size of the range as the number of thousand separators. This logical error yielded a negative value for written_bytes and eventually an infinite loop, as written_bytes was converted to an unsigned value. --- src/json.hpp | 3 ++- src/json.hpp.re2c | 3 ++- 2 files changed, 4 insertions(+), 2 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 3e5a45cf..5cf06249 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -6648,7 +6648,8 @@ class basic_json { const auto end = std::remove(m_buf.begin(), m_buf.begin() + written_bytes, thousands_sep); std::fill(end, m_buf.end(), '\0'); - written_bytes -= (m_buf.end() - end); + assert((end - m_buf.begin()) <= written_bytes); + written_bytes = (end - m_buf.begin()); } // convert decimal point to '.' diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index e648b41b..c7784266 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -6648,7 +6648,8 @@ class basic_json { const auto end = std::remove(m_buf.begin(), m_buf.begin() + written_bytes, thousands_sep); std::fill(end, m_buf.end(), '\0'); - written_bytes -= (m_buf.end() - end); + assert((end - m_buf.begin()) <= written_bytes); + written_bytes = (end - m_buf.begin()); } // convert decimal point to '.' From 224f99070b8083e2c574014c1c7c13607795ec5a Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 28 Feb 2017 16:28:22 +0100 Subject: [PATCH 014/530] :zap: micro-optimization of dump() A lot of small changes to avoid memory allocations: - The locale is only queried once rather than with every number serialization. - The indentation string is recycled between different calls. - The string escape function avoids a copy if no escaping is necessary. - The string escape and the space function use a complete switch case instead of cascaded ifs. Cachegrind measures some 15% performance improvement. --- src/json.hpp | 176 ++++++++++++++++++++++------------ src/json.hpp.re2c | 176 ++++++++++++++++++++++------------ test/src/unit-convenience.cpp | 85 ++++++++-------- 3 files changed, 277 insertions(+), 160 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 5cf06249..8d545a93 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -34,6 +34,7 @@ SOFTWARE. #include // assert #include // isdigit #include // and, not, or +#include // lconv, localeconv #include // isfinite, labs, ldexp, signbit #include // nullptr_t, ptrdiff_t, size_t #include // int64_t, uint64_t @@ -6203,7 +6204,9 @@ class basic_json { public: serializer(std::ostream& s) - : o(s) + : o(s), loc(std::localeconv()), + thousands_sep(!loc->thousands_sep ? '\0' : loc->thousands_sep[0]), + decimal_point(!loc->decimal_point ? '\0' : loc->decimal_point[0]) {} /*! @@ -6244,7 +6247,10 @@ class basic_json // variable to hold indentation for recursive calls const auto new_indent = current_indent + indent_step; - string_t indent_string = string_t(new_indent, ' '); + if (indent_string.size() < new_indent) + { + indent_string.resize(new_indent, ' '); + } // first n-1 elements auto i = val.m_value.object->cbegin(); @@ -6252,8 +6258,7 @@ class basic_json { o.write(indent_string.c_str(), new_indent); o.put('\"'); - const auto s = escape_string(i->first); - o.write(s.c_str(), static_cast(s.size())); + dump_escaped(i->first); o.write("\": ", 3); dump(i->second, true, indent_step, new_indent); o.write(",\n", 2); @@ -6263,8 +6268,7 @@ class basic_json assert(i != val.m_value.object->cend()); o.write(indent_string.c_str(), new_indent); o.put('\"'); - const auto s = escape_string(i->first); - o.write(s.c_str(), static_cast(s.size())); + dump_escaped(i->first); o.write("\": ", 3); dump(i->second, true, indent_step, new_indent); @@ -6281,8 +6285,7 @@ class basic_json for (size_t cnt = 0; cnt < val.m_value.object->size() - 1; ++cnt, ++i) { o.put('\"'); - const auto s = escape_string(i->first); - o.write(s.c_str(), static_cast(s.size())); + dump_escaped(i->first); o.write("\":", 2); dump(i->second, false, indent_step, current_indent); o.put(','); @@ -6291,8 +6294,7 @@ class basic_json // last element assert(i != val.m_value.object->cend()); o.put('\"'); - const auto s = escape_string(i->first); - o.write(s.c_str(), static_cast(s.size())); + dump_escaped(i->first); o.write("\":", 2); dump(i->second, false, indent_step, current_indent); @@ -6316,7 +6318,10 @@ class basic_json // variable to hold indentation for recursive calls const auto new_indent = current_indent + indent_step; - string_t indent_string = string_t(new_indent, ' '); + if (indent_string.size() < new_indent) + { + indent_string.resize(new_indent, ' '); + } // first n-1 elements for (auto i = val.m_value.array->cbegin(); i != val.m_value.array->cend() - 1; ++i) @@ -6359,8 +6364,7 @@ class basic_json case value_t::string: { o.put('\"'); - const auto s = escape_string(*val.m_value.string); - o.write(s.c_str(), static_cast(s.size())); + dump_escaped(*val.m_value.string); o.put('\"'); return; } @@ -6380,19 +6384,19 @@ class basic_json case value_t::number_integer: { - x_write(val.m_value.number_integer); + dump_integer(val.m_value.number_integer); return; } case value_t::number_unsigned: { - x_write(val.m_value.number_unsigned); + dump_integer(val.m_value.number_unsigned); return; } case value_t::number_float: { - x_write(val.m_value.number_float); + dump_float(val.m_value.number_float); return; } @@ -6438,14 +6442,40 @@ class basic_json return res + 1; } + case 0x00: + case 0x01: + case 0x02: + case 0x03: + case 0x04: + case 0x05: + case 0x06: + case 0x07: + case 0x0b: + case 0x0e: + case 0x0f: + case 0x10: + case 0x11: + case 0x12: + case 0x13: + case 0x14: + case 0x15: + case 0x16: + case 0x17: + case 0x18: + case 0x19: + case 0x1a: + case 0x1b: + case 0x1c: + case 0x1d: + case 0x1e: + case 0x1f: + { + // from c (1 byte) to \uxxxx (6 bytes) + return res + 5; + } + default: { - if (c >= 0x00 and c <= 0x1f) - { - // from c (1 byte) to \uxxxx (6 bytes) - return res + 5; - } - return res; } } @@ -6465,12 +6495,13 @@ class basic_json @complexity Linear in the length of string @a s. */ - static string_t escape_string(const string_t& s) + void dump_escaped(const string_t& s) const { const auto space = extra_space(s); if (space == 0) { - return s; + o.write(s.c_str(), static_cast(s.size())); + return; } // create a result string of necessary size @@ -6537,43 +6568,69 @@ class basic_json break; } + case 0x00: + case 0x01: + case 0x02: + case 0x03: + case 0x04: + case 0x05: + case 0x06: + case 0x07: + case 0x0b: + case 0x0e: + case 0x0f: + case 0x10: + case 0x11: + case 0x12: + case 0x13: + case 0x14: + case 0x15: + case 0x16: + case 0x17: + case 0x18: + case 0x19: + case 0x1a: + case 0x1b: + case 0x1c: + case 0x1d: + case 0x1e: + case 0x1f: + { + // convert a number 0..15 to its hex representation + // (0..f) + static const char hexify[16] = + { + '0', '1', '2', '3', '4', '5', '6', '7', + '8', '9', 'a', 'b', 'c', 'd', 'e', 'f' + }; + + // print character c as \uxxxx + for (const char m : + { 'u', '0', '0', hexify[c >> 4], hexify[c & 0x0f] + }) + { + result[++pos] = m; + } + + ++pos; + break; + } + default: { - if (c >= 0x00 and c <= 0x1f) - { - // convert a number 0..15 to its hex representation - // (0..f) - static const char hexify[16] = - { - '0', '1', '2', '3', '4', '5', '6', '7', - '8', '9', 'a', 'b', 'c', 'd', 'e', 'f' - }; - - // print character c as \uxxxx - for (const char m : - { 'u', '0', '0', hexify[c >> 4], hexify[c & 0x0f] - }) - { - result[++pos] = m; - } - - ++pos; - } - else - { - // all other characters are added as-is - result[pos++] = c; - } + // all other characters are added as-is + result[pos++] = c; break; } } } - return result; + assert(pos == s.size() + space); + o.write(result.c_str(), static_cast(result.size())); } template - void x_write(NumberType x) + void dump_integer(NumberType x) { // special case for "0" if (x == 0) @@ -6607,7 +6664,7 @@ class basic_json o.write(m_buf.data(), static_cast(i)); } - void x_write(number_float_t x) + void dump_float(number_float_t x) { // special case for 0.0 and -0.0 if (x == 0) @@ -6634,15 +6691,6 @@ class basic_json // check if buffer was large enough assert(static_cast(written_bytes) < m_buf.size()); - // read information from locale - const auto loc = localeconv(); - assert(loc != nullptr); - const char thousands_sep = !loc->thousands_sep ? '\0' - : loc->thousands_sep[0]; - - const char decimal_point = !loc->decimal_point ? '\0' - : loc->decimal_point[0]; - // erase thousands separator if (thousands_sep != '\0') { @@ -6687,6 +6735,12 @@ class basic_json /// a (hopefully) large enough character buffer std::array < char, 64 > m_buf{{}}; + + const std::lconv* loc = nullptr; + const char thousands_sep = '\0'; + const char decimal_point = '\0'; + + string_t indent_string = string_t(512, ' '); }; public: diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index c7784266..b14fb68d 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -34,6 +34,7 @@ SOFTWARE. #include // assert #include // isdigit #include // and, not, or +#include // lconv, localeconv #include // isfinite, labs, ldexp, signbit #include // nullptr_t, ptrdiff_t, size_t #include // int64_t, uint64_t @@ -6203,7 +6204,9 @@ class basic_json { public: serializer(std::ostream& s) - : o(s) + : o(s), loc(std::localeconv()), + thousands_sep(!loc->thousands_sep ? '\0' : loc->thousands_sep[0]), + decimal_point(!loc->decimal_point ? '\0' : loc->decimal_point[0]) {} /*! @@ -6244,7 +6247,10 @@ class basic_json // variable to hold indentation for recursive calls const auto new_indent = current_indent + indent_step; - string_t indent_string = string_t(new_indent, ' '); + if (indent_string.size() < new_indent) + { + indent_string.resize(new_indent, ' '); + } // first n-1 elements auto i = val.m_value.object->cbegin(); @@ -6252,8 +6258,7 @@ class basic_json { o.write(indent_string.c_str(), new_indent); o.put('\"'); - const auto s = escape_string(i->first); - o.write(s.c_str(), static_cast(s.size())); + dump_escaped(i->first); o.write("\": ", 3); dump(i->second, true, indent_step, new_indent); o.write(",\n", 2); @@ -6263,8 +6268,7 @@ class basic_json assert(i != val.m_value.object->cend()); o.write(indent_string.c_str(), new_indent); o.put('\"'); - const auto s = escape_string(i->first); - o.write(s.c_str(), static_cast(s.size())); + dump_escaped(i->first); o.write("\": ", 3); dump(i->second, true, indent_step, new_indent); @@ -6281,8 +6285,7 @@ class basic_json for (size_t cnt = 0; cnt < val.m_value.object->size() - 1; ++cnt, ++i) { o.put('\"'); - const auto s = escape_string(i->first); - o.write(s.c_str(), static_cast(s.size())); + dump_escaped(i->first); o.write("\":", 2); dump(i->second, false, indent_step, current_indent); o.put(','); @@ -6291,8 +6294,7 @@ class basic_json // last element assert(i != val.m_value.object->cend()); o.put('\"'); - const auto s = escape_string(i->first); - o.write(s.c_str(), static_cast(s.size())); + dump_escaped(i->first); o.write("\":", 2); dump(i->second, false, indent_step, current_indent); @@ -6316,7 +6318,10 @@ class basic_json // variable to hold indentation for recursive calls const auto new_indent = current_indent + indent_step; - string_t indent_string = string_t(new_indent, ' '); + if (indent_string.size() < new_indent) + { + indent_string.resize(new_indent, ' '); + } // first n-1 elements for (auto i = val.m_value.array->cbegin(); i != val.m_value.array->cend() - 1; ++i) @@ -6359,8 +6364,7 @@ class basic_json case value_t::string: { o.put('\"'); - const auto s = escape_string(*val.m_value.string); - o.write(s.c_str(), static_cast(s.size())); + dump_escaped(*val.m_value.string); o.put('\"'); return; } @@ -6380,19 +6384,19 @@ class basic_json case value_t::number_integer: { - x_write(val.m_value.number_integer); + dump_integer(val.m_value.number_integer); return; } case value_t::number_unsigned: { - x_write(val.m_value.number_unsigned); + dump_integer(val.m_value.number_unsigned); return; } case value_t::number_float: { - x_write(val.m_value.number_float); + dump_float(val.m_value.number_float); return; } @@ -6438,14 +6442,40 @@ class basic_json return res + 1; } + case 0x00: + case 0x01: + case 0x02: + case 0x03: + case 0x04: + case 0x05: + case 0x06: + case 0x07: + case 0x0b: + case 0x0e: + case 0x0f: + case 0x10: + case 0x11: + case 0x12: + case 0x13: + case 0x14: + case 0x15: + case 0x16: + case 0x17: + case 0x18: + case 0x19: + case 0x1a: + case 0x1b: + case 0x1c: + case 0x1d: + case 0x1e: + case 0x1f: + { + // from c (1 byte) to \uxxxx (6 bytes) + return res + 5; + } + default: { - if (c >= 0x00 and c <= 0x1f) - { - // from c (1 byte) to \uxxxx (6 bytes) - return res + 5; - } - return res; } } @@ -6465,12 +6495,13 @@ class basic_json @complexity Linear in the length of string @a s. */ - static string_t escape_string(const string_t& s) + void dump_escaped(const string_t& s) const { const auto space = extra_space(s); if (space == 0) { - return s; + o.write(s.c_str(), static_cast(s.size())); + return; } // create a result string of necessary size @@ -6537,43 +6568,69 @@ class basic_json break; } + case 0x00: + case 0x01: + case 0x02: + case 0x03: + case 0x04: + case 0x05: + case 0x06: + case 0x07: + case 0x0b: + case 0x0e: + case 0x0f: + case 0x10: + case 0x11: + case 0x12: + case 0x13: + case 0x14: + case 0x15: + case 0x16: + case 0x17: + case 0x18: + case 0x19: + case 0x1a: + case 0x1b: + case 0x1c: + case 0x1d: + case 0x1e: + case 0x1f: + { + // convert a number 0..15 to its hex representation + // (0..f) + static const char hexify[16] = + { + '0', '1', '2', '3', '4', '5', '6', '7', + '8', '9', 'a', 'b', 'c', 'd', 'e', 'f' + }; + + // print character c as \uxxxx + for (const char m : + { 'u', '0', '0', hexify[c >> 4], hexify[c & 0x0f] + }) + { + result[++pos] = m; + } + + ++pos; + break; + } + default: { - if (c >= 0x00 and c <= 0x1f) - { - // convert a number 0..15 to its hex representation - // (0..f) - static const char hexify[16] = - { - '0', '1', '2', '3', '4', '5', '6', '7', - '8', '9', 'a', 'b', 'c', 'd', 'e', 'f' - }; - - // print character c as \uxxxx - for (const char m : - { 'u', '0', '0', hexify[c >> 4], hexify[c & 0x0f] - }) - { - result[++pos] = m; - } - - ++pos; - } - else - { - // all other characters are added as-is - result[pos++] = c; - } + // all other characters are added as-is + result[pos++] = c; break; } } } - return result; + assert(pos == s.size() + space); + o.write(result.c_str(), static_cast(result.size())); } template - void x_write(NumberType x) + void dump_integer(NumberType x) { // special case for "0" if (x == 0) @@ -6607,7 +6664,7 @@ class basic_json o.write(m_buf.data(), static_cast(i)); } - void x_write(number_float_t x) + void dump_float(number_float_t x) { // special case for 0.0 and -0.0 if (x == 0) @@ -6634,15 +6691,6 @@ class basic_json // check if buffer was large enough assert(static_cast(written_bytes) < m_buf.size()); - // read information from locale - const auto loc = localeconv(); - assert(loc != nullptr); - const char thousands_sep = !loc->thousands_sep ? '\0' - : loc->thousands_sep[0]; - - const char decimal_point = !loc->decimal_point ? '\0' - : loc->decimal_point[0]; - // erase thousands separator if (thousands_sep != '\0') { @@ -6687,6 +6735,12 @@ class basic_json /// a (hopefully) large enough character buffer std::array < char, 64 > m_buf{{}}; + + const std::lconv* loc = nullptr; + const char thousands_sep = '\0'; + const char decimal_point = '\0'; + + string_t indent_string = string_t(512, ' '); }; public: diff --git a/test/src/unit-convenience.cpp b/test/src/unit-convenience.cpp index 45637033..33556311 100644 --- a/test/src/unit-convenience.cpp +++ b/test/src/unit-convenience.cpp @@ -49,44 +49,53 @@ TEST_CASE("convenience functions") SECTION("string escape") { - CHECK(json::serializer::escape_string("\"") == "\\\""); - CHECK(json::serializer::escape_string("\\") == "\\\\"); - CHECK(json::serializer::escape_string("\b") == "\\b"); - CHECK(json::serializer::escape_string("\f") == "\\f"); - CHECK(json::serializer::escape_string("\n") == "\\n"); - CHECK(json::serializer::escape_string("\r") == "\\r"); - CHECK(json::serializer::escape_string("\t") == "\\t"); + const auto check_escaped = [](const char* original, + const char* escaped) + { + std::stringstream ss; + json::serializer s(ss); + s.dump_escaped(original); + CHECK(ss.str() == escaped); + }; - CHECK(json::serializer::escape_string("\x01") == "\\u0001"); - CHECK(json::serializer::escape_string("\x02") == "\\u0002"); - CHECK(json::serializer::escape_string("\x03") == "\\u0003"); - CHECK(json::serializer::escape_string("\x04") == "\\u0004"); - CHECK(json::serializer::escape_string("\x05") == "\\u0005"); - CHECK(json::serializer::escape_string("\x06") == "\\u0006"); - CHECK(json::serializer::escape_string("\x07") == "\\u0007"); - CHECK(json::serializer::escape_string("\x08") == "\\b"); - CHECK(json::serializer::escape_string("\x09") == "\\t"); - CHECK(json::serializer::escape_string("\x0a") == "\\n"); - CHECK(json::serializer::escape_string("\x0b") == "\\u000b"); - CHECK(json::serializer::escape_string("\x0c") == "\\f"); - CHECK(json::serializer::escape_string("\x0d") == "\\r"); - CHECK(json::serializer::escape_string("\x0e") == "\\u000e"); - CHECK(json::serializer::escape_string("\x0f") == "\\u000f"); - CHECK(json::serializer::escape_string("\x10") == "\\u0010"); - CHECK(json::serializer::escape_string("\x11") == "\\u0011"); - CHECK(json::serializer::escape_string("\x12") == "\\u0012"); - CHECK(json::serializer::escape_string("\x13") == "\\u0013"); - CHECK(json::serializer::escape_string("\x14") == "\\u0014"); - CHECK(json::serializer::escape_string("\x15") == "\\u0015"); - CHECK(json::serializer::escape_string("\x16") == "\\u0016"); - CHECK(json::serializer::escape_string("\x17") == "\\u0017"); - CHECK(json::serializer::escape_string("\x18") == "\\u0018"); - CHECK(json::serializer::escape_string("\x19") == "\\u0019"); - CHECK(json::serializer::escape_string("\x1a") == "\\u001a"); - CHECK(json::serializer::escape_string("\x1b") == "\\u001b"); - CHECK(json::serializer::escape_string("\x1c") == "\\u001c"); - CHECK(json::serializer::escape_string("\x1d") == "\\u001d"); - CHECK(json::serializer::escape_string("\x1e") == "\\u001e"); - CHECK(json::serializer::escape_string("\x1f") == "\\u001f"); + check_escaped("\"", "\\\""); + check_escaped("\\", "\\\\"); + check_escaped("\b", "\\b"); + check_escaped("\f", "\\f"); + check_escaped("\n", "\\n"); + check_escaped("\r", "\\r"); + check_escaped("\t", "\\t"); + + check_escaped("\x01", "\\u0001"); + check_escaped("\x02", "\\u0002"); + check_escaped("\x03", "\\u0003"); + check_escaped("\x04", "\\u0004"); + check_escaped("\x05", "\\u0005"); + check_escaped("\x06", "\\u0006"); + check_escaped("\x07", "\\u0007"); + check_escaped("\x08", "\\b"); + check_escaped("\x09", "\\t"); + check_escaped("\x0a", "\\n"); + check_escaped("\x0b", "\\u000b"); + check_escaped("\x0c", "\\f"); + check_escaped("\x0d", "\\r"); + check_escaped("\x0e", "\\u000e"); + check_escaped("\x0f", "\\u000f"); + check_escaped("\x10", "\\u0010"); + check_escaped("\x11", "\\u0011"); + check_escaped("\x12", "\\u0012"); + check_escaped("\x13", "\\u0013"); + check_escaped("\x14", "\\u0014"); + check_escaped("\x15", "\\u0015"); + check_escaped("\x16", "\\u0016"); + check_escaped("\x17", "\\u0017"); + check_escaped("\x18", "\\u0018"); + check_escaped("\x19", "\\u0019"); + check_escaped("\x1a", "\\u001a"); + check_escaped("\x1b", "\\u001b"); + check_escaped("\x1c", "\\u001c"); + check_escaped("\x1d", "\\u001d"); + check_escaped("\x1e", "\\u001e"); + check_escaped("\x1f", "\\u001f"); } } From 059f21aadae5da6f1bebbdd12e3cc1421b39b46d Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 28 Feb 2017 17:24:03 +0100 Subject: [PATCH 015/530] :lipstick: fixed a warning snprintf returns an int, but we later assign it a difference_type which is usually a long. --- src/json.hpp | 2 +- src/json.hpp.re2c | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 8d545a93..739f3540 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -6684,7 +6684,7 @@ class basic_json static constexpr auto d = std::numeric_limits::digits10; // the actual conversion - auto written_bytes = snprintf(m_buf.data(), m_buf.size(), "%.*g", d, x); + long written_bytes = snprintf(m_buf.data(), m_buf.size(), "%.*g", d, x); // negative value indicates an error assert(written_bytes > 0); diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index b14fb68d..a55792c5 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -6684,7 +6684,7 @@ class basic_json static constexpr auto d = std::numeric_limits::digits10; // the actual conversion - auto written_bytes = snprintf(m_buf.data(), m_buf.size(), "%.*g", d, x); + long written_bytes = snprintf(m_buf.data(), m_buf.size(), "%.*g", d, x); // negative value indicates an error assert(written_bytes > 0); From d69242c6ba40bb809b18d43eb4712b15ccb7a912 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 28 Feb 2017 19:20:50 +0100 Subject: [PATCH 016/530] :lipstick: cleanup MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit - Added comments for the serializer class. - Added test case for resizing of the indentation string. - Using std::none_of to check if “.0” needs to be added to floating-point number. --- src/json.hpp | 98 +++++++++++++++++++++++------------- src/json.hpp.re2c | 98 +++++++++++++++++++++++------------- test/src/unit-inspection.cpp | 12 +++++ 3 files changed, 138 insertions(+), 70 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 739f3540..507ec9f2 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -6203,6 +6203,9 @@ class basic_json class serializer { public: + /*! + @param[in] s output stream to serialize to + */ serializer(std::ostream& s) : o(s), loc(std::localeconv()), thousands_sep(!loc->thousands_sep ? '\0' : loc->thousands_sep[0]), @@ -6212,10 +6215,10 @@ class basic_json /*! @brief internal implementation of the serialization function - This function is called by the public member function dump and organizes - the serialization internally. The indentation level is propagated as - additional parameter. In case of arrays and objects, the function is - called recursively. Note that + This function is called by the public member function dump and + organizes the serialization internally. The indentation level is + propagated as additional parameter. In case of arrays and objects, the + function is called recursively. - strings and object keys are escaped using `escape_string()` - integer numbers are converted implicitly via `operator<<` @@ -6483,15 +6486,14 @@ class basic_json } /*! - @brief escape a string + @brief dump escaped string - Escape a string by replacing certain special characters by a sequence of - an escape character (backslash) and another character and other control - characters by a sequence of "\u" followed by a four-digit hex - representation. + Escape a string by replacing certain special characters by a sequence + of an escape character (backslash) and another character and other + control characters by a sequence of "\u" followed by a four-digit hex + representation. The escaped string is written to output stream @a o. @param[in] s the string to escape - @return the escaped string @complexity Linear in the length of string @a s. */ @@ -6629,7 +6631,18 @@ class basic_json o.write(result.c_str(), static_cast(result.size())); } - template + /*! + @brief dump an integer + + Dump a given integer to output stream @a o. Works internally with + @a number_buffer. + + @param[in] x integer number (signed or unsigned) to dump + @tparam NumberType either @a number_integer_t or @a number_unsigned_t + */ + template::value or + std::is_same::value, int> = 0> void dump_integer(NumberType x) { // special case for "0" @@ -6643,10 +6656,10 @@ class basic_json size_t i = 0; // spare 1 byte for '\0' - while (x != 0 and i < m_buf.size() - 1) + while (x != 0 and i < number_buffer.size() - 1) { const auto digit = std::labs(static_cast(x % 10)); - m_buf[i++] = static_cast('0' + digit); + number_buffer[i++] = static_cast('0' + digit); x /= 10; } @@ -6656,14 +6669,22 @@ class basic_json if (is_negative) { // make sure there is capacity for the '-' - assert(i < m_buf.size() - 2); - m_buf[i++] = '-'; + assert(i < number_buffer.size() - 2); + number_buffer[i++] = '-'; } - std::reverse(m_buf.begin(), m_buf.begin() + i); - o.write(m_buf.data(), static_cast(i)); + std::reverse(number_buffer.begin(), number_buffer.begin() + i); + o.write(number_buffer.data(), static_cast(i)); } + /*! + @brief dump a floating-point number + + Dump a given floating-point number to output stream @a o. Works + internally with @a number_buffer. + + @param[in] x floating-point number to dump + */ void dump_float(number_float_t x) { // special case for 0.0 and -0.0 @@ -6684,26 +6705,29 @@ class basic_json static constexpr auto d = std::numeric_limits::digits10; // the actual conversion - long written_bytes = snprintf(m_buf.data(), m_buf.size(), "%.*g", d, x); + long len = snprintf(number_buffer.data(), number_buffer.size(), + "%.*g", d, x); // negative value indicates an error - assert(written_bytes > 0); + assert(len > 0); // check if buffer was large enough - assert(static_cast(written_bytes) < m_buf.size()); + assert(static_cast(len) < number_buffer.size()); // erase thousands separator if (thousands_sep != '\0') { - const auto end = std::remove(m_buf.begin(), m_buf.begin() + written_bytes, thousands_sep); - std::fill(end, m_buf.end(), '\0'); - assert((end - m_buf.begin()) <= written_bytes); - written_bytes = (end - m_buf.begin()); + const auto end = std::remove(number_buffer.begin(), + number_buffer.begin() + len, + thousands_sep); + std::fill(end, number_buffer.end(), '\0'); + assert((end - number_buffer.begin()) <= len); + len = (end - number_buffer.begin()); } // convert decimal point to '.' if (decimal_point != '\0' and decimal_point != '.') { - for (auto& c : m_buf) + for (auto& c : number_buffer) { if (c == decimal_point) { @@ -6713,16 +6737,15 @@ class basic_json } } - // determine if need to append ".0" - bool value_is_int_like = true; - for (size_t i = 0; i < static_cast(written_bytes); ++i) - { - // check if we find non-int character - value_is_int_like = value_is_int_like and m_buf[i] != '.' and - m_buf[i] != 'e'; - } + o.write(number_buffer.data(), static_cast(len)); - o.write(m_buf.data(), static_cast(written_bytes)); + // determine if need to append ".0" + const bool value_is_int_like = std::none_of(number_buffer.begin(), + number_buffer.begin() + len + 1, + [](char c) + { + return c == '.' or c == 'e'; + }); if (value_is_int_like) { @@ -6731,15 +6754,20 @@ class basic_json } private: + /// the output of the serializer std::ostream& o; /// a (hopefully) large enough character buffer - std::array < char, 64 > m_buf{{}}; + std::array number_buffer{{}}; + /// the locale const std::lconv* loc = nullptr; + /// the locale's thousand separator character const char thousands_sep = '\0'; + /// the locale's decimal point character const char decimal_point = '\0'; + /// the indentation string string_t indent_string = string_t(512, ' '); }; diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index a55792c5..1dbbb3b5 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -6203,6 +6203,9 @@ class basic_json class serializer { public: + /*! + @param[in] s output stream to serialize to + */ serializer(std::ostream& s) : o(s), loc(std::localeconv()), thousands_sep(!loc->thousands_sep ? '\0' : loc->thousands_sep[0]), @@ -6212,10 +6215,10 @@ class basic_json /*! @brief internal implementation of the serialization function - This function is called by the public member function dump and organizes - the serialization internally. The indentation level is propagated as - additional parameter. In case of arrays and objects, the function is - called recursively. Note that + This function is called by the public member function dump and + organizes the serialization internally. The indentation level is + propagated as additional parameter. In case of arrays and objects, the + function is called recursively. - strings and object keys are escaped using `escape_string()` - integer numbers are converted implicitly via `operator<<` @@ -6483,15 +6486,14 @@ class basic_json } /*! - @brief escape a string + @brief dump escaped string - Escape a string by replacing certain special characters by a sequence of - an escape character (backslash) and another character and other control - characters by a sequence of "\u" followed by a four-digit hex - representation. + Escape a string by replacing certain special characters by a sequence + of an escape character (backslash) and another character and other + control characters by a sequence of "\u" followed by a four-digit hex + representation. The escaped string is written to output stream @a o. @param[in] s the string to escape - @return the escaped string @complexity Linear in the length of string @a s. */ @@ -6629,7 +6631,18 @@ class basic_json o.write(result.c_str(), static_cast(result.size())); } - template + /*! + @brief dump an integer + + Dump a given integer to output stream @a o. Works internally with + @a number_buffer. + + @param[in] x integer number (signed or unsigned) to dump + @tparam NumberType either @a number_integer_t or @a number_unsigned_t + */ + template::value or + std::is_same::value, int> = 0> void dump_integer(NumberType x) { // special case for "0" @@ -6643,10 +6656,10 @@ class basic_json size_t i = 0; // spare 1 byte for '\0' - while (x != 0 and i < m_buf.size() - 1) + while (x != 0 and i < number_buffer.size() - 1) { const auto digit = std::labs(static_cast(x % 10)); - m_buf[i++] = static_cast('0' + digit); + number_buffer[i++] = static_cast('0' + digit); x /= 10; } @@ -6656,14 +6669,22 @@ class basic_json if (is_negative) { // make sure there is capacity for the '-' - assert(i < m_buf.size() - 2); - m_buf[i++] = '-'; + assert(i < number_buffer.size() - 2); + number_buffer[i++] = '-'; } - std::reverse(m_buf.begin(), m_buf.begin() + i); - o.write(m_buf.data(), static_cast(i)); + std::reverse(number_buffer.begin(), number_buffer.begin() + i); + o.write(number_buffer.data(), static_cast(i)); } + /*! + @brief dump a floating-point number + + Dump a given floating-point number to output stream @a o. Works + internally with @a number_buffer. + + @param[in] x floating-point number to dump + */ void dump_float(number_float_t x) { // special case for 0.0 and -0.0 @@ -6684,26 +6705,29 @@ class basic_json static constexpr auto d = std::numeric_limits::digits10; // the actual conversion - long written_bytes = snprintf(m_buf.data(), m_buf.size(), "%.*g", d, x); + long len = snprintf(number_buffer.data(), number_buffer.size(), + "%.*g", d, x); // negative value indicates an error - assert(written_bytes > 0); + assert(len > 0); // check if buffer was large enough - assert(static_cast(written_bytes) < m_buf.size()); + assert(static_cast(len) < number_buffer.size()); // erase thousands separator if (thousands_sep != '\0') { - const auto end = std::remove(m_buf.begin(), m_buf.begin() + written_bytes, thousands_sep); - std::fill(end, m_buf.end(), '\0'); - assert((end - m_buf.begin()) <= written_bytes); - written_bytes = (end - m_buf.begin()); + const auto end = std::remove(number_buffer.begin(), + number_buffer.begin() + len, + thousands_sep); + std::fill(end, number_buffer.end(), '\0'); + assert((end - number_buffer.begin()) <= len); + len = (end - number_buffer.begin()); } // convert decimal point to '.' if (decimal_point != '\0' and decimal_point != '.') { - for (auto& c : m_buf) + for (auto& c : number_buffer) { if (c == decimal_point) { @@ -6713,16 +6737,15 @@ class basic_json } } - // determine if need to append ".0" - bool value_is_int_like = true; - for (size_t i = 0; i < static_cast(written_bytes); ++i) - { - // check if we find non-int character - value_is_int_like = value_is_int_like and m_buf[i] != '.' and - m_buf[i] != 'e'; - } + o.write(number_buffer.data(), static_cast(len)); - o.write(m_buf.data(), static_cast(written_bytes)); + // determine if need to append ".0" + const bool value_is_int_like = std::none_of(number_buffer.begin(), + number_buffer.begin() + len + 1, + [](char c) + { + return c == '.' or c == 'e'; + }); if (value_is_int_like) { @@ -6731,15 +6754,20 @@ class basic_json } private: + /// the output of the serializer std::ostream& o; /// a (hopefully) large enough character buffer - std::array < char, 64 > m_buf{{}}; + std::array number_buffer{{}}; + /// the locale const std::lconv* loc = nullptr; + /// the locale's thousand separator character const char thousands_sep = '\0'; + /// the locale's decimal point character const char decimal_point = '\0'; + /// the indentation string string_t indent_string = string_t(512, ' '); }; diff --git a/test/src/unit-inspection.cpp b/test/src/unit-inspection.cpp index aead1258..4900e425 100644 --- a/test/src/unit-inspection.cpp +++ b/test/src/unit-inspection.cpp @@ -213,6 +213,18 @@ TEST_CASE("object inspection") "{\n \"array\": [\n 1,\n 2,\n 3,\n 4\n ],\n \"boolean\": false,\n \"null\": null,\n \"number\": 42,\n \"object\": {},\n \"string\": \"Hello world\"\n}"); } + SECTION("indent=x") + { + CHECK(j.dump().size() == 94); + CHECK(j.dump(1).size() == 127); + CHECK(j.dump(2).size() == 142); + CHECK(j.dump(512).size() == 7792); + + // important test, because it yields a resize of the indent_string + // inside the dump() function + CHECK(j.dump(1024).size() == 15472); + } + SECTION("dump and floating-point numbers") { auto s = json(42.23).dump(); From f84ac523aa857fe4acc1398da923209fbaad4f92 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Wed, 1 Mar 2017 10:15:07 +0100 Subject: [PATCH 017/530] :memo: added a note to ordered maps The library does not preserve the insertion order of object keys. There are frequent requests to change the library in this aspect. The README and the contribution guidelines now contain links to containers that can be used to replace std::map to preserve the insertion order. --- .github/CONTRIBUTING.md | 2 +- README.md | 1 + 2 files changed, 2 insertions(+), 1 deletion(-) diff --git a/.github/CONTRIBUTING.md b/.github/CONTRIBUTING.md index 5110aea6..4a621c07 100644 --- a/.github/CONTRIBUTING.md +++ b/.github/CONTRIBUTING.md @@ -64,7 +64,7 @@ Please understand that I cannot accept pull requests changing only file `src/jso - Specifically, I am aware of compilation problems with **Microsoft Visual Studio** (there even is an [issue label](https://github.com/nlohmann/json/issues?utf8=✓&q=label%3A%22visual+studio%22+) for these kind of bugs). I understand that even in 2016, complete C++11 support isn't there yet. But please also understand that I do not want to drop features or uglify the code just to make Microsoft's sub-standard compiler happy. The past has shown that there are ways to express the functionality such that the code compiles with the most recent MSVC - unfortunately, this is not the main objective of the project. - Please refrain from proposing changes that would **break [JSON](http://json.org) conformance**. If you propose a conformant extension of JSON to be supported by the library, please motivate this extension. - We shall not extend the library to **support comments**. There is quite some [controversy](https://www.reddit.com/r/programming/comments/4v6chu/why_json_doesnt_support_comments_douglas_crockford/) around this topic, and there were quite some [issues](https://github.com/nlohmann/json/issues/376) on this. We believe that JSON is fine without comments. - - We do not preserve the **insertion order of object elements**. The [JSON standard](https://tools.ietf.org/html/rfc7159.html) defines objects as "an unordered collection of zero or more name/value pairs". To this end, this library does not preserve insertion order of name/value pairs. (In fact, keys will be traversed in alphabetical order as `std::map` with `std::less` is used by default.) Note this behavior conforms to the standard, and we shall not it to any other order. + - We do not preserve the **insertion order of object elements**. The [JSON standard](https://tools.ietf.org/html/rfc7159.html) defines objects as "an unordered collection of zero or more name/value pairs". To this end, this library does not preserve insertion order of name/value pairs. (In fact, keys will be traversed in alphabetical order as `std::map` with `std::less` is used by default.) Note this behavior conforms to the standard, and we shall not it to any other order. If you do want to preserve the insertion order, you can specialize the object type with containers like [`tsl::ordered_map`](https://github.com/Tessil/ordered-map) or [`nlohmann::fifo_map`](https://github.com/nlohmann/fifo_map). - Please do not open pull requests that address **multiple issues**. diff --git a/README.md b/README.md index 8dea44aa..697a5743 100644 --- a/README.md +++ b/README.md @@ -876,6 +876,7 @@ The library is currently used in Apple macOS Sierra and iOS 10. I am not sure wh - The strings stored in the library are UTF-8 encoded. When using the default string type (`std::string`), note that its length/size functions return the number of stored bytes rather than the number of characters or glyphs. - The code can be compiled without C++ **runtime type identification** features; that is, you can use the `-fno-rtti` compiler flag. - **Exceptions** are used widly within the library. They can, however, be switched off with either using the compiler flag `-fno-exceptions` or by defining the symbol `JSON_NOEXCEPTION`. In this case, exceptions are replaced by an `abort()` call. +- By default, the library does not preserve the **insertion order of object elements**. This is standards-compliant, as the [JSON standard](https://tools.ietf.org/html/rfc7159.html) defines objects as "an unordered collection of zero or more name/value pairs". If you do want to preserve the insertion order, you can specialize the object type with containers like [`tsl::ordered_map`](https://github.com/Tessil/ordered-map) or [`nlohmann::fifo_map`](https://github.com/nlohmann/fifo_map). ## Execute unit tests From 6b3912d936039b56c24f49e0f81e8bac6627e11f Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Wed, 1 Mar 2017 17:26:32 +0100 Subject: [PATCH 018/530] :memo: added note to 3.0.0 wiki page #474 I created a wiki page https://github.com/nlohmann/json/wiki/Road-toward-3.0.0 to describe the transition toward version 3.0.0. On this page, all API-breaking changes shall be documented. --- README.md | 2 ++ 1 file changed, 2 insertions(+) diff --git a/README.md b/README.md index 697a5743..1ada712c 100644 --- a/README.md +++ b/README.md @@ -65,6 +65,8 @@ to the files you want to use JSON objects. That's it. Do not forget to set the n :beer: If you are using OS X and [Homebrew](http://brew.sh), just type `brew tap nlohmann/json` and `brew install nlohmann_json` and you're set. If you want the bleeding edge rather than the latest release, use `brew install nlohmann_json --HEAD`. +:warning: [Version 3.0.0](https://github.com/nlohmann/json/wiki/Road-toward-3.0.0) is currently under development. Branch `develop` is used for the ongoing work and is probably **unstable**. Please use the `master` branch for the last stable version 2.1.1. + ## Examples From 7b8fd864e220eeb2b7eac06f20a921ea0d93983f Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Wed, 1 Mar 2017 17:49:03 +0100 Subject: [PATCH 019/530] :fire: removed deprecated constructor #480 The constructor basic_json(std::istream&, const parser_callback_t) has been deprecated since version 2.0.0. This commit removes it together with its code example, deprecation macro, and test cases. The code now also compiles with -W-deprecated-declarations. --- Makefile | 2 - doc/examples/basic_json__istream.cpp | 57 ------------------------- doc/examples/basic_json__istream.link | 1 - doc/examples/basic_json__istream.output | 34 --------------- src/json.hpp | 44 ------------------- src/json.hpp.re2c | 44 ------------------- test/src/unit-constructor1.cpp | 36 ---------------- 7 files changed, 218 deletions(-) delete mode 100644 doc/examples/basic_json__istream.cpp delete mode 100644 doc/examples/basic_json__istream.link delete mode 100644 doc/examples/basic_json__istream.output diff --git a/Makefile b/Makefile index ac75bcaf..1b7cf8c1 100644 --- a/Makefile +++ b/Makefile @@ -49,7 +49,6 @@ doctest: # -Wno-documentation-unknown-command: code uses user-defined commands like @complexity # -Wno-exit-time-destructors: warning in Catch code # -Wno-keyword-macro: unit-tests use "#define private public" -# -Wno-deprecated-declarations: some functions are deprecated until 3.0.0 # -Wno-range-loop-analysis: iterator_wrapper tests tests "for(const auto i...)" pedantic: $(MAKE) json_unit CXXFLAGS="\ @@ -59,7 +58,6 @@ pedantic: -Wno-documentation-unknown-command \ -Wno-exit-time-destructors \ -Wno-keyword-macro \ - -Wno-deprecated-declarations \ -Wno-range-loop-analysis" diff --git a/doc/examples/basic_json__istream.cpp b/doc/examples/basic_json__istream.cpp deleted file mode 100644 index 32885b22..00000000 --- a/doc/examples/basic_json__istream.cpp +++ /dev/null @@ -1,57 +0,0 @@ -#include - -using json = nlohmann::json; - -int main() -{ - // a JSON text - auto text = R"( - { - "Image": { - "Width": 800, - "Height": 600, - "Title": "View from 15th Floor", - "Thumbnail": { - "Url": "http://www.example.com/image/481989943", - "Height": 125, - "Width": 100 - }, - "Animated" : false, - "IDs": [116, 943, 234, 38793] - } - } - )"; - - // fill a stream with JSON text - std::stringstream ss; - ss << text; - - // create JSON from stream - json j_complete(ss); // deprecated! - // shall be replaced by: json j_complete = json::parse(ss); - std::cout << std::setw(4) << j_complete << "\n\n"; - - - // define parser callback - json::parser_callback_t cb = [](int depth, json::parse_event_t event, json & parsed) - { - // skip object elements with key "Thumbnail" - if (event == json::parse_event_t::key and parsed == json("Thumbnail")) - { - return false; - } - else - { - return true; - } - }; - - // fill a stream with JSON text - ss.clear(); - ss << text; - - // create JSON from stream (with callback) - json j_filtered(ss, cb); - // shall be replaced by: json j_filtered = json::parse(ss, cb); - std::cout << std::setw(4) << j_filtered << '\n'; -} \ No newline at end of file diff --git a/doc/examples/basic_json__istream.link b/doc/examples/basic_json__istream.link deleted file mode 100644 index eb165e2f..00000000 --- a/doc/examples/basic_json__istream.link +++ /dev/null @@ -1 +0,0 @@ -online \ No newline at end of file diff --git a/doc/examples/basic_json__istream.output b/doc/examples/basic_json__istream.output deleted file mode 100644 index 279a7ff7..00000000 --- a/doc/examples/basic_json__istream.output +++ /dev/null @@ -1,34 +0,0 @@ -{ - "Image": { - "Animated": false, - "Height": 600, - "IDs": [ - 116, - 943, - 234, - 38793 - ], - "Thumbnail": { - "Height": 125, - "Url": "http://www.example.com/image/481989943", - "Width": 100 - }, - "Title": "View from 15th Floor", - "Width": 800 - } -} - -{ - "Image": { - "Animated": false, - "Height": 600, - "IDs": [ - 116, - 943, - 234, - 38793 - ], - "Title": "View from 15th Floor", - "Width": 800 - } -} diff --git a/src/json.hpp b/src/json.hpp index 507ec9f2..ff0ce5ad 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -81,15 +81,6 @@ SOFTWARE. #pragma GCC diagnostic ignored "-Wdocumentation" #endif -// allow for portable deprecation warnings -#if defined(__clang__) || defined(__GNUC__) || defined(__GNUG__) - #define JSON_DEPRECATED __attribute__((deprecated)) -#elif defined(_MSC_VER) - #define JSON_DEPRECATED __declspec(deprecated) -#else - #define JSON_DEPRECATED -#endif - // allow to disable exceptions #if not defined(JSON_NOEXCEPTION) || defined(__EXCEPTIONS) #define JSON_THROW(exception) throw exception @@ -2362,40 +2353,6 @@ class basic_json assert_invariant(); } - /*! - @brief construct a JSON value given an input stream - - @param[in,out] i stream to read a serialized JSON value from - @param[in] cb a parser callback function of type @ref parser_callback_t - which is used to control the deserialization by filtering unwanted values - (optional) - - @complexity Linear in the length of the input. The parser is a predictive - LL(1) parser. The complexity can be higher if the parser callback function - @a cb has a super-linear complexity. - - @note A UTF-8 byte order mark is silently ignored. - - @deprecated This constructor is deprecated and will be removed in version - 3.0.0 to unify the interface of the library. Deserialization will be - done by stream operators or by calling one of the `parse` functions, - e.g. @ref parse(std::istream&, const parser_callback_t). That is, calls - like `json j(i);` for an input stream @a i need to be replaced by - `json j = json::parse(i);`. See the example below. - - @liveexample{The example below demonstrates constructing a JSON value from - a `std::stringstream` with and without callback - function.,basic_json__istream} - - @since version 2.0.0, deprecated in version 2.0.3, to be removed in - version 3.0.0 - */ - JSON_DEPRECATED - explicit basic_json(std::istream& i, const parser_callback_t cb = nullptr) - { - *this = parser(i, cb).parse(); - assert_invariant(); - } /////////////////////////////////////// // other constructors and destructor // @@ -13113,7 +13070,6 @@ inline nlohmann::json::json_pointer operator "" _json_pointer(const char* s, std // clean up #undef JSON_CATCH -#undef JSON_DEPRECATED #undef JSON_THROW #undef JSON_TRY diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 1dbbb3b5..fc7cf965 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -81,15 +81,6 @@ SOFTWARE. #pragma GCC diagnostic ignored "-Wdocumentation" #endif -// allow for portable deprecation warnings -#if defined(__clang__) || defined(__GNUC__) || defined(__GNUG__) - #define JSON_DEPRECATED __attribute__((deprecated)) -#elif defined(_MSC_VER) - #define JSON_DEPRECATED __declspec(deprecated) -#else - #define JSON_DEPRECATED -#endif - // allow to disable exceptions #if not defined(JSON_NOEXCEPTION) || defined(__EXCEPTIONS) #define JSON_THROW(exception) throw exception @@ -2362,40 +2353,6 @@ class basic_json assert_invariant(); } - /*! - @brief construct a JSON value given an input stream - - @param[in,out] i stream to read a serialized JSON value from - @param[in] cb a parser callback function of type @ref parser_callback_t - which is used to control the deserialization by filtering unwanted values - (optional) - - @complexity Linear in the length of the input. The parser is a predictive - LL(1) parser. The complexity can be higher if the parser callback function - @a cb has a super-linear complexity. - - @note A UTF-8 byte order mark is silently ignored. - - @deprecated This constructor is deprecated and will be removed in version - 3.0.0 to unify the interface of the library. Deserialization will be - done by stream operators or by calling one of the `parse` functions, - e.g. @ref parse(std::istream&, const parser_callback_t). That is, calls - like `json j(i);` for an input stream @a i need to be replaced by - `json j = json::parse(i);`. See the example below. - - @liveexample{The example below demonstrates constructing a JSON value from - a `std::stringstream` with and without callback - function.,basic_json__istream} - - @since version 2.0.0, deprecated in version 2.0.3, to be removed in - version 3.0.0 - */ - JSON_DEPRECATED - explicit basic_json(std::istream& i, const parser_callback_t cb = nullptr) - { - *this = parser(i, cb).parse(); - assert_invariant(); - } /////////////////////////////////////// // other constructors and destructor // @@ -12147,7 +12104,6 @@ inline nlohmann::json::json_pointer operator "" _json_pointer(const char* s, std // clean up #undef JSON_CATCH -#undef JSON_DEPRECATED #undef JSON_THROW #undef JSON_TRY diff --git a/test/src/unit-constructor1.cpp b/test/src/unit-constructor1.cpp index b3354b58..9363f0ba 100644 --- a/test/src/unit-constructor1.cpp +++ b/test/src/unit-constructor1.cpp @@ -1281,40 +1281,4 @@ TEST_CASE("constructors") } } } - - SECTION("create a JSON value from an input stream") - { - SECTION("std::stringstream") - { - std::stringstream ss; - ss << "[\"foo\",1,2,3,false,{\"one\":1}]"; - json j(ss); - CHECK(j == json({"foo", 1, 2, 3, false, {{"one", 1}}})); - } - - SECTION("with callback function") - { - std::stringstream ss; - ss << "[\"foo\",1,2,3,false,{\"one\":1}]"; - json j(ss, [](int, json::parse_event_t, const json & val) - { - // filter all number(2) elements - if (val == json(2)) - { - return false; - } - else - { - return true; - } - }); - CHECK(j == json({"foo", 1, 3, false, {{"one", 1}}})); - } - - SECTION("std::ifstream") - { - std::ifstream f("test/data/json_tests/pass1.json"); - json j(f); - } - } } From c085e3bac28076bf0dc047b9981d06555c1e709e Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Wed, 1 Mar 2017 21:28:44 +0100 Subject: [PATCH 020/530] :hammer: started with user-defined exceptions #301 #244 Added class hierarchy for user-defined exceptions (#244). Integrated parse exceptions 101-103. Parse exceptions include the byte count of the last read character to locate the position of the error (#301). --- doc/Makefile | 1 + src/json.hpp | 209 +++++++++++++++- src/json.hpp.re2c | 210 +++++++++++++++- test/src/unit-class_lexer.cpp | 8 +- test/src/unit-class_parser.cpp | 392 ++++++++++++++++-------------- test/src/unit-deserialization.cpp | 58 +++-- test/src/unit-regression.cpp | 7 +- test/src/unit-testsuites.cpp | 6 +- test/src/unit-unicode.cpp | 11 +- 9 files changed, 650 insertions(+), 252 deletions(-) diff --git a/doc/Makefile b/doc/Makefile index 37a0e19f..cd23a9f7 100644 --- a/doc/Makefile +++ b/doc/Makefile @@ -58,6 +58,7 @@ doxygen: create_output create_links $(SED) -i 's@< ObjectType, ArrayType, StringType, BooleanType, NumberIntegerType, NumberUnsignedType, NumberFloatType, AllocatorType, JSONSerializer >@@g' html/*.html $(SED) -i 's@< ObjectType, ArrayType, StringType, BooleanType, NumberIntegerType, NumberUnsignedType, NumberFloatType, AllocatorType, JSONSerializer >@@g' html/*.html $(SED) -i 's@< ObjectType, ArrayType, StringType, BooleanType, NumberIntegerType, NumberUnsignedType, NumberFloatType, AllocatorType JSONSerializer >@@g' html/*.html + $(SED) -i 's@template<template< typename U, typename V, typename... Args > class ObjectType = std::map, template< typename U, typename... Args > class ArrayType = std::vector, class StringType = std::string, class BooleanType = bool, class NumberIntegerType = std::int64_t, class NumberUnsignedType = std::uint64_t, class NumberFloatType = double, template< typename U > class AllocatorType = std::allocator, template< typename T, typename SFINAE=void > class JSONSerializer = adl_serializer>@@g' html/*.html upload: clean doxygen check_output cd html ; ../scripts/git-update-ghpages nlohmann/json diff --git a/src/json.hpp b/src/json.hpp index ff0ce5ad..d4e87e5e 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -1088,6 +1088,178 @@ class basic_json /// @} + public: + //////////////// + // exceptions // + //////////////// + + /// @name exceptions + /// Classes to implement user-defined exceptions. + /// @{ + + /*! + @brief general exception of the @ref basic_json class + + Extension of std::exception objects with a member @a id for exception ids. + + name / id | example massage | description + ------------------------------ | --------------- | ------------------------- + json.exception.[parse_error](@ref parse_error).101 | `"parse error at 2: unexpected end of input; expected string literal"` | This error indicates a syntax error while deserializing a JSON text. The error message describes that an unexpected token (character) was encountered, and the member @ref parse_error::byte indicates the error position. + json.exception.[parse_error](@ref parse_error).102 | `"parse error at 14: missing or wrong low surrogate"` | JSON uses the `\uxxxx` format to describe Unicode characters. Code points above above 0xFFFF are split into two `\uxxxx` entries ("surrogate pairs"). This error indicates that the surrogate pair is incomplete or contains an invalid code point. + json.exception.[parse_error](@ref parse_error).103 | `"parse error: code points above 0x10FFFF are invalid"` | Unicode supports code points up to 0x10FFFF. Code points above 0x10FFFF are invalid. + json.exception.[parse_error](@ref parse_error).104 | "parse error: JSON patch must be an array of objects" | [RFC 6902](https://tools.ietf.org/html/rfc6902) requires a JSON Patch document to be a JSON documentthat represents an array of objects. + json.exception.[parse_error](@ref parse_error).105 | "parse error: operation must have string member 'op'" | An operation of a JSON Patch document must contain Operation objects MUST have exactly one "op" member, whose value indicates the operation to perform. Its value MUST be one of "add", "remove", "replace", "move", "copy", or "test"; other values are errors. + json.exception.[parse_error](@ref parse_error).106 | "parse error: array index must not begin with '0'" | An array index in a JSON Pointer ([RFC 6901](https://tools.ietf.org/html/rfc6901)) may be `0` or any number wihtout a leading `0`. + json.exception.[parse_error](@ref parse_error).107 | "parse error: JSON pointer must be empty or begin with '/'" | A JSON Pointer must be a Unicode string containing a sequence of zero or more reference tokens, each prefixed by a `/` character. + json.exception.[parse_error](@ref parse_error).108 | "parse error: escape character '~' must be followed with '0' or '1'" | In a JSON Pointer, only `~0` and `~1` are valid escape sequences. + json.exception.[parse_error](@ref parse_error).109 | "parse error: array index 'one' is not a number" | A JSON Pointer array index must be a number. + json.exception.[invalid_iterator](@ref invalid_iterator).201 | "iterators are not compatible" | The iterators passed to constructor @ref basic_json(InputIT first, InputIT last) are not compatible, meaning they do not belong to the same container. Therefore, the range (@a first, @a last) is invalid. + json.exception.[invalid_iterator](@ref invalid_iterator).202 | "iterator does not fit current value" | In an erase or insert function, the passed iterator @a pos does not belong to the JSON value for which the function was called. It hence does not define a valid position for the deletion/insertion. + json.exception.[invalid_iterator](@ref invalid_iterator).203 | "iterators do not fit current value" | Either iterator passed to function @ref erase(IteratorType first, IteratorType last) does not belong to the JSON value from which values shall be erased. It hence does not define a valid range to delete values from. + json.exception.[invalid_iterator](@ref invalid_iterator).204 | "iterators out of range" | When an iterator range for a primitive type (number, boolean, or string) is passed to a constructor or an erase function, this range has to be exactly (@ref begin(), @ref end()), because this is the only way the single stored value is expressed. All other ranges are invalid. + json.exception.[invalid_iterator](@ref invalid_iterator).205 | "iterator out of range" | When an iterator for a primitive type (number, boolean, or string) is passed to an erase function, the iterator has to be the @ref begin() iterator, because it is the only way to address the stored value. All other iterators are invalid. + json.exception.[invalid_iterator](@ref invalid_iterator).206 | "cannot construct with iterators from null" | The iterators passed to constructor @ref basic_json(InputIT first, InputIT last) belong to a JSON null value and hence to not define a valid range. + json.exception.[invalid_iterator](@ref invalid_iterator).207 | "cannot use key() for non-object iterators" | The key() member function can only be used on iterators belonging to a JSON object, because other types do not have a concept of a key. + json.exception.[invalid_iterator](@ref invalid_iterator).208 | "cannot use operator[] for object iterators" | The operator[] to specify a concrete offset cannot be used on iterators belonging to a JSON object, because JSON objects are unordered. + json.exception.[invalid_iterator](@ref invalid_iterator).209 | "cannot use offsets with object iterators" | The offset operators (+, -, +=, -=) cannot be used on iterators belonging to a JSON object, because JSON objects are unordered. + json.exception.[invalid_iterator](@ref invalid_iterator).210 | "iterators do not fit" | The iterator range passed to the insert function are not compatible, meaning they do not belong to the same container. Therefore, the range (@a first, @a last) is invalid. + json.exception.[invalid_iterator](@ref invalid_iterator).211 | "passed iterators may not belong to container" | The iterator range passed to the insert function must not be a subrange of the container to insert to. + json.exception.[invalid_iterator](@ref invalid_iterator).212 | "cannot compare iterators of different containers" | When two iterators are compared, they must belong to the same container. + json.exception.[invalid_iterator](@ref invalid_iterator).213 | "cannot compare order of object iterators" | The order of object iterators cannot be compated, because JSON objects are unordered. + json.exception.[invalid_iterator](@ref invalid_iterator).214 | "cannot get value" | Cannot get value for iterator: Either the iterator belongs to a null value or it is an iterator to a primitive type (number, boolean, or string), but the iterator is different to @ref begin(). + json.exception.[type_error](@ref type_error).301 | "cannot create object from initializer list" | To create an object from an initializer list, the initializer list must consist only of a list of pairs whose first element is a string. When this constraint is violated, an array is created instead. + json.exception.[type_error](@ref type_error).302 | "type must be object, but is array" | During implicit or explicit value conversion, the JSON type must be compatible to the target type. For instance, a JSON string can only be converted into string types, but not into numbers or boolean types. + json.exception.[type_error](@ref type_error).303 | "incompatible ReferenceType for get_ref, actual type is object" | To retrieve a reference to a value stored in a @ref basic_json object with @ref get_ref, the type of the reference must match the value type. For instance, for a JSON array, the @a ReferenceType must be @ref array_t&. + json.exception.[type_error](@ref type_error).304 | "cannot use at() with string" | The @ref at() member functions can only be executed for certain JSON types. + json.exception.[type_error](@ref type_error).305 | "cannot use operator[] with string" | The @ref operator[] member functions can only be executed for certain JSON types. + json.exception.[type_error](@ref type_error).306 | "cannot use value() with string" | The @ref value() member functions can only be executed for certain JSON types. + json.exception.[type_error](@ref type_error).307 | "cannot use erase() with string" | The @ref erase() member functions can only be executed for certain JSON types. + json.exception.[type_error](@ref type_error).308 | "cannot use push_back() with string" | The @ref push_back() and @ref operator+= member functions can only be executed for certain JSON types. + json.exception.[type_error](@ref type_error).309 | "cannot use insert() with" | The @ref insert() member functions can only be executed for certain JSON types. + json.exception.[type_error](@ref type_error).310 | "cannot use swap() with number" | The @ref swap() member functions can only be executed for certain JSON types. + json.exception.[type_error](@ref type_error).313 | "invalid value to unflatten" | The @ref unflatten function converts an object whose keys are JSON Pointers back into an arbitrary nested JSON value. The JSON Pointers must not overlap, because then the resulting value would not be well defined. + json.exception.[type_error](@ref type_error).314 | "only objects can be unflattened" | The @ref unflatten function only works for an object whose keys are JSON Pointers. + json.exception.[type_error](@ref type_error).315 | "values in object must be primitive" | The @ref unflatten function only works for an object whose keys are JSON Pointers and whose values are primitive. + json.exception.[out_of_range](@ref out_of_range).401 | "array index 3 is out of range" | The provided array index @a i is larger than @a size-1. + json.exception.[out_of_range](@ref out_of_range).402 | "array index '-' (3) is out of range" | The special array index `-` in a JSON Pointer never describes a valid element of the array, but the index past the end. + json.exception.[out_of_range](@ref out_of_range).403 | "key 'foo' not found" | The provided key was not found in the JSON object. + json.exception.[out_of_range](@ref out_of_range).404 | "unresolved reference token 'foo'" | A reference token in a JSON Pointer could not be resolved. + json.exception.[out_of_range](@ref out_of_range).405 | "JSON pointer has no parent" | The JSON Patch operations 'remove' and 'add' can not be applied to the root element of the JSON value. + json.exception.other.500 | "unsuccessful" | A JSON Patch operation 'test' failed. + + @since version 3.0.0 + */ + class exception : public std::exception + { + public: + /// create exception with id an explanatory string + exception(int id_, const std::string& ename, const std::string& what_arg_) + : id(id_), + what_arg("[json.exception." + ename + "." + std::to_string(id_) + "] " + what_arg_) + {} + + /// returns the explanatory string + virtual const char* what() const noexcept + { + return what_arg.c_str(); + } + + /// the id of the exception + const int id; + + private: + /// the explanatory string + const std::string what_arg; + }; + + /*! + @brief exception indicating a parse error + + This excpetion is thrown by the library when a parse error occurs. Parse + errors can occur during the deserialization of JSON text as well as when + using JSON Patch. + + Exceptions have ids 1xx. + + @since version 3.0.0 + */ + class parse_error : public exception + { + public: + /*! + @brief create a parse error exception + @param[in] id_ the id of the exception + @param[in] byte_ the byte index where the error occured (or 0 if + the position cannot be determined) + @param[in] what_arg_ the explanatory string + */ + parse_error(int id_, size_t byte_, const std::string& what_arg_) + : exception(id_, "parse_error", "parse error" + + (byte_ != 0 ? (" at " + std::to_string(byte_)) : "") + + ": " + what_arg_), + byte(byte_) + {} + + /*! + @brief byte index of the parse error + + The byte index of the last read character in the input file. + + @note For an input with n bytes, 1 is the index of the first character + and n+1 is the index of the terminating null byte or the end of + file. + */ + const size_t byte; + }; + + /*! + @brief exception indicating errors with iterators + + Exceptions have ids 2xx. + + @since version 3.0.0 + */ + class invalid_iterator : public exception + { + public: + invalid_iterator(int id_, const std::string& what_arg_) + : exception(id_, "invalid_iterator", what_arg_) + {} + }; + + /*! + @brief exception indicating executing a member function with a wrong type + + Exceptions have ids 3xx. + + @since version 3.0.0 + */ + class type_error : public exception + { + public: + type_error(int id_, const std::string& what_arg_) + : exception(id_, "type_error", what_arg_) + {} + }; + + /*! + @brief exception indicating access out of the defined range + + Exceptions have ids 4xx. + + @since version 3.0.0 + */ + class out_of_range : public exception + { + public: + out_of_range(int id_, const std::string& what_arg_) + : exception(id_, "out_of_range", what_arg_) + {} + }; + + /// @} + + /*! @brief returns the allocator associated with the container */ @@ -6801,6 +6973,9 @@ class basic_json @return result of the deserialization + @throw parse_error.101 if a parse error occurs; example: `""unexpected end + of input; expected string literal""` + @complexity Linear in the length of the input. The parser is a predictive LL(1) parser. The complexity can be higher if the parser callback function @a cb has a super-linear complexity. @@ -9707,17 +9882,17 @@ class basic_json @return string representation of the code point; the length of the result string is between 1 and 4 characters. - @throw std::out_of_range if code point is > 0x10ffff; example: `"code - points above 0x10FFFF are invalid"` - @throw std::invalid_argument if the low surrogate is invalid; example: + @throw parse_error.102 if the low surrogate is invalid; example: `""missing or wrong low surrogate""` + @throw parse_error.103 if code point is > 0x10ffff; example: `"code + points above 0x10FFFF are invalid"` @complexity Constant. @see */ - static string_t to_unicode(const std::size_t codepoint1, - const std::size_t codepoint2 = 0) + string_t to_unicode(const std::size_t codepoint1, + const std::size_t codepoint2 = 0) const { // calculate the code point from the given code points std::size_t codepoint = codepoint1; @@ -9740,7 +9915,7 @@ class basic_json } else { - JSON_THROW(std::invalid_argument("missing or wrong low surrogate")); + JSON_THROW(parse_error(102, get_position(), "missing or wrong low surrogate")); } } @@ -9774,7 +9949,7 @@ class basic_json } else { - JSON_THROW(std::out_of_range("code points above 0x10FFFF are invalid")); + JSON_THROW(parse_error(103, get_position(), "code points above 0x10FFFF are invalid")); } return result; @@ -10034,6 +10209,7 @@ basic_json_parser_6: goto basic_json_parser_6; } { + position += static_cast((m_cursor - m_start)); continue; } basic_json_parser_9: @@ -10889,6 +11065,7 @@ basic_json_parser_74: } + position += static_cast((m_cursor - m_start)); return last_token_type; } @@ -11044,7 +11221,8 @@ basic_json_parser_74: @return string value of current token without opening and closing quotes - @throw std::out_of_range if to_unicode fails + @throw parse_error.102 if to_unicode fails + @throw parse_error.103 if to_unicode fails */ string_t get_string() const { @@ -11130,7 +11308,7 @@ basic_json_parser_74: // make sure there is a subsequent unicode if ((i + 6 >= m_limit) or * (i + 5) != '\\' or * (i + 6) != 'u') { - JSON_THROW(std::invalid_argument("missing low surrogate")); + JSON_THROW(parse_error(102, get_position(), "missing low surrogate")); } // get code yyyy from uxxxx\uyyyy @@ -11143,7 +11321,7 @@ basic_json_parser_74: else if (codepoint >= 0xDC00 and codepoint <= 0xDFFF) { // we found a lone low surrogate - JSON_THROW(std::invalid_argument("missing high surrogate")); + JSON_THROW(parse_error(102, get_position(), "missing high surrogate")); } else { @@ -11398,6 +11576,11 @@ basic_json_parser_74: return false; } + constexpr size_t get_position() const + { + return position; + } + private: /// optional input stream std::istream* m_stream = nullptr; @@ -11417,6 +11600,8 @@ basic_json_parser_74: const lexer_char_t* m_limit = nullptr; /// the last token type token_type last_token_type = token_type::end_of_input; + /// current position in the input (read bytes) + size_t position = 0; }; /*! @@ -11678,7 +11863,7 @@ basic_json_parser_74: "'") : lexer::token_type_name(last_token)); error_msg += "; expected " + lexer::token_type_name(t); - JSON_THROW(std::invalid_argument(error_msg)); + JSON_THROW(parse_error(101, m_lexer.get_position(), error_msg)); } } @@ -11690,7 +11875,7 @@ basic_json_parser_74: error_msg += (last_token == lexer::token_type::parse_error ? ("'" + m_lexer.get_token_string() + "'") : lexer::token_type_name(last_token)); - JSON_THROW(std::invalid_argument(error_msg)); + JSON_THROW(parse_error(101, m_lexer.get_position(), error_msg)); } } diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index fc7cf965..1afc025a 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -1088,6 +1088,178 @@ class basic_json /// @} + public: + //////////////// + // exceptions // + //////////////// + + /// @name exceptions + /// Classes to implement user-defined exceptions. + /// @{ + + /*! + @brief general exception of the @ref basic_json class + + Extension of std::exception objects with a member @a id for exception ids. + + name / id | example massage | description + ------------------------------ | --------------- | ------------------------- + json.exception.[parse_error](@ref parse_error).101 | `"parse error at 2: unexpected end of input; expected string literal"` | This error indicates a syntax error while deserializing a JSON text. The error message describes that an unexpected token (character) was encountered, and the member @ref parse_error::byte indicates the error position. + json.exception.[parse_error](@ref parse_error).102 | `"parse error at 14: missing or wrong low surrogate"` | JSON uses the `\uxxxx` format to describe Unicode characters. Code points above above 0xFFFF are split into two `\uxxxx` entries ("surrogate pairs"). This error indicates that the surrogate pair is incomplete or contains an invalid code point. + json.exception.[parse_error](@ref parse_error).103 | `"parse error: code points above 0x10FFFF are invalid"` | Unicode supports code points up to 0x10FFFF. Code points above 0x10FFFF are invalid. + json.exception.[parse_error](@ref parse_error).104 | "parse error: JSON patch must be an array of objects" | [RFC 6902](https://tools.ietf.org/html/rfc6902) requires a JSON Patch document to be a JSON documentthat represents an array of objects. + json.exception.[parse_error](@ref parse_error).105 | "parse error: operation must have string member 'op'" | An operation of a JSON Patch document must contain Operation objects MUST have exactly one "op" member, whose value indicates the operation to perform. Its value MUST be one of "add", "remove", "replace", "move", "copy", or "test"; other values are errors. + json.exception.[parse_error](@ref parse_error).106 | "parse error: array index must not begin with '0'" | An array index in a JSON Pointer ([RFC 6901](https://tools.ietf.org/html/rfc6901)) may be `0` or any number wihtout a leading `0`. + json.exception.[parse_error](@ref parse_error).107 | "parse error: JSON pointer must be empty or begin with '/'" | A JSON Pointer must be a Unicode string containing a sequence of zero or more reference tokens, each prefixed by a `/` character. + json.exception.[parse_error](@ref parse_error).108 | "parse error: escape character '~' must be followed with '0' or '1'" | In a JSON Pointer, only `~0` and `~1` are valid escape sequences. + json.exception.[parse_error](@ref parse_error).109 | "parse error: array index 'one' is not a number" | A JSON Pointer array index must be a number. + json.exception.[invalid_iterator](@ref invalid_iterator).201 | "iterators are not compatible" | The iterators passed to constructor @ref basic_json(InputIT first, InputIT last) are not compatible, meaning they do not belong to the same container. Therefore, the range (@a first, @a last) is invalid. + json.exception.[invalid_iterator](@ref invalid_iterator).202 | "iterator does not fit current value" | In an erase or insert function, the passed iterator @a pos does not belong to the JSON value for which the function was called. It hence does not define a valid position for the deletion/insertion. + json.exception.[invalid_iterator](@ref invalid_iterator).203 | "iterators do not fit current value" | Either iterator passed to function @ref erase(IteratorType first, IteratorType last) does not belong to the JSON value from which values shall be erased. It hence does not define a valid range to delete values from. + json.exception.[invalid_iterator](@ref invalid_iterator).204 | "iterators out of range" | When an iterator range for a primitive type (number, boolean, or string) is passed to a constructor or an erase function, this range has to be exactly (@ref begin(), @ref end()), because this is the only way the single stored value is expressed. All other ranges are invalid. + json.exception.[invalid_iterator](@ref invalid_iterator).205 | "iterator out of range" | When an iterator for a primitive type (number, boolean, or string) is passed to an erase function, the iterator has to be the @ref begin() iterator, because it is the only way to address the stored value. All other iterators are invalid. + json.exception.[invalid_iterator](@ref invalid_iterator).206 | "cannot construct with iterators from null" | The iterators passed to constructor @ref basic_json(InputIT first, InputIT last) belong to a JSON null value and hence to not define a valid range. + json.exception.[invalid_iterator](@ref invalid_iterator).207 | "cannot use key() for non-object iterators" | The key() member function can only be used on iterators belonging to a JSON object, because other types do not have a concept of a key. + json.exception.[invalid_iterator](@ref invalid_iterator).208 | "cannot use operator[] for object iterators" | The operator[] to specify a concrete offset cannot be used on iterators belonging to a JSON object, because JSON objects are unordered. + json.exception.[invalid_iterator](@ref invalid_iterator).209 | "cannot use offsets with object iterators" | The offset operators (+, -, +=, -=) cannot be used on iterators belonging to a JSON object, because JSON objects are unordered. + json.exception.[invalid_iterator](@ref invalid_iterator).210 | "iterators do not fit" | The iterator range passed to the insert function are not compatible, meaning they do not belong to the same container. Therefore, the range (@a first, @a last) is invalid. + json.exception.[invalid_iterator](@ref invalid_iterator).211 | "passed iterators may not belong to container" | The iterator range passed to the insert function must not be a subrange of the container to insert to. + json.exception.[invalid_iterator](@ref invalid_iterator).212 | "cannot compare iterators of different containers" | When two iterators are compared, they must belong to the same container. + json.exception.[invalid_iterator](@ref invalid_iterator).213 | "cannot compare order of object iterators" | The order of object iterators cannot be compated, because JSON objects are unordered. + json.exception.[invalid_iterator](@ref invalid_iterator).214 | "cannot get value" | Cannot get value for iterator: Either the iterator belongs to a null value or it is an iterator to a primitive type (number, boolean, or string), but the iterator is different to @ref begin(). + json.exception.[type_error](@ref type_error).301 | "cannot create object from initializer list" | To create an object from an initializer list, the initializer list must consist only of a list of pairs whose first element is a string. When this constraint is violated, an array is created instead. + json.exception.[type_error](@ref type_error).302 | "type must be object, but is array" | During implicit or explicit value conversion, the JSON type must be compatible to the target type. For instance, a JSON string can only be converted into string types, but not into numbers or boolean types. + json.exception.[type_error](@ref type_error).303 | "incompatible ReferenceType for get_ref, actual type is object" | To retrieve a reference to a value stored in a @ref basic_json object with @ref get_ref, the type of the reference must match the value type. For instance, for a JSON array, the @a ReferenceType must be @ref array_t&. + json.exception.[type_error](@ref type_error).304 | "cannot use at() with string" | The @ref at() member functions can only be executed for certain JSON types. + json.exception.[type_error](@ref type_error).305 | "cannot use operator[] with string" | The @ref operator[] member functions can only be executed for certain JSON types. + json.exception.[type_error](@ref type_error).306 | "cannot use value() with string" | The @ref value() member functions can only be executed for certain JSON types. + json.exception.[type_error](@ref type_error).307 | "cannot use erase() with string" | The @ref erase() member functions can only be executed for certain JSON types. + json.exception.[type_error](@ref type_error).308 | "cannot use push_back() with string" | The @ref push_back() and @ref operator+= member functions can only be executed for certain JSON types. + json.exception.[type_error](@ref type_error).309 | "cannot use insert() with" | The @ref insert() member functions can only be executed for certain JSON types. + json.exception.[type_error](@ref type_error).310 | "cannot use swap() with number" | The @ref swap() member functions can only be executed for certain JSON types. + json.exception.[type_error](@ref type_error).313 | "invalid value to unflatten" | The @ref unflatten function converts an object whose keys are JSON Pointers back into an arbitrary nested JSON value. The JSON Pointers must not overlap, because then the resulting value would not be well defined. + json.exception.[type_error](@ref type_error).314 | "only objects can be unflattened" | The @ref unflatten function only works for an object whose keys are JSON Pointers. + json.exception.[type_error](@ref type_error).315 | "values in object must be primitive" | The @ref unflatten function only works for an object whose keys are JSON Pointers and whose values are primitive. + json.exception.[out_of_range](@ref out_of_range).401 | "array index 3 is out of range" | The provided array index @a i is larger than @a size-1. + json.exception.[out_of_range](@ref out_of_range).402 | "array index '-' (3) is out of range" | The special array index `-` in a JSON Pointer never describes a valid element of the array, but the index past the end. + json.exception.[out_of_range](@ref out_of_range).403 | "key 'foo' not found" | The provided key was not found in the JSON object. + json.exception.[out_of_range](@ref out_of_range).404 | "unresolved reference token 'foo'" | A reference token in a JSON Pointer could not be resolved. + json.exception.[out_of_range](@ref out_of_range).405 | "JSON pointer has no parent" | The JSON Patch operations 'remove' and 'add' can not be applied to the root element of the JSON value. + json.exception.other.500 | "unsuccessful" | A JSON Patch operation 'test' failed. + + @since version 3.0.0 + */ + class exception : public std::exception + { + public: + /// create exception with id an explanatory string + exception(int id_, const std::string& ename, const std::string& what_arg_) + : id(id_), + what_arg("[json.exception." + ename + "." + std::to_string(id_) + "] " + what_arg_) + {} + + /// returns the explanatory string + virtual const char* what() const noexcept + { + return what_arg.c_str(); + } + + /// the id of the exception + const int id; + + private: + /// the explanatory string + const std::string what_arg; + }; + + /*! + @brief exception indicating a parse error + + This excpetion is thrown by the library when a parse error occurs. Parse + errors can occur during the deserialization of JSON text as well as when + using JSON Patch. + + Exceptions have ids 1xx. + + @since version 3.0.0 + */ + class parse_error : public exception + { + public: + /*! + @brief create a parse error exception + @param[in] id_ the id of the exception + @param[in] byte_ the byte index where the error occured (or 0 if + the position cannot be determined) + @param[in] what_arg_ the explanatory string + */ + parse_error(int id_, size_t byte_, const std::string& what_arg_) + : exception(id_, "parse_error", "parse error" + + (byte_ != 0 ? (" at " + std::to_string(byte_)) : "") + + ": " + what_arg_), + byte(byte_) + {} + + /*! + @brief byte index of the parse error + + The byte index of the last read character in the input file. + + @note For an input with n bytes, 1 is the index of the first character + and n+1 is the index of the terminating null byte or the end of + file. + */ + const size_t byte; + }; + + /*! + @brief exception indicating errors with iterators + + Exceptions have ids 2xx. + + @since version 3.0.0 + */ + class invalid_iterator : public exception + { + public: + invalid_iterator(int id_, const std::string& what_arg_) + : exception(id_, "invalid_iterator", what_arg_) + {} + }; + + /*! + @brief exception indicating executing a member function with a wrong type + + Exceptions have ids 3xx. + + @since version 3.0.0 + */ + class type_error : public exception + { + public: + type_error(int id_, const std::string& what_arg_) + : exception(id_, "type_error", what_arg_) + {} + }; + + /*! + @brief exception indicating access out of the defined range + + Exceptions have ids 4xx. + + @since version 3.0.0 + */ + class out_of_range : public exception + { + public: + out_of_range(int id_, const std::string& what_arg_) + : exception(id_, "out_of_range", what_arg_) + {} + }; + + /// @} + + /*! @brief returns the allocator associated with the container */ @@ -6801,6 +6973,9 @@ class basic_json @return result of the deserialization + @throw parse_error.101 if a parse error occurs; example: `""unexpected end + of input; expected string literal""` + @complexity Linear in the length of the input. The parser is a predictive LL(1) parser. The complexity can be higher if the parser callback function @a cb has a super-linear complexity. @@ -9707,17 +9882,17 @@ class basic_json @return string representation of the code point; the length of the result string is between 1 and 4 characters. - @throw std::out_of_range if code point is > 0x10ffff; example: `"code - points above 0x10FFFF are invalid"` - @throw std::invalid_argument if the low surrogate is invalid; example: + @throw parse_error.102 if the low surrogate is invalid; example: `""missing or wrong low surrogate""` + @throw parse_error.103 if code point is > 0x10ffff; example: `"code + points above 0x10FFFF are invalid"` @complexity Constant. @see */ - static string_t to_unicode(const std::size_t codepoint1, - const std::size_t codepoint2 = 0) + string_t to_unicode(const std::size_t codepoint1, + const std::size_t codepoint2 = 0) const { // calculate the code point from the given code points std::size_t codepoint = codepoint1; @@ -9740,7 +9915,7 @@ class basic_json } else { - JSON_THROW(std::invalid_argument("missing or wrong low surrogate")); + JSON_THROW(parse_error(102, get_position(), "missing or wrong low surrogate")); } } @@ -9774,7 +9949,7 @@ class basic_json } else { - JSON_THROW(std::out_of_range("code points above 0x10FFFF are invalid")); + JSON_THROW(parse_error(103, get_position(), "code points above 0x10FFFF are invalid")); } return result; @@ -9869,7 +10044,7 @@ class basic_json // ignore whitespace ws = [ \t\n\r]+; - ws { continue; } + ws { position += static_cast((m_cursor - m_start)); continue; } // structural characters "[" { last_token_type = token_type::begin_array; break; } @@ -9923,6 +10098,7 @@ class basic_json */ } + position += static_cast((m_cursor - m_start)); return last_token_type; } @@ -10078,7 +10254,8 @@ class basic_json @return string value of current token without opening and closing quotes - @throw std::out_of_range if to_unicode fails + @throw parse_error.102 if to_unicode fails + @throw parse_error.103 if to_unicode fails */ string_t get_string() const { @@ -10164,7 +10341,7 @@ class basic_json // make sure there is a subsequent unicode if ((i + 6 >= m_limit) or * (i + 5) != '\\' or * (i + 6) != 'u') { - JSON_THROW(std::invalid_argument("missing low surrogate")); + JSON_THROW(parse_error(102, get_position(), "missing low surrogate")); } // get code yyyy from uxxxx\uyyyy @@ -10177,7 +10354,7 @@ class basic_json else if (codepoint >= 0xDC00 and codepoint <= 0xDFFF) { // we found a lone low surrogate - JSON_THROW(std::invalid_argument("missing high surrogate")); + JSON_THROW(parse_error(102, get_position(), "missing high surrogate")); } else { @@ -10432,6 +10609,11 @@ class basic_json return false; } + constexpr size_t get_position() const + { + return position; + } + private: /// optional input stream std::istream* m_stream = nullptr; @@ -10451,6 +10633,8 @@ class basic_json const lexer_char_t* m_limit = nullptr; /// the last token type token_type last_token_type = token_type::end_of_input; + /// current position in the input (read bytes) + size_t position = 0; }; /*! @@ -10712,7 +10896,7 @@ class basic_json "'") : lexer::token_type_name(last_token)); error_msg += "; expected " + lexer::token_type_name(t); - JSON_THROW(std::invalid_argument(error_msg)); + JSON_THROW(parse_error(101, m_lexer.get_position(), error_msg)); } } @@ -10724,7 +10908,7 @@ class basic_json error_msg += (last_token == lexer::token_type::parse_error ? ("'" + m_lexer.get_token_string() + "'") : lexer::token_type_name(last_token)); - JSON_THROW(std::invalid_argument(error_msg)); + JSON_THROW(parse_error(101, m_lexer.get_position(), error_msg)); } } diff --git a/test/src/unit-class_lexer.cpp b/test/src/unit-class_lexer.cpp index 0b019bfe..b9e602b7 100644 --- a/test/src/unit-class_lexer.cpp +++ b/test/src/unit-class_lexer.cpp @@ -190,8 +190,10 @@ TEST_CASE("lexer class") SECTION("to_unicode") { - CHECK(json::lexer::to_unicode(0x1F4A9) == "💩"); - CHECK_THROWS_AS(json::lexer::to_unicode(0x200000), std::out_of_range); - CHECK_THROWS_WITH(json::lexer::to_unicode(0x200000), "code points above 0x10FFFF are invalid"); + // lexer to call to_unicode on + json::lexer dummy_lexer(reinterpret_cast(""), 0); + CHECK(dummy_lexer.to_unicode(0x1F4A9) == "💩"); + CHECK_THROWS_AS(dummy_lexer.to_unicode(0x200000), json::parse_error); + CHECK_THROWS_WITH(dummy_lexer.to_unicode(0x200000), "[json.exception.parse_error.103] parse error: code points above 0x10FFFF are invalid"); } } diff --git a/test/src/unit-class_parser.cpp b/test/src/unit-class_parser.cpp index e3ad3723..d5e7c99b 100644 --- a/test/src/unit-class_parser.cpp +++ b/test/src/unit-class_parser.cpp @@ -89,52 +89,56 @@ TEST_CASE("parser class") SECTION("errors") { // error: tab in string - CHECK_THROWS_AS(json::parser("\"\t\"").parse(), std::invalid_argument); - CHECK_THROWS_WITH(json::parser("\"\t\"").parse(), "parse error - unexpected '\"'"); + CHECK_THROWS_AS(json::parser("\"\t\"").parse(), json::parse_error); + CHECK_THROWS_WITH(json::parser("\"\t\"").parse(), + "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '\"'"); // error: newline in string - CHECK_THROWS_AS(json::parser("\"\n\"").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("\"\r\"").parse(), std::invalid_argument); - CHECK_THROWS_WITH(json::parser("\"\n\"").parse(), "parse error - unexpected '\"'"); - CHECK_THROWS_WITH(json::parser("\"\r\"").parse(), "parse error - unexpected '\"'"); + CHECK_THROWS_AS(json::parser("\"\n\"").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("\"\r\"").parse(), json::parse_error); + CHECK_THROWS_WITH(json::parser("\"\n\"").parse(), + "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '\"'"); + CHECK_THROWS_WITH(json::parser("\"\r\"").parse(), + "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '\"'"); // error: backspace in string - CHECK_THROWS_AS(json::parser("\"\b\"").parse(), std::invalid_argument); - CHECK_THROWS_WITH(json::parser("\"\b\"").parse(), "parse error - unexpected '\"'"); + CHECK_THROWS_AS(json::parser("\"\b\"").parse(), json::parse_error); + CHECK_THROWS_WITH(json::parser("\"\b\"").parse(), + "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '\"'"); // improve code coverage - CHECK_THROWS_AS(json::parser("\uFF01").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("[-4:1,]").parse(), std::invalid_argument); + CHECK_THROWS_AS(json::parser("\uFF01").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("[-4:1,]").parse(), json::parse_error); // unescaped control characters - CHECK_THROWS_AS(json::parser("\"\x00\"").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("\"\x01\"").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("\"\x02\"").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("\"\x03\"").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("\"\x04\"").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("\"\x05\"").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("\"\x06\"").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("\"\x07\"").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("\"\x08\"").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("\"\x09\"").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("\"\x0a\"").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("\"\x0b\"").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("\"\x0c\"").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("\"\x0d\"").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("\"\x0e\"").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("\"\x0f\"").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("\"\x10\"").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("\"\x11\"").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("\"\x12\"").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("\"\x13\"").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("\"\x14\"").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("\"\x15\"").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("\"\x16\"").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("\"\x17\"").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("\"\x18\"").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("\"\x19\"").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("\"\x1a\"").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("\"\x1b\"").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("\"\x1c\"").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("\"\x1d\"").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("\"\x1e\"").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("\"\x1f\"").parse(), std::invalid_argument); + CHECK_THROWS_AS(json::parser("\"\x00\"").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("\"\x01\"").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("\"\x02\"").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("\"\x03\"").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("\"\x04\"").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("\"\x05\"").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("\"\x06\"").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("\"\x07\"").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("\"\x08\"").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("\"\x09\"").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("\"\x0a\"").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("\"\x0b\"").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("\"\x0c\"").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("\"\x0d\"").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("\"\x0e\"").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("\"\x0f\"").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("\"\x10\"").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("\"\x11\"").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("\"\x12\"").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("\"\x13\"").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("\"\x14\"").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("\"\x15\"").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("\"\x16\"").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("\"\x17\"").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("\"\x18\"").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("\"\x19\"").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("\"\x1a\"").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("\"\x1b\"").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("\"\x1c\"").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("\"\x1d\"").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("\"\x1e\"").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("\"\x1f\"").parse(), json::parse_error); } SECTION("escaped") @@ -277,60 +281,61 @@ TEST_CASE("parser class") SECTION("invalid numbers") { - CHECK_THROWS_AS(json::parser("01").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("--1").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("1.").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("1E").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("1E-").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("1.E1").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("-1E").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("-0E#").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("-0E-#").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("-0#").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("-0.0:").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("-0.0Z").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("-0E123:").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("-0e0-:").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("-0e-:").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("-0f").parse(), std::invalid_argument); + CHECK_THROWS_AS(json::parser("01").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("--1").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("1.").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("1E").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("1E-").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("1.E1").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("-1E").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("-0E#").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("-0E-#").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("-0#").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("-0.0:").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("-0.0Z").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("-0E123:").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("-0e0-:").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("-0e-:").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("-0f").parse(), json::parse_error); // numbers must not begin with "+" - CHECK_THROWS_AS(json::parser("+1").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("+0").parse(), std::invalid_argument); + CHECK_THROWS_AS(json::parser("+1").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("+0").parse(), json::parse_error); CHECK_THROWS_WITH(json::parser("01").parse(), - "parse error - unexpected '01'"); + "[json.exception.parse_error.101] parse error at 2: parse error - unexpected '01'"); CHECK_THROWS_WITH(json::parser("-01").parse(), - "parse error - unexpected '-01'"); - CHECK_THROWS_WITH(json::parser("--1").parse(), "parse error - unexpected '-'"); + "[json.exception.parse_error.101] parse error at 3: parse error - unexpected '-01'"); + CHECK_THROWS_WITH(json::parser("--1").parse(), + "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '-'"); CHECK_THROWS_WITH(json::parser("1.").parse(), - "parse error - unexpected '.'; expected end of input"); + "[json.exception.parse_error.101] parse error at 2: parse error - unexpected '.'; expected end of input"); CHECK_THROWS_WITH(json::parser("1E").parse(), - "parse error - unexpected 'E'; expected end of input"); + "[json.exception.parse_error.101] parse error at 2: parse error - unexpected 'E'; expected end of input"); CHECK_THROWS_WITH(json::parser("1E-").parse(), - "parse error - unexpected 'E'; expected end of input"); + "[json.exception.parse_error.101] parse error at 2: parse error - unexpected 'E'; expected end of input"); CHECK_THROWS_WITH(json::parser("1.E1").parse(), - "parse error - unexpected '.'; expected end of input"); + "[json.exception.parse_error.101] parse error at 2: parse error - unexpected '.'; expected end of input"); CHECK_THROWS_WITH(json::parser("-1E").parse(), - "parse error - unexpected 'E'; expected end of input"); + "[json.exception.parse_error.101] parse error at 3: parse error - unexpected 'E'; expected end of input"); CHECK_THROWS_WITH(json::parser("-0E#").parse(), - "parse error - unexpected 'E'; expected end of input"); + "[json.exception.parse_error.101] parse error at 3: parse error - unexpected 'E'; expected end of input"); CHECK_THROWS_WITH(json::parser("-0E-#").parse(), - "parse error - unexpected 'E'; expected end of input"); + "[json.exception.parse_error.101] parse error at 3: parse error - unexpected 'E'; expected end of input"); CHECK_THROWS_WITH(json::parser("-0#").parse(), - "parse error - unexpected '#'; expected end of input"); + "[json.exception.parse_error.101] parse error at 3: parse error - unexpected '#'; expected end of input"); CHECK_THROWS_WITH(json::parser("-0.0:").parse(), - "parse error - unexpected ':'; expected end of input"); + "[json.exception.parse_error.101] parse error at 5: parse error - unexpected ':'; expected end of input"); CHECK_THROWS_WITH(json::parser("-0.0Z").parse(), - "parse error - unexpected 'Z'; expected end of input"); + "[json.exception.parse_error.101] parse error at 5: parse error - unexpected 'Z'; expected end of input"); CHECK_THROWS_WITH(json::parser("-0E123:").parse(), - "parse error - unexpected ':'; expected end of input"); + "[json.exception.parse_error.101] parse error at 7: parse error - unexpected ':'; expected end of input"); CHECK_THROWS_WITH(json::parser("-0e0-:").parse(), - "parse error - unexpected '-'; expected end of input"); + "[json.exception.parse_error.101] parse error at 5: parse error - unexpected '-'; expected end of input"); CHECK_THROWS_WITH(json::parser("-0e-:").parse(), - "parse error - unexpected 'e'; expected end of input"); + "[json.exception.parse_error.101] parse error at 3: parse error - unexpected 'e'; expected end of input"); CHECK_THROWS_WITH(json::parser("-0f").parse(), - "parse error - unexpected 'f'; expected end of input"); + "[json.exception.parse_error.101] parse error at 3: parse error - unexpected 'f'; expected end of input"); } } } @@ -338,147 +343,150 @@ TEST_CASE("parser class") SECTION("parse errors") { // unexpected end of number - CHECK_THROWS_AS(json::parser("0.").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("-").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("--").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("-0.").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("-.").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("-:").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("0.:").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("e.").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("1e.").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("1e/").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("1e:").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("1E.").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("1E/").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("1E:").parse(), std::invalid_argument); + CHECK_THROWS_AS(json::parser("0.").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("-").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("--").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("-0.").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("-.").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("-:").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("0.:").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("e.").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("1e.").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("1e/").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("1e:").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("1E.").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("1E/").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("1E:").parse(), json::parse_error); CHECK_THROWS_WITH(json::parser("0.").parse(), - "parse error - unexpected '.'; expected end of input"); - CHECK_THROWS_WITH(json::parser("-").parse(), "parse error - unexpected '-'"); + "[json.exception.parse_error.101] parse error at 2: parse error - unexpected '.'; expected end of input"); + CHECK_THROWS_WITH(json::parser("-").parse(), + "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '-'"); CHECK_THROWS_WITH(json::parser("--").parse(), - "parse error - unexpected '-'"); + "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '-'"); CHECK_THROWS_WITH(json::parser("-0.").parse(), - "parse error - unexpected '.'; expected end of input"); + "[json.exception.parse_error.101] parse error at 3: parse error - unexpected '.'; expected end of input"); CHECK_THROWS_WITH(json::parser("-.").parse(), - "parse error - unexpected '-'"); + "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '-'"); CHECK_THROWS_WITH(json::parser("-:").parse(), - "parse error - unexpected '-'"); + "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '-'"); CHECK_THROWS_WITH(json::parser("0.:").parse(), - "parse error - unexpected '.'; expected end of input"); + "[json.exception.parse_error.101] parse error at 2: parse error - unexpected '.'; expected end of input"); CHECK_THROWS_WITH(json::parser("e.").parse(), - "parse error - unexpected 'e'"); + "[json.exception.parse_error.101] parse error at 1: parse error - unexpected 'e'"); CHECK_THROWS_WITH(json::parser("1e.").parse(), - "parse error - unexpected 'e'; expected end of input"); + "[json.exception.parse_error.101] parse error at 2: parse error - unexpected 'e'; expected end of input"); CHECK_THROWS_WITH(json::parser("1e/").parse(), - "parse error - unexpected 'e'; expected end of input"); + "[json.exception.parse_error.101] parse error at 2: parse error - unexpected 'e'; expected end of input"); CHECK_THROWS_WITH(json::parser("1e:").parse(), - "parse error - unexpected 'e'; expected end of input"); + "[json.exception.parse_error.101] parse error at 2: parse error - unexpected 'e'; expected end of input"); CHECK_THROWS_WITH(json::parser("1E.").parse(), - "parse error - unexpected 'E'; expected end of input"); + "[json.exception.parse_error.101] parse error at 2: parse error - unexpected 'E'; expected end of input"); CHECK_THROWS_WITH(json::parser("1E/").parse(), - "parse error - unexpected 'E'; expected end of input"); + "[json.exception.parse_error.101] parse error at 2: parse error - unexpected 'E'; expected end of input"); CHECK_THROWS_WITH(json::parser("1E:").parse(), - "parse error - unexpected 'E'; expected end of input"); + "[json.exception.parse_error.101] parse error at 2: parse error - unexpected 'E'; expected end of input"); // unexpected end of null - CHECK_THROWS_AS(json::parser("n").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("nu").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("nul").parse(), std::invalid_argument); - CHECK_THROWS_WITH(json::parser("n").parse(), "parse error - unexpected 'n'"); + CHECK_THROWS_AS(json::parser("n").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("nu").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("nul").parse(), json::parse_error); + CHECK_THROWS_WITH(json::parser("n").parse(), "[json.exception.parse_error.101] parse error at 1: parse error - unexpected 'n'"); CHECK_THROWS_WITH(json::parser("nu").parse(), - "parse error - unexpected 'n'"); + "[json.exception.parse_error.101] parse error at 1: parse error - unexpected 'n'"); CHECK_THROWS_WITH(json::parser("nul").parse(), - "parse error - unexpected 'n'"); + "[json.exception.parse_error.101] parse error at 1: parse error - unexpected 'n'"); // unexpected end of true - CHECK_THROWS_AS(json::parser("t").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("tr").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("tru").parse(), std::invalid_argument); - CHECK_THROWS_WITH(json::parser("t").parse(), "parse error - unexpected 't'"); + CHECK_THROWS_AS(json::parser("t").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("tr").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("tru").parse(), json::parse_error); + CHECK_THROWS_WITH(json::parser("t").parse(), "[json.exception.parse_error.101] parse error at 1: parse error - unexpected 't'"); CHECK_THROWS_WITH(json::parser("tr").parse(), - "parse error - unexpected 't'"); + "[json.exception.parse_error.101] parse error at 1: parse error - unexpected 't'"); CHECK_THROWS_WITH(json::parser("tru").parse(), - "parse error - unexpected 't'"); + "[json.exception.parse_error.101] parse error at 1: parse error - unexpected 't'"); // unexpected end of false - CHECK_THROWS_AS(json::parser("f").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("fa").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("fal").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("fals").parse(), std::invalid_argument); - CHECK_THROWS_WITH(json::parser("f").parse(), "parse error - unexpected 'f'"); + CHECK_THROWS_AS(json::parser("f").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("fa").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("fal").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("fals").parse(), json::parse_error); + CHECK_THROWS_WITH(json::parser("f").parse(), "[json.exception.parse_error.101] parse error at 1: parse error - unexpected 'f'"); CHECK_THROWS_WITH(json::parser("fa").parse(), - "parse error - unexpected 'f'"); + "[json.exception.parse_error.101] parse error at 1: parse error - unexpected 'f'"); CHECK_THROWS_WITH(json::parser("fal").parse(), - "parse error - unexpected 'f'"); + "[json.exception.parse_error.101] parse error at 1: parse error - unexpected 'f'"); CHECK_THROWS_WITH(json::parser("fals").parse(), - "parse error - unexpected 'f'"); + "[json.exception.parse_error.101] parse error at 1: parse error - unexpected 'f'"); // missing/unexpected end of array - CHECK_THROWS_AS(json::parser("[").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("[1").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("[1,").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("[1,]").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("]").parse(), std::invalid_argument); + CHECK_THROWS_AS(json::parser("[").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("[1").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("[1,").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("[1,]").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("]").parse(), json::parse_error); CHECK_THROWS_WITH(json::parser("[").parse(), - "parse error - unexpected end of input"); + "[json.exception.parse_error.101] parse error at 2: parse error - unexpected end of input"); CHECK_THROWS_WITH(json::parser("[1").parse(), - "parse error - unexpected end of input; expected ']'"); + "[json.exception.parse_error.101] parse error at 3: parse error - unexpected end of input; expected ']'"); CHECK_THROWS_WITH(json::parser("[1,").parse(), - "parse error - unexpected end of input"); + "[json.exception.parse_error.101] parse error at 4: parse error - unexpected end of input"); CHECK_THROWS_WITH(json::parser("[1,]").parse(), - "parse error - unexpected ']'"); - CHECK_THROWS_WITH(json::parser("]").parse(), "parse error - unexpected ']'"); + "[json.exception.parse_error.101] parse error at 4: parse error - unexpected ']'"); + CHECK_THROWS_WITH(json::parser("]").parse(), + "[json.exception.parse_error.101] parse error at 1: parse error - unexpected ']'"); // missing/unexpected end of object - CHECK_THROWS_AS(json::parser("{").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("{\"foo\"").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("{\"foo\":").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("{\"foo\":}").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("{\"foo\":1,}").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("}").parse(), std::invalid_argument); + CHECK_THROWS_AS(json::parser("{").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("{\"foo\"").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("{\"foo\":").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("{\"foo\":}").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("{\"foo\":1,}").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("}").parse(), json::parse_error); CHECK_THROWS_WITH(json::parser("{").parse(), - "parse error - unexpected end of input; expected string literal"); + "[json.exception.parse_error.101] parse error at 2: parse error - unexpected end of input; expected string literal"); CHECK_THROWS_WITH(json::parser("{\"foo\"").parse(), - "parse error - unexpected end of input; expected ':'"); + "[json.exception.parse_error.101] parse error at 7: parse error - unexpected end of input; expected ':'"); CHECK_THROWS_WITH(json::parser("{\"foo\":").parse(), - "parse error - unexpected end of input"); + "[json.exception.parse_error.101] parse error at 8: parse error - unexpected end of input"); CHECK_THROWS_WITH(json::parser("{\"foo\":}").parse(), - "parse error - unexpected '}'"); + "[json.exception.parse_error.101] parse error at 8: parse error - unexpected '}'"); CHECK_THROWS_WITH(json::parser("{\"foo\":1,}").parse(), - "parse error - unexpected '}'; expected string literal"); - CHECK_THROWS_WITH(json::parser("}").parse(), "parse error - unexpected '}'"); + "[json.exception.parse_error.101] parse error at 10: parse error - unexpected '}'; expected string literal"); + CHECK_THROWS_WITH(json::parser("}").parse(), + "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '}'"); // missing/unexpected end of string - CHECK_THROWS_AS(json::parser("\"").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("\"\\\"").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("\"\\u\"").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("\"\\u0\"").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("\"\\u01\"").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("\"\\u012\"").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("\"\\u").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("\"\\u0").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("\"\\u01").parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser("\"\\u012").parse(), std::invalid_argument); + CHECK_THROWS_AS(json::parser("\"").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("\"\\\"").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("\"\\u\"").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("\"\\u0\"").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("\"\\u01\"").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("\"\\u012\"").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("\"\\u").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("\"\\u0").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("\"\\u01").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser("\"\\u012").parse(), json::parse_error); CHECK_THROWS_WITH(json::parser("\"").parse(), - "parse error - unexpected '\"'"); + "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '\"'"); CHECK_THROWS_WITH(json::parser("\"\\\"").parse(), - "parse error - unexpected '\"'"); + "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '\"'"); CHECK_THROWS_WITH(json::parser("\"\\u\"").parse(), - "parse error - unexpected '\"'"); + "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '\"'"); CHECK_THROWS_WITH(json::parser("\"\\u0\"").parse(), - "parse error - unexpected '\"'"); + "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '\"'"); CHECK_THROWS_WITH(json::parser("\"\\u01\"").parse(), - "parse error - unexpected '\"'"); + "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '\"'"); CHECK_THROWS_WITH(json::parser("\"\\u012\"").parse(), - "parse error - unexpected '\"'"); + "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '\"'"); CHECK_THROWS_WITH(json::parser("\"\\u").parse(), - "parse error - unexpected '\"'"); + "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '\"'"); CHECK_THROWS_WITH(json::parser("\"\\u0").parse(), - "parse error - unexpected '\"'"); + "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '\"'"); CHECK_THROWS_WITH(json::parser("\"\\u01").parse(), - "parse error - unexpected '\"'"); + "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '\"'"); CHECK_THROWS_WITH(json::parser("\"\\u012").parse(), - "parse error - unexpected '\"'"); + "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '\"'"); // invalid escapes for (int c = 1; c < 128; ++c) @@ -510,8 +518,9 @@ TEST_CASE("parser class") // any other combination of backslash and character is invalid default: { - CHECK_THROWS_AS(json::parser(s.c_str()).parse(), std::invalid_argument); - CHECK_THROWS_WITH(json::parser(s.c_str()).parse(), "parse error - unexpected '\"'"); + CHECK_THROWS_AS(json::parser(s.c_str()).parse(), json::parse_error); + CHECK_THROWS_WITH(json::parser(s.c_str()).parse(), + "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '\"'"); break; } } @@ -576,40 +585,49 @@ TEST_CASE("parser class") } else { - CHECK_THROWS_AS(json::parser(s1.c_str()).parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser(s2.c_str()).parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser(s3.c_str()).parse(), std::invalid_argument); - CHECK_THROWS_AS(json::parser(s4.c_str()).parse(), std::invalid_argument); + CHECK_THROWS_AS(json::parser(s1.c_str()).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(s2.c_str()).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(s3.c_str()).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(s4.c_str()).parse(), json::parse_error); - CHECK_THROWS_WITH(json::parser(s1.c_str()).parse(), "parse error - unexpected '\"'"); - CHECK_THROWS_WITH(json::parser(s2.c_str()).parse(), "parse error - unexpected '\"'"); - CHECK_THROWS_WITH(json::parser(s3.c_str()).parse(), "parse error - unexpected '\"'"); - CHECK_THROWS_WITH(json::parser(s4.c_str()).parse(), "parse error - unexpected '\"'"); + CHECK_THROWS_WITH(json::parser(s1.c_str()).parse(), + "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '\"'"); + CHECK_THROWS_WITH(json::parser(s2.c_str()).parse(), + "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '\"'"); + CHECK_THROWS_WITH(json::parser(s3.c_str()).parse(), + "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '\"'"); + CHECK_THROWS_WITH(json::parser(s4.c_str()).parse(), + "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '\"'"); } } } // missing part of a surrogate pair - CHECK_THROWS_AS(json::parse("\"\\uD80C\""), std::invalid_argument); - CHECK_THROWS_WITH(json::parse("\"\\uD80C\""), "missing low surrogate"); + CHECK_THROWS_AS(json::parse("\"\\uD80C\""), json::parse_error); + CHECK_THROWS_WITH(json::parse("\"\\uD80C\""), + "[json.exception.parse_error.102] parse error at 8: missing low surrogate"); // invalid surrogate pair - CHECK_THROWS_AS(json::parse("\"\\uD80C\\uD80C\""), std::invalid_argument); - CHECK_THROWS_AS(json::parse("\"\\uD80C\\u0000\""), std::invalid_argument); - CHECK_THROWS_AS(json::parse("\"\\uD80C\\uFFFF\""), std::invalid_argument); + CHECK_THROWS_AS(json::parse("\"\\uD80C\\uD80C\""), json::parse_error); + CHECK_THROWS_AS(json::parse("\"\\uD80C\\u0000\""), json::parse_error); + CHECK_THROWS_AS(json::parse("\"\\uD80C\\uFFFF\""), json::parse_error); CHECK_THROWS_WITH(json::parse("\"\\uD80C\\uD80C\""), - "missing or wrong low surrogate"); + "[json.exception.parse_error.102] parse error at 14: missing or wrong low surrogate"); CHECK_THROWS_WITH(json::parse("\"\\uD80C\\u0000\""), - "missing or wrong low surrogate"); + "[json.exception.parse_error.102] parse error at 14: missing or wrong low surrogate"); CHECK_THROWS_WITH(json::parse("\"\\uD80C\\uFFFF\""), - "missing or wrong low surrogate"); + "[json.exception.parse_error.102] parse error at 14: missing or wrong low surrogate"); } SECTION("tests found by mutate++") { // test case to make sure no comma preceeds the first key - CHECK_THROWS_AS(json::parser("{,\"key\": false}").parse(), std::invalid_argument); + CHECK_THROWS_AS(json::parser("{,\"key\": false}").parse(), json::parse_error); + CHECK_THROWS_WITH(json::parser("{,\"key\": false}").parse(), + "[json.exception.parse_error.101] parse error at 2: parse error - unexpected ','"); // test case to make sure an object is properly closed - CHECK_THROWS_AS(json::parser("[{\"key\": false true]").parse(), std::invalid_argument); + CHECK_THROWS_AS(json::parser("[{\"key\": false true]").parse(), json::parse_error); + CHECK_THROWS_WITH(json::parser("[{\"key\": false true]").parse(), + "[json.exception.parse_error.101] parse error at 19: parse error - unexpected true literal; expected '}'"); // test case to make sure the callback is properly evaluated after reading a key { diff --git a/test/src/unit-deserialization.cpp b/test/src/unit-deserialization.cpp index 9028fdfb..cacf6687 100644 --- a/test/src/unit-deserialization.cpp +++ b/test/src/unit-deserialization.cpp @@ -90,15 +90,17 @@ TEST_CASE("deserialization") std::stringstream ss1, ss2; ss1 << "[\"foo\",1,2,3,false,{\"one\":1}"; ss2 << "[\"foo\",1,2,3,false,{\"one\":1}"; - CHECK_THROWS_AS(json::parse(ss1), std::invalid_argument); - CHECK_THROWS_WITH(json::parse(ss2), "parse error - unexpected end of input; expected ']'"); + CHECK_THROWS_AS(json::parse(ss1), json::parse_error); + CHECK_THROWS_WITH(json::parse(ss2), + "[json.exception.parse_error.101] parse error at 30: parse error - unexpected end of input; expected ']'"); } SECTION("string") { json::string_t s = "[\"foo\",1,2,3,false,{\"one\":1}"; - CHECK_THROWS_AS(json::parse(s), std::invalid_argument); - CHECK_THROWS_WITH(json::parse(s), "parse error - unexpected end of input; expected ']'"); + CHECK_THROWS_AS(json::parse(s), json::parse_error); + CHECK_THROWS_WITH(json::parse(s), + "[json.exception.parse_error.101] parse error at 29: parse error - unexpected end of input; expected ']'"); } SECTION("operator<<") @@ -107,8 +109,9 @@ TEST_CASE("deserialization") ss1 << "[\"foo\",1,2,3,false,{\"one\":1}"; ss2 << "[\"foo\",1,2,3,false,{\"one\":1}"; json j; - CHECK_THROWS_AS(j << ss1, std::invalid_argument); - CHECK_THROWS_WITH(j << ss2, "parse error - unexpected end of input; expected ']'"); + CHECK_THROWS_AS(j << ss1, json::parse_error); + CHECK_THROWS_WITH(j << ss2, + "[json.exception.parse_error.101] parse error at 30: parse error - unexpected end of input; expected ']'"); } SECTION("operator>>") @@ -117,15 +120,16 @@ TEST_CASE("deserialization") ss1 << "[\"foo\",1,2,3,false,{\"one\":1}"; ss2 << "[\"foo\",1,2,3,false,{\"one\":1}"; json j; - CHECK_THROWS_AS(ss1 >> j, std::invalid_argument); - CHECK_THROWS_WITH(ss2 >> j, "parse error - unexpected end of input; expected ']'"); + CHECK_THROWS_AS(ss1 >> j, json::parse_error); + CHECK_THROWS_WITH(ss2 >> j, + "[json.exception.parse_error.101] parse error at 30: parse error - unexpected end of input; expected ']'"); } SECTION("user-defined string literal") { - CHECK_THROWS_AS("[\"foo\",1,2,3,false,{\"one\":1}"_json, std::invalid_argument); + CHECK_THROWS_AS("[\"foo\",1,2,3,false,{\"one\":1}"_json, json::parse_error); CHECK_THROWS_WITH("[\"foo\",1,2,3,false,{\"one\":1}"_json, - "parse error - unexpected end of input; expected ']'"); + "[json.exception.parse_error.101] parse error at 29: parse error - unexpected end of input; expected ']'"); } } @@ -178,7 +182,7 @@ TEST_CASE("deserialization") SECTION("empty container") { std::vector v; - CHECK_THROWS_AS(json::parse(v), std::invalid_argument); + CHECK_THROWS_AS(json::parse(v), json::parse_error); } } @@ -223,7 +227,7 @@ TEST_CASE("deserialization") SECTION("with empty range") { std::vector v; - CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), std::invalid_argument); + CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); } } @@ -233,91 +237,91 @@ TEST_CASE("deserialization") SECTION("case 1") { uint8_t v[] = {'\"', 'a', 'a', 'a', 'a', 'a', 'a', '\\', 'u'}; - CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), std::invalid_argument); + CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); } SECTION("case 2") { uint8_t v[] = {'\"', 'a', 'a', 'a', 'a', 'a', 'a', '\\', 'u', '1'}; - CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), std::invalid_argument); + CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); } SECTION("case 3") { uint8_t v[] = {'\"', 'a', 'a', 'a', 'a', 'a', 'a', '\\', 'u', '1', '1', '1', '1', '1', '1', '1', '1'}; - CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), std::invalid_argument); + CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); } SECTION("case 4") { uint8_t v[] = {'\"', 'a', 'a', 'a', 'a', 'a', 'a', 'u', '1', '1', '1', '1', '1', '1', '1', '1', '\\'}; - CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), std::invalid_argument); + CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); } SECTION("case 5") { uint8_t v[] = {'\"', 0x7F, 0xC1}; - CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), std::invalid_argument); + CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); } SECTION("case 6") { uint8_t v[] = {'\"', 0x7F, 0xDF, 0x7F}; - CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), std::invalid_argument); + CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); } SECTION("case 7") { uint8_t v[] = {'\"', 0x7F, 0xDF, 0xC0}; - CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), std::invalid_argument); + CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); } SECTION("case 8") { uint8_t v[] = {'\"', 0x7F, 0xE0, 0x9F}; - CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), std::invalid_argument); + CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); } SECTION("case 9") { uint8_t v[] = {'\"', 0x7F, 0xEF, 0xC0}; - CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), std::invalid_argument); + CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); } SECTION("case 10") { uint8_t v[] = {'\"', 0x7F, 0xED, 0x7F}; - CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), std::invalid_argument); + CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); } SECTION("case 11") { uint8_t v[] = {'\"', 0x7F, 0xF0, 0x8F}; - CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), std::invalid_argument); + CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); } SECTION("case 12") { uint8_t v[] = {'\"', 0x7F, 0xF0, 0xC0}; - CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), std::invalid_argument); + CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); } SECTION("case 13") { uint8_t v[] = {'\"', 0x7F, 0xF3, 0x7F}; - CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), std::invalid_argument); + CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); } SECTION("case 14") { uint8_t v[] = {'\"', 0x7F, 0xF3, 0xC0}; - CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), std::invalid_argument); + CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); } SECTION("case 15") { uint8_t v[] = {'\"', 0x7F, 0xF4, 0x7F}; - CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), std::invalid_argument); + CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); } } } diff --git a/test/src/unit-regression.cpp b/test/src/unit-regression.cpp index 7980371b..9dea8083 100644 --- a/test/src/unit-regression.cpp +++ b/test/src/unit-regression.cpp @@ -579,8 +579,9 @@ TEST_CASE("regression tests") // ss is not at EOF; this yielded an error before the fix // (threw basic_string::append). No, it should just throw // a parse error because of the EOF. - CHECK_THROWS_AS(j << ss, std::invalid_argument); - CHECK_THROWS_WITH(j << ss, "parse error - unexpected end of input"); + CHECK_THROWS_AS(j << ss, json::parse_error); + CHECK_THROWS_WITH(j << ss, + "[json.exception.parse_error.101] parse error at 1: parse error - unexpected end of input"); } SECTION("issue #389 - Integer-overflow (OSS-Fuzz issue 267)") @@ -778,7 +779,7 @@ TEST_CASE("regression tests") SECTION("issue #452 - Heap-buffer-overflow (OSS-Fuzz issue 585)") { std::vector vec = {'-', '0', '1', '2', '2', '7', '4'}; - CHECK_THROWS_AS(json::parse(vec), std::invalid_argument); + CHECK_THROWS_AS(json::parse(vec), json::parse_error); } SECTION("issue #454 - doubles are printed as integers") diff --git a/test/src/unit-testsuites.cpp b/test/src/unit-testsuites.cpp index e83d5b74..701348a8 100644 --- a/test/src/unit-testsuites.cpp +++ b/test/src/unit-testsuites.cpp @@ -79,7 +79,7 @@ TEST_CASE("compliance tests from json.org") CAPTURE(filename); json j; std::ifstream f(filename); - CHECK_THROWS_AS(j << f, std::invalid_argument); + CHECK_THROWS_AS(j << f, json::parse_error); } } @@ -757,7 +757,7 @@ TEST_CASE("nst's JSONTestSuite") CAPTURE(filename); std::ifstream f(filename); json j; - CHECK_THROWS_AS(j << f, std::invalid_argument); + CHECK_THROWS_AS(j << f, json::parse_error); } } @@ -810,7 +810,7 @@ TEST_CASE("nst's JSONTestSuite") CAPTURE(filename); std::ifstream f(filename); json j; - CHECK_THROWS_AS(j << f, std::invalid_argument); + CHECK_THROWS_AS(j << f, json::parse_error); } } } diff --git a/test/src/unit-unicode.cpp b/test/src/unit-unicode.cpp index c429b91c..f7cf0ada 100644 --- a/test/src/unit-unicode.cpp +++ b/test/src/unit-unicode.cpp @@ -38,6 +38,9 @@ TEST_CASE("Unicode", "[hide]") { SECTION("full enumeration of Unicode code points") { + // lexer to call to_unicode on + json::lexer dummy_lexer(reinterpret_cast(""), 0); + // create an escaped string from a code point const auto codepoint_to_unicode = [](std::size_t cp) { @@ -85,7 +88,7 @@ TEST_CASE("Unicode", "[hide]") // they are checked with codepoint_to_unicode. if (cp > 0x1f and cp != 0x22 and cp != 0x5c) { - unescaped_string = json::lexer::to_unicode(cp); + unescaped_string = dummy_lexer.to_unicode(cp); } } else @@ -97,7 +100,7 @@ TEST_CASE("Unicode", "[hide]") const auto codepoint2 = 0xdc00u + ((cp - 0x10000u) & 0x3ffu); escaped_string = codepoint_to_unicode(codepoint1); escaped_string += codepoint_to_unicode(codepoint2); - unescaped_string += json::lexer::to_unicode(codepoint1, codepoint2); + unescaped_string += dummy_lexer.to_unicode(codepoint1, codepoint2); } // all other code points are valid and must not yield parse errors @@ -170,7 +173,7 @@ TEST_CASE("Unicode", "[hide]") SECTION("error for incomplete/wrong BOM") { - CHECK_THROWS_AS(json::parse("\xef\xbb"), std::invalid_argument); - CHECK_THROWS_AS(json::parse("\xef\xbb\xbb"), std::invalid_argument); + CHECK_THROWS_AS(json::parse("\xef\xbb"), json::parse_error); + CHECK_THROWS_AS(json::parse("\xef\xbb\xbb"), json::parse_error); } } From 06815d274e6dc95e61411e01596d8f0cc8748e99 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Thu, 2 Mar 2017 17:55:13 +0100 Subject: [PATCH 021/530] :hammer: added user-defined exceptions 104 and 105 These exceptions are thrown in case of parse errors in JSON patch documents. --- src/json.hpp | 21 +++--- src/json.hpp.re2c | 21 +++--- test/src/unit-json_patch.cpp | 120 +++++++++++++++++++++-------------- 3 files changed, 94 insertions(+), 68 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index d4e87e5e..6451d732 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -1107,8 +1107,8 @@ class basic_json json.exception.[parse_error](@ref parse_error).101 | `"parse error at 2: unexpected end of input; expected string literal"` | This error indicates a syntax error while deserializing a JSON text. The error message describes that an unexpected token (character) was encountered, and the member @ref parse_error::byte indicates the error position. json.exception.[parse_error](@ref parse_error).102 | `"parse error at 14: missing or wrong low surrogate"` | JSON uses the `\uxxxx` format to describe Unicode characters. Code points above above 0xFFFF are split into two `\uxxxx` entries ("surrogate pairs"). This error indicates that the surrogate pair is incomplete or contains an invalid code point. json.exception.[parse_error](@ref parse_error).103 | `"parse error: code points above 0x10FFFF are invalid"` | Unicode supports code points up to 0x10FFFF. Code points above 0x10FFFF are invalid. - json.exception.[parse_error](@ref parse_error).104 | "parse error: JSON patch must be an array of objects" | [RFC 6902](https://tools.ietf.org/html/rfc6902) requires a JSON Patch document to be a JSON documentthat represents an array of objects. - json.exception.[parse_error](@ref parse_error).105 | "parse error: operation must have string member 'op'" | An operation of a JSON Patch document must contain Operation objects MUST have exactly one "op" member, whose value indicates the operation to perform. Its value MUST be one of "add", "remove", "replace", "move", "copy", or "test"; other values are errors. + json.exception.[parse_error](@ref parse_error).104 | `"parse error: JSON patch must be an array of objects"` | [RFC 6902](https://tools.ietf.org/html/rfc6902) requires a JSON Patch document to be a JSON document that represents an array of objects. + json.exception.[parse_error](@ref parse_error).105 | `"parse error: operation must have string member 'op'"` | An operation of a JSON Patch document must contain exactly one "op" member, whose value indicates the operation to perform. Its value must be one of "add", "remove", "replace", "move", "copy", or "test"; other values are errors. json.exception.[parse_error](@ref parse_error).106 | "parse error: array index must not begin with '0'" | An array index in a JSON Pointer ([RFC 6901](https://tools.ietf.org/html/rfc6901)) may be `0` or any number wihtout a leading `0`. json.exception.[parse_error](@ref parse_error).107 | "parse error: JSON pointer must be empty or begin with '/'" | A JSON Pointer must be a Unicode string containing a sequence of zero or more reference tokens, each prefixed by a `/` character. json.exception.[parse_error](@ref parse_error).108 | "parse error: escape character '~' must be followed with '0' or '1'" | In a JSON Pointer, only `~0` and `~1` are valid escape sequences. @@ -12713,6 +12713,8 @@ basic_json_parser_74: not found"` @throw invalid_argument if the JSON patch is malformed (e.g., mandatory attributes are missing); example: `"operation add must have member path"` + @throw parse_error.104 if the JSON patch does not consist of an array of + objects @complexity Linear in the size of the JSON value and the length of the JSON patch. As usually only a fraction of the JSON value is affected by @@ -12858,11 +12860,10 @@ basic_json_parser_74: } }; - // type check + // type check: top level value must be an array if (not json_patch.is_array()) { - // a JSON patch must be an array of objects - JSON_THROW(std::invalid_argument("JSON patch must be an array of objects")); + JSON_THROW(parse_error(104, 0, "JSON patch must be an array of objects")); } // iterate and apply the operations @@ -12882,23 +12883,23 @@ basic_json_parser_74: // check if desired value is present if (it == val.m_value.object->end()) { - JSON_THROW(std::invalid_argument(error_msg + " must have member '" + member + "'")); + JSON_THROW(parse_error(105, 0, error_msg + " must have member '" + member + "'")); } // check if result is of type string if (string_type and not it->second.is_string()) { - JSON_THROW(std::invalid_argument(error_msg + " must have string member '" + member + "'")); + JSON_THROW(parse_error(105, 0, error_msg + " must have string member '" + member + "'")); } // no error: return value return it->second; }; - // type check + // type check: every element of the array must be an object if (not val.is_object()) { - JSON_THROW(std::invalid_argument("JSON patch must be an array of objects")); + JSON_THROW(parse_error(104, 0, "JSON patch must be an array of objects")); } // collect mandatory members @@ -12981,7 +12982,7 @@ basic_json_parser_74: { // op must be "add", "remove", "replace", "move", "copy", or // "test" - JSON_THROW(std::invalid_argument("operation value '" + op + "' is invalid")); + JSON_THROW(parse_error(105, 0, "operation value '" + op + "' is invalid")); } } } diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 1afc025a..3ed26c53 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -1107,8 +1107,8 @@ class basic_json json.exception.[parse_error](@ref parse_error).101 | `"parse error at 2: unexpected end of input; expected string literal"` | This error indicates a syntax error while deserializing a JSON text. The error message describes that an unexpected token (character) was encountered, and the member @ref parse_error::byte indicates the error position. json.exception.[parse_error](@ref parse_error).102 | `"parse error at 14: missing or wrong low surrogate"` | JSON uses the `\uxxxx` format to describe Unicode characters. Code points above above 0xFFFF are split into two `\uxxxx` entries ("surrogate pairs"). This error indicates that the surrogate pair is incomplete or contains an invalid code point. json.exception.[parse_error](@ref parse_error).103 | `"parse error: code points above 0x10FFFF are invalid"` | Unicode supports code points up to 0x10FFFF. Code points above 0x10FFFF are invalid. - json.exception.[parse_error](@ref parse_error).104 | "parse error: JSON patch must be an array of objects" | [RFC 6902](https://tools.ietf.org/html/rfc6902) requires a JSON Patch document to be a JSON documentthat represents an array of objects. - json.exception.[parse_error](@ref parse_error).105 | "parse error: operation must have string member 'op'" | An operation of a JSON Patch document must contain Operation objects MUST have exactly one "op" member, whose value indicates the operation to perform. Its value MUST be one of "add", "remove", "replace", "move", "copy", or "test"; other values are errors. + json.exception.[parse_error](@ref parse_error).104 | `"parse error: JSON patch must be an array of objects"` | [RFC 6902](https://tools.ietf.org/html/rfc6902) requires a JSON Patch document to be a JSON document that represents an array of objects. + json.exception.[parse_error](@ref parse_error).105 | `"parse error: operation must have string member 'op'"` | An operation of a JSON Patch document must contain exactly one "op" member, whose value indicates the operation to perform. Its value must be one of "add", "remove", "replace", "move", "copy", or "test"; other values are errors. json.exception.[parse_error](@ref parse_error).106 | "parse error: array index must not begin with '0'" | An array index in a JSON Pointer ([RFC 6901](https://tools.ietf.org/html/rfc6901)) may be `0` or any number wihtout a leading `0`. json.exception.[parse_error](@ref parse_error).107 | "parse error: JSON pointer must be empty or begin with '/'" | A JSON Pointer must be a Unicode string containing a sequence of zero or more reference tokens, each prefixed by a `/` character. json.exception.[parse_error](@ref parse_error).108 | "parse error: escape character '~' must be followed with '0' or '1'" | In a JSON Pointer, only `~0` and `~1` are valid escape sequences. @@ -11746,6 +11746,8 @@ class basic_json not found"` @throw invalid_argument if the JSON patch is malformed (e.g., mandatory attributes are missing); example: `"operation add must have member path"` + @throw parse_error.104 if the JSON patch does not consist of an array of + objects @complexity Linear in the size of the JSON value and the length of the JSON patch. As usually only a fraction of the JSON value is affected by @@ -11891,11 +11893,10 @@ class basic_json } }; - // type check + // type check: top level value must be an array if (not json_patch.is_array()) { - // a JSON patch must be an array of objects - JSON_THROW(std::invalid_argument("JSON patch must be an array of objects")); + JSON_THROW(parse_error(104, 0, "JSON patch must be an array of objects")); } // iterate and apply the operations @@ -11915,23 +11916,23 @@ class basic_json // check if desired value is present if (it == val.m_value.object->end()) { - JSON_THROW(std::invalid_argument(error_msg + " must have member '" + member + "'")); + JSON_THROW(parse_error(105, 0, error_msg + " must have member '" + member + "'")); } // check if result is of type string if (string_type and not it->second.is_string()) { - JSON_THROW(std::invalid_argument(error_msg + " must have string member '" + member + "'")); + JSON_THROW(parse_error(105, 0, error_msg + " must have string member '" + member + "'")); } // no error: return value return it->second; }; - // type check + // type check: every element of the array must be an object if (not val.is_object()) { - JSON_THROW(std::invalid_argument("JSON patch must be an array of objects")); + JSON_THROW(parse_error(104, 0, "JSON patch must be an array of objects")); } // collect mandatory members @@ -12014,7 +12015,7 @@ class basic_json { // op must be "add", "remove", "replace", "move", "copy", or // "test" - JSON_THROW(std::invalid_argument("operation value '" + op + "' is invalid")); + JSON_THROW(parse_error(105, 0, "operation value '" + op + "' is invalid")); } } } diff --git a/test/src/unit-json_patch.cpp b/test/src/unit-json_patch.cpp index 05ed7502..4dd89b75 100644 --- a/test/src/unit-json_patch.cpp +++ b/test/src/unit-json_patch.cpp @@ -666,40 +666,45 @@ TEST_CASE("JSON patch") { json j; json patch = {{"op", "add"}, {"path", ""}, {"value", 1}}; - CHECK_THROWS_AS(j.patch(patch), std::invalid_argument); - CHECK_THROWS_WITH(j.patch(patch), "JSON patch must be an array of objects"); + CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_WITH(j.patch(patch), + "[json.exception.parse_error.104] parse error: JSON patch must be an array of objects"); } SECTION("not an array of objects") { json j; json patch = {"op", "add", "path", "", "value", 1}; - CHECK_THROWS_AS(j.patch(patch), std::invalid_argument); - CHECK_THROWS_WITH(j.patch(patch), "JSON patch must be an array of objects"); + CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_WITH(j.patch(patch), + "[json.exception.parse_error.104] parse error: JSON patch must be an array of objects"); } SECTION("missing 'op'") { json j; json patch = {{{"foo", "bar"}}}; - CHECK_THROWS_AS(j.patch(patch), std::invalid_argument); - CHECK_THROWS_WITH(j.patch(patch), "operation must have member 'op'"); + CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_WITH(j.patch(patch), + "[json.exception.parse_error.105] parse error: operation must have member 'op'"); } SECTION("non-string 'op'") { json j; json patch = {{{"op", 1}}}; - CHECK_THROWS_AS(j.patch(patch), std::invalid_argument); - CHECK_THROWS_WITH(j.patch(patch), "operation must have string member 'op'"); + CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_WITH(j.patch(patch), + "[json.exception.parse_error.105] parse error: operation must have string member 'op'"); } SECTION("invalid operation") { json j; json patch = {{{"op", "foo"}, {"path", ""}}}; - CHECK_THROWS_AS(j.patch(patch), std::invalid_argument); - CHECK_THROWS_WITH(j.patch(patch), "operation value 'foo' is invalid"); + CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_WITH(j.patch(patch), + "[json.exception.parse_error.105] parse error: operation value 'foo' is invalid"); } } @@ -709,24 +714,27 @@ TEST_CASE("JSON patch") { json j; json patch = {{{"op", "add"}}}; - CHECK_THROWS_AS(j.patch(patch), std::invalid_argument); - CHECK_THROWS_WITH(j.patch(patch), "operation 'add' must have member 'path'"); + CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_WITH(j.patch(patch), + "[json.exception.parse_error.105] parse error: operation 'add' must have member 'path'"); } SECTION("non-string 'path'") { json j; json patch = {{{"op", "add"}, {"path", 1}}}; - CHECK_THROWS_AS(j.patch(patch), std::invalid_argument); - CHECK_THROWS_WITH(j.patch(patch), "operation 'add' must have string member 'path'"); + CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_WITH(j.patch(patch), + "[json.exception.parse_error.105] parse error: operation 'add' must have string member 'path'"); } SECTION("missing 'value'") { json j; json patch = {{{"op", "add"}, {"path", ""}}}; - CHECK_THROWS_AS(j.patch(patch), std::invalid_argument); - CHECK_THROWS_WITH(j.patch(patch), "operation 'add' must have member 'value'"); + CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_WITH(j.patch(patch), + "[json.exception.parse_error.105] parse error: operation 'add' must have member 'value'"); } SECTION("invalid array index") @@ -744,16 +752,18 @@ TEST_CASE("JSON patch") { json j; json patch = {{{"op", "remove"}}}; - CHECK_THROWS_AS(j.patch(patch), std::invalid_argument); - CHECK_THROWS_WITH(j.patch(patch), "operation 'remove' must have member 'path'"); + CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_WITH(j.patch(patch), + "[json.exception.parse_error.105] parse error: operation 'remove' must have member 'path'"); } SECTION("non-string 'path'") { json j; json patch = {{{"op", "remove"}, {"path", 1}}}; - CHECK_THROWS_AS(j.patch(patch), std::invalid_argument); - CHECK_THROWS_WITH(j.patch(patch), "operation 'remove' must have string member 'path'"); + CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_WITH(j.patch(patch), + "[json.exception.parse_error.105] parse error: operation 'remove' must have string member 'path'"); } SECTION("nonexisting target location (array)") @@ -787,24 +797,27 @@ TEST_CASE("JSON patch") { json j; json patch = {{{"op", "replace"}}}; - CHECK_THROWS_AS(j.patch(patch), std::invalid_argument); - CHECK_THROWS_WITH(j.patch(patch), "operation 'replace' must have member 'path'"); + CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_WITH(j.patch(patch), + "[json.exception.parse_error.105] parse error: operation 'replace' must have member 'path'"); } SECTION("non-string 'path'") { json j; json patch = {{{"op", "replace"}, {"path", 1}}}; - CHECK_THROWS_AS(j.patch(patch), std::invalid_argument); - CHECK_THROWS_WITH(j.patch(patch), "operation 'replace' must have string member 'path'"); + CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_WITH(j.patch(patch), + "[json.exception.parse_error.105] parse error: operation 'replace' must have string member 'path'"); } SECTION("missing 'value'") { json j; json patch = {{{"op", "replace"}, {"path", ""}}}; - CHECK_THROWS_AS(j.patch(patch), std::invalid_argument); - CHECK_THROWS_WITH(j.patch(patch), "operation 'replace' must have member 'value'"); + CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_WITH(j.patch(patch), + "[json.exception.parse_error.105] parse error: operation 'replace' must have member 'value'"); } SECTION("nonexisting target location (array)") @@ -830,32 +843,36 @@ TEST_CASE("JSON patch") { json j; json patch = {{{"op", "move"}}}; - CHECK_THROWS_AS(j.patch(patch), std::invalid_argument); - CHECK_THROWS_WITH(j.patch(patch), "operation 'move' must have member 'path'"); + CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_WITH(j.patch(patch), + "[json.exception.parse_error.105] parse error: operation 'move' must have member 'path'"); } SECTION("non-string 'path'") { json j; json patch = {{{"op", "move"}, {"path", 1}}}; - CHECK_THROWS_AS(j.patch(patch), std::invalid_argument); - CHECK_THROWS_WITH(j.patch(patch), "operation 'move' must have string member 'path'"); + CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_WITH(j.patch(patch), + "[json.exception.parse_error.105] parse error: operation 'move' must have string member 'path'"); } SECTION("missing 'from'") { json j; json patch = {{{"op", "move"}, {"path", ""}}}; - CHECK_THROWS_AS(j.patch(patch), std::invalid_argument); - CHECK_THROWS_WITH(j.patch(patch), "operation 'move' must have member 'from'"); + CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_WITH(j.patch(patch), + "[json.exception.parse_error.105] parse error: operation 'move' must have member 'from'"); } SECTION("non-string 'from'") { json j; json patch = {{{"op", "move"}, {"path", ""}, {"from", 1}}}; - CHECK_THROWS_AS(j.patch(patch), std::invalid_argument); - CHECK_THROWS_WITH(j.patch(patch), "operation 'move' must have string member 'from'"); + CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_WITH(j.patch(patch), + "[json.exception.parse_error.105] parse error: operation 'move' must have string member 'from'"); } SECTION("nonexisting from location (array)") @@ -881,32 +898,36 @@ TEST_CASE("JSON patch") { json j; json patch = {{{"op", "copy"}}}; - CHECK_THROWS_AS(j.patch(patch), std::invalid_argument); - CHECK_THROWS_WITH(j.patch(patch), "operation 'copy' must have member 'path'"); + CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_WITH(j.patch(patch), + "[json.exception.parse_error.105] parse error: operation 'copy' must have member 'path'"); } SECTION("non-string 'path'") { json j; json patch = {{{"op", "copy"}, {"path", 1}}}; - CHECK_THROWS_AS(j.patch(patch), std::invalid_argument); - CHECK_THROWS_WITH(j.patch(patch), "operation 'copy' must have string member 'path'"); + CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_WITH(j.patch(patch), + "[json.exception.parse_error.105] parse error: operation 'copy' must have string member 'path'"); } SECTION("missing 'from'") { json j; json patch = {{{"op", "copy"}, {"path", ""}}}; - CHECK_THROWS_AS(j.patch(patch), std::invalid_argument); - CHECK_THROWS_WITH(j.patch(patch), "operation 'copy' must have member 'from'"); + CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_WITH(j.patch(patch), + "[json.exception.parse_error.105] parse error: operation 'copy' must have member 'from'"); } SECTION("non-string 'from'") { json j; json patch = {{{"op", "copy"}, {"path", ""}, {"from", 1}}}; - CHECK_THROWS_AS(j.patch(patch), std::invalid_argument); - CHECK_THROWS_WITH(j.patch(patch), "operation 'copy' must have string member 'from'"); + CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_WITH(j.patch(patch), + "[json.exception.parse_error.105] parse error: operation 'copy' must have string member 'from'"); } SECTION("nonexisting from location (array)") @@ -932,24 +953,27 @@ TEST_CASE("JSON patch") { json j; json patch = {{{"op", "test"}}}; - CHECK_THROWS_AS(j.patch(patch), std::invalid_argument); - CHECK_THROWS_WITH(j.patch(patch), "operation 'test' must have member 'path'"); + CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_WITH(j.patch(patch), + "[json.exception.parse_error.105] parse error: operation 'test' must have member 'path'"); } SECTION("non-string 'path'") { json j; json patch = {{{"op", "test"}, {"path", 1}}}; - CHECK_THROWS_AS(j.patch(patch), std::invalid_argument); - CHECK_THROWS_WITH(j.patch(patch), "operation 'test' must have string member 'path'"); + CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_WITH(j.patch(patch), + "[json.exception.parse_error.105] parse error: operation 'test' must have string member 'path'"); } SECTION("missing 'value'") { json j; json patch = {{{"op", "test"}, {"path", ""}}}; - CHECK_THROWS_AS(j.patch(patch), std::invalid_argument); - CHECK_THROWS_WITH(j.patch(patch), "operation 'test' must have member 'value'"); + CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_WITH(j.patch(patch), + "[json.exception.parse_error.105] parse error: operation 'test' must have member 'value'"); } } } From 01d3a006b1dd8b80ee9898de7b5ab2a17484f46e Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Thu, 2 Mar 2017 18:13:19 +0100 Subject: [PATCH 022/530] :arrow_up: updated Catch to v1.8.1 --- test/thirdparty/catch/catch.hpp | 801 ++++++++++++++++++++------------ 1 file changed, 507 insertions(+), 294 deletions(-) diff --git a/test/thirdparty/catch/catch.hpp b/test/thirdparty/catch/catch.hpp index 2a09fd19..6f9334ba 100644 --- a/test/thirdparty/catch/catch.hpp +++ b/test/thirdparty/catch/catch.hpp @@ -1,6 +1,6 @@ /* - * Catch v1.7.2 - * Generated: 2017-02-13 15:57:33.350226 + * Catch v1.8.1 + * Generated: 2017-03-01 16:04:19.016511 * ---------------------------------------------------------- * This file has been merged from multiple headers. Please don't edit it directly * Copyright (c) 2012 Two Blue Cubes Ltd. All rights reserved. @@ -40,6 +40,12 @@ #elif defined __GNUC__ # pragma GCC diagnostic ignored "-Wvariadic-macros" # pragma GCC diagnostic ignored "-Wunused-variable" + + // For newer version we can use __Pragma to disable the warnings locally +# if __GNUC__ == 4 && __GNUC_MINOR__ >= 4 && __GNUC_MINOR__ <= 7 +# pragma GCC diagnostic ignored "-Wparentheses" +# endif + # pragma GCC diagnostic push # pragma GCC diagnostic ignored "-Wpadded" #endif @@ -82,6 +88,7 @@ // CATCH_CONFIG_VARIADIC_MACROS : are variadic macros supported? // CATCH_CONFIG_COUNTER : is the __COUNTER__ macro supported? // CATCH_CONFIG_WINDOWS_SEH : is Windows SEH supported? +// CATCH_CONFIG_POSIX_SIGNALS : are POSIX signals supported? // **************** // Note to maintainers: if new toggles are added please document them // in configuration.md, too @@ -117,11 +124,29 @@ # endif # if defined(CATCH_CPP11_OR_GREATER) -# define CATCH_INTERNAL_SUPPRESS_PARENTHESES_WARNINGS _Pragma( "clang diagnostic ignored \"-Wparentheses\"" ) +# define CATCH_INTERNAL_SUPPRESS_PARENTHESES_WARNINGS \ + _Pragma( "clang diagnostic push" ) \ + _Pragma( "clang diagnostic ignored \"-Wparentheses\"" ) +# define CATCH_INTERNAL_UNSUPPRESS_PARENTHESES_WARNINGS \ + _Pragma( "clang diagnostic pop" ) # endif #endif // __clang__ +//////////////////////////////////////////////////////////////////////////////// +// Cygwin +#ifdef __CYGWIN__ + +# if !defined(CATCH_CONFIG_POSIX_SIGNALS) +# define CATCH_INTERNAL_CONFIG_NO_POSIX_SIGNALS +# endif + +// Required for some versions of Cygwin to declare gettimeofday +// see: http://stackoverflow.com/questions/36901803/gettimeofday-not-declared-in-this-scope-cygwin +# define _BSD_SOURCE + +#endif // __CYGWIN__ + //////////////////////////////////////////////////////////////////////////////// // Borland #ifdef __BORLANDC__ @@ -144,12 +169,20 @@ // GCC #ifdef __GNUC__ +# if __GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 8) +# define CATCH_GCC_HAS_NEW_PRAGMA +# endif + # if __GNUC__ == 4 && __GNUC_MINOR__ >= 6 && defined(__GXX_EXPERIMENTAL_CXX0X__) # define CATCH_INTERNAL_CONFIG_CPP11_NULLPTR # endif -# if !defined(CATCH_INTERNAL_SUPPRESS_PARENTHESES_WARNINGS) && defined(CATCH_CPP11_OR_GREATER) -# define CATCH_INTERNAL_SUPPRESS_PARENTHESES_WARNINGS _Pragma( "GCC diagnostic ignored \"-Wparentheses\"" ) +# if !defined(CATCH_INTERNAL_SUPPRESS_PARENTHESES_WARNINGS) && defined(CATCH_GCC_HAS_NEW_PRAGMA) +# define CATCH_INTERNAL_SUPPRESS_PARENTHESES_WARNINGS \ + _Pragma( "GCC diagnostic push" ) \ + _Pragma( "GCC diagnostic ignored \"-Wparentheses\"" ) +# define CATCH_INTERNAL_UNSUPPRESS_PARENTHESES_WARNINGS \ + _Pragma( "GCC diagnostic pop" ) # endif // - otherwise more recent versions define __cplusplus >= 201103L @@ -290,9 +323,14 @@ #if defined(CATCH_INTERNAL_CONFIG_WINDOWS_SEH) && !defined(CATCH_CONFIG_NO_WINDOWS_SEH) && !defined(CATCH_CONFIG_WINDOWS_SEH) # define CATCH_CONFIG_WINDOWS_SEH #endif +// This is set by default, because we assume that unix compilers are posix-signal-compatible by default. +#if !defined(CATCH_INTERNAL_CONFIG_NO_POSIX_SIGNALS) && !defined(CATCH_CONFIG_NO_POSIX_SIGNALS) && !defined(CATCH_CONFIG_POSIX_SIGNALS) +# define CATCH_CONFIG_POSIX_SIGNALS +#endif #if !defined(CATCH_INTERNAL_SUPPRESS_PARENTHESES_WARNINGS) # define CATCH_INTERNAL_SUPPRESS_PARENTHESES_WARNINGS +# define CATCH_INTERNAL_UNSUPPRESS_PARENTHESES_WARNINGS #endif // noexcept support: @@ -871,6 +909,9 @@ namespace Catch { template STATIC_ASSERT_Expression_Too_Complex_Please_Rewrite_As_Binary_Comparison& operator % ( T const& ); template STATIC_ASSERT_Expression_Too_Complex_Please_Rewrite_As_Binary_Comparison& operator && ( T const& ); template STATIC_ASSERT_Expression_Too_Complex_Please_Rewrite_As_Binary_Comparison& operator || ( T const& ); + + private: + DecomposedExpression& operator = (DecomposedExpression const&); }; struct AssertionInfo @@ -967,313 +1008,153 @@ namespace Catch { namespace Matchers { namespace Impl { - namespace Generic { - template class AllOf; - template class AnyOf; - template class Not; - } + template struct MatchAllOf; + template struct MatchAnyOf; + template struct MatchNotOf; - template - struct Matcher : SharedImpl - { - typedef ExpressionT ExpressionType; - - virtual ~Matcher() {} - virtual Ptr clone() const = 0; - virtual bool match( ExpressionT const& expr ) const = 0; - virtual std::string toString() const = 0; - - Generic::AllOf operator && ( Matcher const& other ) const; - Generic::AnyOf operator || ( Matcher const& other ) const; - Generic::Not operator ! () const; - }; - - template - struct MatcherImpl : Matcher { - - virtual Ptr > clone() const { - return Ptr >( new DerivedT( static_cast( *this ) ) ); - } - }; - - namespace Generic { - template - class Not : public MatcherImpl, ExpressionT> { + class MatcherUntypedBase { public: - explicit Not( Matcher const& matcher ) : m_matcher(matcher.clone()) {} - Not( Not const& other ) : m_matcher( other.m_matcher ) {} - - virtual bool match( ExpressionT const& expr ) const CATCH_OVERRIDE { - return !m_matcher->match( expr ); + std::string toString() const { + if( m_cachedToString.empty() ) + m_cachedToString = describe(); + return m_cachedToString; } - virtual std::string toString() const CATCH_OVERRIDE { - return "not " + m_matcher->toString(); - } + protected: + virtual std::string describe() const = 0; + mutable std::string m_cachedToString; private: - Ptr< Matcher > m_matcher; + MatcherUntypedBase& operator = ( MatcherUntypedBase const& ); }; - template - class AllOf : public MatcherImpl, ExpressionT> { - public: + template + struct MatcherBase : MatcherUntypedBase { - AllOf() {} - AllOf( AllOf const& other ) : m_matchers( other.m_matchers ) {} + virtual bool match( ObjectT const& arg ) const = 0; - AllOf& add( Matcher const& matcher ) { - m_matchers.push_back( matcher.clone() ); - return *this; - } - virtual bool match( ExpressionT const& expr ) const - { - for( std::size_t i = 0; i < m_matchers.size(); ++i ) - if( !m_matchers[i]->match( expr ) ) + MatchAllOf operator && ( MatcherBase const& other ) const; + MatchAnyOf operator || ( MatcherBase const& other ) const; + MatchNotOf operator ! () const; + }; + + template + struct MatchAllOf : MatcherBase { + virtual bool match( ArgT const& arg ) const CATCH_OVERRIDE { + for( std::size_t i = 0; i < m_matchers.size(); ++i ) { + if (!m_matchers[i]->match(arg)) return false; + } return true; } - virtual std::string toString() const { - std::ostringstream oss; - oss << "( "; + virtual std::string describe() const CATCH_OVERRIDE { + std::string description; + description.reserve( 4 + m_matchers.size()*32 ); + description += "( "; for( std::size_t i = 0; i < m_matchers.size(); ++i ) { if( i != 0 ) - oss << " and "; - oss << m_matchers[i]->toString(); + description += " and "; + description += m_matchers[i]->toString(); } - oss << " )"; - return oss.str(); + description += " )"; + return description; } - AllOf operator && ( Matcher const& other ) const { - AllOf allOfExpr( *this ); - allOfExpr.add( other ); - return allOfExpr; - } - - private: - std::vector > > m_matchers; - }; - - template - class AnyOf : public MatcherImpl, ExpressionT> { - public: - - AnyOf() {} - AnyOf( AnyOf const& other ) : m_matchers( other.m_matchers ) {} - - AnyOf& add( Matcher const& matcher ) { - m_matchers.push_back( matcher.clone() ); + MatchAllOf& operator && ( MatcherBase const& other ) { + m_matchers.push_back( &other ); return *this; } - virtual bool match( ExpressionT const& expr ) const - { - for( std::size_t i = 0; i < m_matchers.size(); ++i ) - if( m_matchers[i]->match( expr ) ) + + std::vector const*> m_matchers; + }; + template + struct MatchAnyOf : MatcherBase { + + virtual bool match( ArgT const& arg ) const CATCH_OVERRIDE { + for( std::size_t i = 0; i < m_matchers.size(); ++i ) { + if (m_matchers[i]->match(arg)) return true; + } return false; } - virtual std::string toString() const { - std::ostringstream oss; - oss << "( "; + virtual std::string describe() const CATCH_OVERRIDE { + std::string description; + description.reserve( 4 + m_matchers.size()*32 ); + description += "( "; for( std::size_t i = 0; i < m_matchers.size(); ++i ) { if( i != 0 ) - oss << " or "; - oss << m_matchers[i]->toString(); + description += " or "; + description += m_matchers[i]->toString(); } - oss << " )"; - return oss.str(); + description += " )"; + return description; } - AnyOf operator || ( Matcher const& other ) const { - AnyOf anyOfExpr( *this ); - anyOfExpr.add( other ); - return anyOfExpr; + MatchAnyOf& operator || ( MatcherBase const& other ) { + m_matchers.push_back( &other ); + return *this; } - private: - std::vector > > m_matchers; + std::vector const*> m_matchers; }; - } // namespace Generic + template + struct MatchNotOf : MatcherBase { - template - Generic::AllOf Matcher::operator && ( Matcher const& other ) const { - Generic::AllOf allOfExpr; - allOfExpr.add( *this ); - allOfExpr.add( other ); - return allOfExpr; - } - - template - Generic::AnyOf Matcher::operator || ( Matcher const& other ) const { - Generic::AnyOf anyOfExpr; - anyOfExpr.add( *this ); - anyOfExpr.add( other ); - return anyOfExpr; - } - - template - Generic::Not Matcher::operator ! () const { - return Generic::Not( *this ); - } - - namespace StdString { - - inline std::string makeString( std::string const& str ) { return str; } - inline std::string makeString( const char* str ) { return str ? std::string( str ) : std::string(); } - - struct CasedString - { - CasedString( std::string const& str, CaseSensitive::Choice caseSensitivity ) - : m_caseSensitivity( caseSensitivity ), - m_str( adjustString( str ) ) - {} - std::string adjustString( std::string const& str ) const { - return m_caseSensitivity == CaseSensitive::No - ? toLower( str ) - : str; + MatchNotOf( MatcherBase const& underlyingMatcher ) : m_underlyingMatcher( underlyingMatcher ) {} + virtual bool match( ArgT const& arg ) const CATCH_OVERRIDE { + return !m_underlyingMatcher.match( arg ); } - std::string toStringSuffix() const - { - return m_caseSensitivity == CaseSensitive::No - ? " (case insensitive)" - : std::string(); + + virtual std::string describe() const CATCH_OVERRIDE { + return "not " + m_underlyingMatcher.toString(); } - CaseSensitive::Choice m_caseSensitivity; - std::string m_str; + MatcherBase const& m_underlyingMatcher; }; - struct Equals : MatcherImpl { - Equals( std::string const& str, CaseSensitive::Choice caseSensitivity = CaseSensitive::Yes ) - : m_data( str, caseSensitivity ) - {} - Equals( Equals const& other ) : m_data( other.m_data ){} + template + MatchAllOf MatcherBase::operator && ( MatcherBase const& other ) const { + return MatchAllOf() && *this && other; + } + template + MatchAnyOf MatcherBase::operator || ( MatcherBase const& other ) const { + return MatchAnyOf() || *this || other; + } + template + MatchNotOf MatcherBase::operator ! () const { + return MatchNotOf( *this ); + } - virtual ~Equals(); - - virtual bool match( std::string const& expr ) const { - return m_data.m_str == m_data.adjustString( expr );; - } - virtual std::string toString() const { - return "equals: \"" + m_data.m_str + '"' + m_data.toStringSuffix(); - } - - CasedString m_data; - }; - - struct Contains : MatcherImpl { - Contains( std::string const& substr, CaseSensitive::Choice caseSensitivity = CaseSensitive::Yes ) - : m_data( substr, caseSensitivity ){} - Contains( Contains const& other ) : m_data( other.m_data ){} - - virtual ~Contains(); - - virtual bool match( std::string const& expr ) const { - return m_data.adjustString( expr ).find( m_data.m_str ) != std::string::npos; - } - virtual std::string toString() const { - return "contains: \"" + m_data.m_str + '"' + m_data.toStringSuffix(); - } - - CasedString m_data; - }; - - struct StartsWith : MatcherImpl { - StartsWith( std::string const& substr, CaseSensitive::Choice caseSensitivity = CaseSensitive::Yes ) - : m_data( substr, caseSensitivity ){} - - StartsWith( StartsWith const& other ) : m_data( other.m_data ){} - - virtual ~StartsWith(); - - virtual bool match( std::string const& expr ) const { - return startsWith( m_data.adjustString( expr ), m_data.m_str ); - } - virtual std::string toString() const { - return "starts with: \"" + m_data.m_str + '"' + m_data.toStringSuffix(); - } - - CasedString m_data; - }; - - struct EndsWith : MatcherImpl { - EndsWith( std::string const& substr, CaseSensitive::Choice caseSensitivity = CaseSensitive::Yes ) - : m_data( substr, caseSensitivity ){} - EndsWith( EndsWith const& other ) : m_data( other.m_data ){} - - virtual ~EndsWith(); - - virtual bool match( std::string const& expr ) const { - return endsWith( m_data.adjustString( expr ), m_data.m_str ); - } - virtual std::string toString() const { - return "ends with: \"" + m_data.m_str + '"' + m_data.toStringSuffix(); - } - - CasedString m_data; - }; - } // namespace StdString } // namespace Impl // The following functions create the actual matcher objects. // This allows the types to be inferred - template - inline Impl::Generic::Not Not( Impl::Matcher const& m ) { - return Impl::Generic::Not( m ); + // - deprecated: prefer ||, && and ! + template + inline Impl::MatchNotOf Not( Impl::MatcherBase const& underlyingMatcher ) { + return Impl::MatchNotOf( underlyingMatcher ); } - - template - inline Impl::Generic::AllOf AllOf( Impl::Matcher const& m1, - Impl::Matcher const& m2 ) { - return Impl::Generic::AllOf().add( m1 ).add( m2 ); + template + inline Impl::MatchAllOf AllOf( Impl::MatcherBase const& m1, Impl::MatcherBase const& m2 ) { + return Impl::MatchAllOf() && m1 && m2; } - template - inline Impl::Generic::AllOf AllOf( Impl::Matcher const& m1, - Impl::Matcher const& m2, - Impl::Matcher const& m3 ) { - return Impl::Generic::AllOf().add( m1 ).add( m2 ).add( m3 ); + template + inline Impl::MatchAllOf AllOf( Impl::MatcherBase const& m1, Impl::MatcherBase const& m2, Impl::MatcherBase const& m3 ) { + return Impl::MatchAllOf() && m1 && m2 && m3; } - template - inline Impl::Generic::AnyOf AnyOf( Impl::Matcher const& m1, - Impl::Matcher const& m2 ) { - return Impl::Generic::AnyOf().add( m1 ).add( m2 ); + template + inline Impl::MatchAnyOf AnyOf( Impl::MatcherBase const& m1, Impl::MatcherBase const& m2 ) { + return Impl::MatchAnyOf() || m1 || m2; } - template - inline Impl::Generic::AnyOf AnyOf( Impl::Matcher const& m1, - Impl::Matcher const& m2, - Impl::Matcher const& m3 ) { - return Impl::Generic::AnyOf().add( m1 ).add( m2 ).add( m3 ); - } - - inline Impl::StdString::Equals Equals( std::string const& str, CaseSensitive::Choice caseSensitivity = CaseSensitive::Yes ) { - return Impl::StdString::Equals( str, caseSensitivity ); - } - inline Impl::StdString::Equals Equals( const char* str, CaseSensitive::Choice caseSensitivity = CaseSensitive::Yes ) { - return Impl::StdString::Equals( Impl::StdString::makeString( str ), caseSensitivity ); - } - inline Impl::StdString::Contains Contains( std::string const& substr, CaseSensitive::Choice caseSensitivity = CaseSensitive::Yes ) { - return Impl::StdString::Contains( substr, caseSensitivity ); - } - inline Impl::StdString::Contains Contains( const char* substr, CaseSensitive::Choice caseSensitivity = CaseSensitive::Yes ) { - return Impl::StdString::Contains( Impl::StdString::makeString( substr ), caseSensitivity ); - } - inline Impl::StdString::StartsWith StartsWith( std::string const& substr ) { - return Impl::StdString::StartsWith( substr ); - } - inline Impl::StdString::StartsWith StartsWith( const char* substr ) { - return Impl::StdString::StartsWith( Impl::StdString::makeString( substr ) ); - } - inline Impl::StdString::EndsWith EndsWith( std::string const& substr ) { - return Impl::StdString::EndsWith( substr ); - } - inline Impl::StdString::EndsWith EndsWith( const char* substr ) { - return Impl::StdString::EndsWith( Impl::StdString::makeString( substr ) ); + template + inline Impl::MatchAnyOf AnyOf( Impl::MatcherBase const& m1, Impl::MatcherBase const& m2, Impl::MatcherBase const& m3 ) { + return Impl::MatchAnyOf() || m1 || m2 || m3; } } // namespace Matchers using namespace Matchers; +using Matchers::Impl::MatcherBase; } // namespace Catch @@ -1328,7 +1209,7 @@ namespace Catch { void captureResult( ResultWas::OfType resultType ); void captureExpression(); void captureExpectedException( std::string const& expectedMessage ); - void captureExpectedException( Matchers::Impl::Matcher const& matcher ); + void captureExpectedException( Matchers::Impl::MatcherBase const& matcher ); void handleResult( AssertionResult const& result ); void react(); bool shouldDebugBreak() const; @@ -1358,6 +1239,7 @@ namespace Catch { #ifdef _MSC_VER #pragma warning(push) #pragma warning(disable:4389) // '==' : signed/unsigned mismatch +#pragma warning(disable:4312) // Converting int to T* using reinterpret_cast (issue on x64 platform) #endif #include @@ -1874,6 +1756,8 @@ class ExpressionLhs : public DecomposedExpression { public: ExpressionLhs( ResultBuilder& rb, T lhs ) : m_rb( rb ), m_lhs( lhs ), m_truthy(false) {} + ExpressionLhs& operator = ( const ExpressionLhs& ); + template BinaryExpression operator == ( RhsT const& rhs ) { @@ -1952,6 +1836,8 @@ public: BinaryExpression( ResultBuilder& rb, LhsT lhs, RhsT rhs ) : m_rb( rb ), m_lhs( lhs ), m_rhs( rhs ) {} + BinaryExpression& operator = ( BinaryExpression& ); + void endExpression() const { m_rb .setResultType( Internal::compare( m_lhs, m_rhs ) ) @@ -2246,6 +2132,14 @@ namespace Catch { } +#if defined(CATCH_CONFIG_FAST_COMPILE) +/////////////////////////////////////////////////////////////////////////////// +// We can speedup compilation significantly by breaking into debugger lower in +// the callstack, because then we don't have to expand CATCH_BREAK_INTO_DEBUGGER +// macro in each assertion +#define INTERNAL_CATCH_REACT( resultBuilder ) \ + resultBuilder.react(); +#else /////////////////////////////////////////////////////////////////////////////// // In the event of a failure works out if the debugger needs to be invoked // and/or an exception thrown and takes appropriate action. @@ -2254,6 +2148,7 @@ namespace Catch { #define INTERNAL_CATCH_REACT( resultBuilder ) \ if( resultBuilder.shouldDebugBreak() ) CATCH_BREAK_INTO_DEBUGGER(); \ resultBuilder.react(); +#endif /////////////////////////////////////////////////////////////////////////////// #define INTERNAL_CATCH_TEST( expr, resultDisposition, macroName ) \ @@ -2262,6 +2157,7 @@ namespace Catch { try { \ CATCH_INTERNAL_SUPPRESS_PARENTHESES_WARNINGS \ ( __catchResult <= expr ).endExpression(); \ + CATCH_INTERNAL_UNSUPPRESS_PARENTHESES_WARNINGS \ } \ catch( ... ) { \ __catchResult.useActiveException( resultDisposition ); \ @@ -2825,12 +2721,14 @@ namespace Detail { public: explicit Approx ( double value ) : m_epsilon( std::numeric_limits::epsilon()*100 ), + m_margin( 0.0 ), m_scale( 1.0 ), m_value( value ) {} Approx( Approx const& other ) : m_epsilon( other.m_epsilon ), + m_margin( other.m_margin ), m_scale( other.m_scale ), m_value( other.m_value ) {} @@ -2842,6 +2740,7 @@ namespace Detail { Approx operator()( double value ) { Approx approx( value ); approx.epsilon( m_epsilon ); + approx.margin( m_margin ); approx.scale( m_scale ); return approx; } @@ -2851,7 +2750,11 @@ namespace Detail { friend bool operator == ( const T& lhs, Approx const& rhs ) { // Thanks to Richard Harris for his help refining this formula auto lhs_v = double(lhs); - return std::fabs( lhs_v - rhs.m_value ) < rhs.m_epsilon * (rhs.m_scale + (std::max)( std::fabs(lhs_v), std::fabs(rhs.m_value) ) ); + bool relativeOK = std::fabs(lhs_v - rhs.m_value) < rhs.m_epsilon * (rhs.m_scale + (std::max)(std::fabs(lhs_v), std::fabs(rhs.m_value))); + if (relativeOK) { + return true; + } + return std::fabs(lhs_v - rhs.m_value) < rhs.m_margin; } template ::value>::type> @@ -2895,7 +2798,11 @@ namespace Detail { #else friend bool operator == ( double lhs, Approx const& rhs ) { // Thanks to Richard Harris for his help refining this formula - return std::fabs( lhs - rhs.m_value ) < rhs.m_epsilon * (rhs.m_scale + (std::max)( std::fabs(lhs), std::fabs(rhs.m_value) ) ); + bool relativeOK = std::fabs( lhs - rhs.m_value ) < rhs.m_epsilon * (rhs.m_scale + (std::max)( std::fabs(lhs), std::fabs(rhs.m_value) ) ); + if (relativeOK) { + return true; + } + return std::fabs(lhs - rhs.m_value) < rhs.m_margin; } friend bool operator == ( Approx const& lhs, double rhs ) { @@ -2936,6 +2843,11 @@ namespace Detail { return *this; } + Approx& margin( double newMargin ) { + m_margin = newMargin; + return *this; + } + Approx& scale( double newScale ) { m_scale = newScale; return *this; @@ -2949,6 +2861,7 @@ namespace Detail { private: double m_epsilon; + double m_margin; double m_scale; double m_value; }; @@ -2961,6 +2874,153 @@ inline std::string toString( Detail::Approx const& value ) { } // end namespace Catch +// #included from: internal/catch_matchers_string.h +#define TWOBLUECUBES_CATCH_MATCHERS_STRING_H_INCLUDED + +namespace Catch { +namespace Matchers { + + namespace StdString { + + struct CasedString + { + CasedString( std::string const& str, CaseSensitive::Choice caseSensitivity ); + std::string adjustString( std::string const& str ) const; + std::string caseSensitivitySuffix() const; + + CaseSensitive::Choice m_caseSensitivity; + std::string m_str; + }; + + struct StringMatcherBase : MatcherBase { + StringMatcherBase( std::string operation, CasedString const& comparator ); + virtual std::string describe() const CATCH_OVERRIDE; + + CasedString m_comparator; + std::string m_operation; + }; + + struct EqualsMatcher : StringMatcherBase { + EqualsMatcher( CasedString const& comparator ); + virtual bool match( std::string const& source ) const CATCH_OVERRIDE; + }; + struct ContainsMatcher : StringMatcherBase { + ContainsMatcher( CasedString const& comparator ); + virtual bool match( std::string const& source ) const CATCH_OVERRIDE; + }; + struct StartsWithMatcher : StringMatcherBase { + StartsWithMatcher( CasedString const& comparator ); + virtual bool match( std::string const& source ) const CATCH_OVERRIDE; + }; + struct EndsWithMatcher : StringMatcherBase { + EndsWithMatcher( CasedString const& comparator ); + virtual bool match( std::string const& source ) const CATCH_OVERRIDE; + }; + + } // namespace StdString + + // The following functions create the actual matcher objects. + // This allows the types to be inferred + + StdString::EqualsMatcher Equals( std::string const& str, CaseSensitive::Choice caseSensitivity = CaseSensitive::Yes ); + StdString::ContainsMatcher Contains( std::string const& str, CaseSensitive::Choice caseSensitivity = CaseSensitive::Yes ); + StdString::EndsWithMatcher EndsWith( std::string const& str, CaseSensitive::Choice caseSensitivity = CaseSensitive::Yes ); + StdString::StartsWithMatcher StartsWith( std::string const& str, CaseSensitive::Choice caseSensitivity = CaseSensitive::Yes ); + +} // namespace Matchers +} // namespace Catch + +// #included from: internal/catch_matchers_vector.h +#define TWOBLUECUBES_CATCH_MATCHERS_VECTOR_H_INCLUDED + +namespace Catch { +namespace Matchers { + + namespace Vector { + + template + struct ContainsElementMatcher : MatcherBase, T> { + + ContainsElementMatcher(T const &comparator) : m_comparator( comparator) {} + + bool match(std::vector const &v) const CATCH_OVERRIDE { + return std::find(v.begin(), v.end(), m_comparator) != v.end(); + } + + virtual std::string describe() const CATCH_OVERRIDE { + return "Contains: " + Catch::toString( m_comparator ); + } + + T const& m_comparator; + }; + + template + struct ContainsMatcher : MatcherBase, std::vector > { + + ContainsMatcher(std::vector const &comparator) : m_comparator( comparator ) {} + + bool match(std::vector const &v) const CATCH_OVERRIDE { + // !TBD: see note in EqualsMatcher + if (m_comparator.size() > v.size()) + return false; + for (size_t i = 0; i < m_comparator.size(); ++i) + if (std::find(v.begin(), v.end(), m_comparator[i]) == v.end()) + return false; + return true; + } + virtual std::string describe() const CATCH_OVERRIDE { + return "Contains: " + Catch::toString( m_comparator ); + } + + std::vector const& m_comparator; + }; + + template + struct EqualsMatcher : MatcherBase, std::vector > { + + EqualsMatcher(std::vector const &comparator) : m_comparator( comparator ) {} + + bool match(std::vector const &v) const CATCH_OVERRIDE { + // !TBD: This currently works if all elements can be compared using != + // - a more general approach would be via a compare template that defaults + // to using !=. but could be specialised for, e.g. std::vector etc + // - then just call that directly + if (m_comparator.size() != v.size()) + return false; + for (size_t i = 0; i < v.size(); ++i) + if (m_comparator[i] != v[i]) + return false; + return true; + } + virtual std::string describe() const CATCH_OVERRIDE { + return "Equals: " + Catch::toString( m_comparator ); + } + std::vector const& m_comparator; + }; + + } // namespace Vector + + // The following functions create the actual matcher objects. + // This allows the types to be inferred + + template + Vector::ContainsMatcher Contains( std::vector const& comparator ) { + return Vector::ContainsMatcher( comparator ); + } + + template + Vector::ContainsElementMatcher VectorContains( T const& comparator ) { + return Vector::ContainsElementMatcher( comparator ); + } + + template + Vector::EqualsMatcher Equals( std::vector const& comparator ) { + return Vector::EqualsMatcher( comparator ); + } + +} // namespace Matchers +} // namespace Catch + // #included from: internal/catch_interfaces_tag_alias_registry.h #define TWOBLUECUBES_CATCH_INTERFACES_TAG_ALIAS_REGISTRY_H_INCLUDED @@ -3342,6 +3402,29 @@ return @ desc; \ #endif #ifdef CATCH_IMPL + +// !TBD: Move the leak detector code into a separate header +#ifdef CATCH_CONFIG_WINDOWS_CRTDBG +#include +class LeakDetector { +public: + LeakDetector() { + int flag = _CrtSetDbgFlag(_CRTDBG_REPORT_FLAG); + flag |= _CRTDBG_LEAK_CHECK_DF; + flag |= _CRTDBG_ALLOC_MEM_DF; + _CrtSetDbgFlag(flag); + _CrtSetReportMode(_CRT_WARN, _CRTDBG_MODE_FILE | _CRTDBG_MODE_DEBUG); + _CrtSetReportFile(_CRT_WARN, _CRTDBG_FILE_STDERR); + // Change this to leaking allocation's number to break there + _CrtSetBreakAlloc(-1); + } +}; +#else +class LeakDetector {}; +#endif + +LeakDetector leakDetector; + // #included from: internal/catch_impl.hpp #define TWOBLUECUBES_CATCH_IMPL_HPP_INCLUDED @@ -3601,7 +3684,7 @@ namespace Catch { void addPattern() { std::string token = subString(); for( size_t i = 0; i < m_escapeChars.size(); ++i ) - token = token.substr( 0, m_escapeChars[i]-i ) + token.substr( m_escapeChars[i]+1-i ); + token = token.substr( 0, m_escapeChars[i]-m_start-i ) + token.substr( m_escapeChars[i]-m_start-i+1 ); m_escapeChars.clear(); if( startsWith( token, "exclude:" ) ) { m_exclusion = true; @@ -6259,6 +6342,16 @@ namespace Catch { #else // Not Windows - assumed to be POSIX compatible ////////////////////////// +# if !defined(CATCH_CONFIG_POSIX_SIGNALS) + +namespace Catch { + struct FatalConditionHandler { + void reset() {} + }; +} + +# else // CATCH_CONFIG_POSIX_SIGNALS is defined + #include namespace Catch { @@ -6337,6 +6430,8 @@ namespace Catch { } // namespace Catch +# endif // CATCH_CONFIG_POSIX_SIGNALS + #endif // not Windows #include @@ -8043,7 +8138,7 @@ namespace Catch { return os; } - Version libraryVersion( 1, 7, 2, "", 0 ); + Version libraryVersion( 1, 8, 1, "", 0 ); } @@ -8214,8 +8309,11 @@ namespace Catch #endif #ifdef CATCH_PLATFORM_WINDOWS + #else + #include + #endif namespace Catch { @@ -8768,12 +8866,12 @@ namespace Catch { void ResultBuilder::captureExpectedException( std::string const& expectedMessage ) { if( expectedMessage.empty() ) - captureExpectedException( Matchers::Impl::Generic::AllOf() ); + captureExpectedException( Matchers::Impl::MatchAllOf() ); else captureExpectedException( Matchers::Equals( expectedMessage ) ); } - void ResultBuilder::captureExpectedException( Matchers::Impl::Matcher const& matcher ) { + void ResultBuilder::captureExpectedException( Matchers::Impl::MatcherBase const& matcher ) { assert( !isFalseTest( m_assertionInfo.resultDisposition ) ); AssertionResultData data = m_data; @@ -8807,6 +8905,15 @@ namespace Catch { } void ResultBuilder::react() { +#if defined(CATCH_CONFIG_FAST_COMPILE) + if (m_shouldDebugBreak) { + /////////////////////////////////////////////////////////////////// + // To inspect the state during test, you need to go one level up the callstack + // To go back to the test and change execution, jump over the throw statement + /////////////////////////////////////////////////////////////////// + CATCH_BREAK_INTO_DEBUGGER(); + } +#endif if( m_shouldThrow ) throw Catch::TestFailureException(); } @@ -8935,6 +9042,86 @@ namespace Catch { } // end namespace Catch +// #included from: catch_matchers_string.hpp + +namespace Catch { +namespace Matchers { + + namespace StdString { + + CasedString::CasedString( std::string const& str, CaseSensitive::Choice caseSensitivity ) + : m_caseSensitivity( caseSensitivity ), + m_str( adjustString( str ) ) + {} + std::string CasedString::adjustString( std::string const& str ) const { + return m_caseSensitivity == CaseSensitive::No + ? toLower( str ) + : str; + } + std::string CasedString::caseSensitivitySuffix() const { + return m_caseSensitivity == CaseSensitive::No + ? " (case insensitive)" + : std::string(); + } + + StringMatcherBase::StringMatcherBase( std::string operation, CasedString const& comparator ) + : m_comparator( comparator ), + m_operation( operation ) { + } + + std::string StringMatcherBase::describe() const { + std::string description; + description.reserve(5 + m_operation.size() + m_comparator.m_str.size() + + m_comparator.caseSensitivitySuffix().size()); + description += m_operation; + description += ": \""; + description += m_comparator.m_str; + description += "\""; + description += m_comparator.caseSensitivitySuffix(); + return description; + } + + EqualsMatcher::EqualsMatcher( CasedString const& comparator ) : StringMatcherBase( "equals", comparator ) {} + + bool EqualsMatcher::match( std::string const& source ) const { + return m_comparator.adjustString( source ) == m_comparator.m_str; + } + + ContainsMatcher::ContainsMatcher( CasedString const& comparator ) : StringMatcherBase( "contains", comparator ) {} + + bool ContainsMatcher::match( std::string const& source ) const { + return contains( m_comparator.adjustString( source ), m_comparator.m_str ); + } + + StartsWithMatcher::StartsWithMatcher( CasedString const& comparator ) : StringMatcherBase( "starts with", comparator ) {} + + bool StartsWithMatcher::match( std::string const& source ) const { + return startsWith( m_comparator.adjustString( source ), m_comparator.m_str ); + } + + EndsWithMatcher::EndsWithMatcher( CasedString const& comparator ) : StringMatcherBase( "ends with", comparator ) {} + + bool EndsWithMatcher::match( std::string const& source ) const { + return endsWith( m_comparator.adjustString( source ), m_comparator.m_str ); + } + + } // namespace StdString + + StdString::EqualsMatcher Equals( std::string const& str, CaseSensitive::Choice caseSensitivity ) { + return StdString::EqualsMatcher( StdString::CasedString( str, caseSensitivity) ); + } + StdString::ContainsMatcher Contains( std::string const& str, CaseSensitive::Choice caseSensitivity ) { + return StdString::ContainsMatcher( StdString::CasedString( str, caseSensitivity) ); + } + StdString::EndsWithMatcher EndsWith( std::string const& str, CaseSensitive::Choice caseSensitivity ) { + return StdString::EndsWithMatcher( StdString::CasedString( str, caseSensitivity) ); + } + StdString::StartsWithMatcher StartsWith( std::string const& str, CaseSensitive::Choice caseSensitivity ) { + return StdString::StartsWithMatcher( StdString::CasedString( str, caseSensitivity) ); + } + +} // namespace Matchers +} // namespace Catch // #included from: ../reporters/catch_reporter_multi.hpp #define TWOBLUECUBES_CATCH_REPORTER_MULTI_HPP_INCLUDED @@ -9078,6 +9265,7 @@ Ptr addReporter( Ptr const& existingRepo #define TWOBLUECUBES_CATCH_REPORTER_BASES_HPP_INCLUDED #include +#include namespace Catch { @@ -9677,6 +9865,12 @@ namespace Catch { return std::string(); } + void writeSourceInfo( SourceLineInfo const& sourceInfo ) { + m_xml + .writeAttribute( "filename", sourceInfo.file ) + .writeAttribute( "line", sourceInfo.line ); + } + public: // StreamingReporterBase virtual void noMatchingTestCases( std::string const& s ) CATCH_OVERRIDE { @@ -9706,6 +9900,8 @@ namespace Catch { .writeAttribute( "description", testInfo.description ) .writeAttribute( "tags", testInfo.tagsAsString ); + writeSourceInfo( testInfo.lineInfo ); + if ( m_config->showDurations() == ShowDurations::Always ) m_testCaseTimer.start(); m_xml.ensureTagClosed(); @@ -9717,6 +9913,7 @@ namespace Catch { m_xml.startElement( "Section" ) .writeAttribute( "name", trim( sectionInfo.name ) ) .writeAttribute( "description", sectionInfo.description ); + writeSourceInfo( sectionInfo.lineInfo ); m_xml.ensureTagClosed(); } } @@ -9749,9 +9946,9 @@ namespace Catch { if( assertionResult.hasExpression() ) { m_xml.startElement( "Expression" ) .writeAttribute( "success", assertionResult.succeeded() ) - .writeAttribute( "type", assertionResult.getTestMacroName() ) - .writeAttribute( "filename", assertionResult.getSourceInfo().file ) - .writeAttribute( "line", assertionResult.getSourceInfo().line ); + .writeAttribute( "type", assertionResult.getTestMacroName() ); + + writeSourceInfo( assertionResult.getSourceInfo() ); m_xml.scopedElement( "Original" ) .writeText( assertionResult.getExpression() ); @@ -9762,16 +9959,16 @@ namespace Catch { // And... Print a result applicable to each result type. switch( assertionResult.getResultType() ) { case ResultWas::ThrewException: - m_xml.scopedElement( "Exception" ) - .writeAttribute( "filename", assertionResult.getSourceInfo().file ) - .writeAttribute( "line", assertionResult.getSourceInfo().line ) - .writeText( assertionResult.getMessage() ); + m_xml.startElement( "Exception" ); + writeSourceInfo( assertionResult.getSourceInfo() ); + m_xml.writeText( assertionResult.getMessage() ); + m_xml.endElement(); break; case ResultWas::FatalErrorCondition: - m_xml.scopedElement( "FatalErrorCondition" ) - .writeAttribute( "filename", assertionResult.getSourceInfo().file ) - .writeAttribute( "line", assertionResult.getSourceInfo().line ) - .writeText( assertionResult.getMessage() ); + m_xml.startElement( "FatalErrorCondition" ); + writeSourceInfo( assertionResult.getSourceInfo() ); + m_xml.writeText( assertionResult.getMessage() ); + m_xml.endElement(); break; case ResultWas::Info: m_xml.scopedElement( "Info" ) @@ -9781,8 +9978,10 @@ namespace Catch { // Warning will already have been written break; case ResultWas::ExplicitFailure: - m_xml.scopedElement( "Failure" ) - .writeText( assertionResult.getMessage() ); + m_xml.startElement( "Failure" ); + writeSourceInfo( assertionResult.getSourceInfo() ); + m_xml.writeText( assertionResult.getMessage() ); + m_xml.endElement(); break; default: break; @@ -10095,8 +10294,30 @@ namespace Catch { // #included from: ../reporters/catch_reporter_console.hpp #define TWOBLUECUBES_CATCH_REPORTER_CONSOLE_HPP_INCLUDED +#include +#include + namespace Catch { + namespace { + // Because formatting using c++ streams is stateful, drop down to C is required + // Alternatively we could use stringstream, but its performance is... not good. + std::string getFormattedDuration( double duration ) { + // Max exponent + 1 is required to represent the whole part + // + 1 for decimal point + // + 3 for the 3 decimal places + // + 1 for null terminator + const size_t maxDoubleSize = DBL_MAX_10_EXP + 1 + 1 + 3 + 1; + char buffer[maxDoubleSize]; +#ifdef _MSC_VER + sprintf_s(buffer, "%.3f", duration); +#else + sprintf(buffer, "%.3f", duration); +#endif + return std::string(buffer); + } + } + struct ConsoleReporter : StreamingReporterBase { ConsoleReporter( ReporterConfig const& _config ) : StreamingReporterBase( _config ), @@ -10149,14 +10370,11 @@ namespace Catch { stream << "\nNo assertions in test case"; stream << " '" << _sectionStats.sectionInfo.name << "'\n" << std::endl; } - if( m_headerPrinted ) { - if( m_config->showDurations() == ShowDurations::Always ) - stream << "Completed in " << _sectionStats.durationInSeconds << 's' << std::endl; - m_headerPrinted = false; + if( m_config->showDurations() == ShowDurations::Always ) { + stream << getFormattedDuration(_sectionStats.durationInSeconds) << " s: " << _sectionStats.sectionInfo.name << std::endl; } - else { - if( m_config->showDurations() == ShowDurations::Always ) - stream << _sectionStats.sectionInfo.name << " completed in " << _sectionStats.durationInSeconds << 's' << std::endl; + if( m_headerPrinted ) { + m_headerPrinted = false; } StreamingReporterBase::sectionEnded( _sectionStats ); } @@ -10860,11 +11078,6 @@ namespace Catch { TestSpec::TagPattern::~TagPattern() {} TestSpec::ExcludedPattern::~ExcludedPattern() {} - Matchers::Impl::StdString::Equals::~Equals() {} - Matchers::Impl::StdString::Contains::~Contains() {} - Matchers::Impl::StdString::StartsWith::~StartsWith() {} - Matchers::Impl::StdString::EndsWith::~EndsWith() {} - void Config::dummy() {} namespace TestCaseTracking { @@ -10889,7 +11102,7 @@ namespace Catch { // Standard C/C++ main entry point int main (int argc, char * argv[]) { - int result = Catch::Session().run( argc, argv ); + int result = Catch::Session().run( argc, argv ); return ( result < 0xff ? result : 0xff ); } From 06c788e4fd9beec4ac7788d65918bfdb5c5e5f65 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Fri, 3 Mar 2017 10:01:16 +0100 Subject: [PATCH 023/530] :memo: added missing word --- .github/CONTRIBUTING.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/CONTRIBUTING.md b/.github/CONTRIBUTING.md index 4a621c07..b11f19fd 100644 --- a/.github/CONTRIBUTING.md +++ b/.github/CONTRIBUTING.md @@ -64,7 +64,7 @@ Please understand that I cannot accept pull requests changing only file `src/jso - Specifically, I am aware of compilation problems with **Microsoft Visual Studio** (there even is an [issue label](https://github.com/nlohmann/json/issues?utf8=✓&q=label%3A%22visual+studio%22+) for these kind of bugs). I understand that even in 2016, complete C++11 support isn't there yet. But please also understand that I do not want to drop features or uglify the code just to make Microsoft's sub-standard compiler happy. The past has shown that there are ways to express the functionality such that the code compiles with the most recent MSVC - unfortunately, this is not the main objective of the project. - Please refrain from proposing changes that would **break [JSON](http://json.org) conformance**. If you propose a conformant extension of JSON to be supported by the library, please motivate this extension. - We shall not extend the library to **support comments**. There is quite some [controversy](https://www.reddit.com/r/programming/comments/4v6chu/why_json_doesnt_support_comments_douglas_crockford/) around this topic, and there were quite some [issues](https://github.com/nlohmann/json/issues/376) on this. We believe that JSON is fine without comments. - - We do not preserve the **insertion order of object elements**. The [JSON standard](https://tools.ietf.org/html/rfc7159.html) defines objects as "an unordered collection of zero or more name/value pairs". To this end, this library does not preserve insertion order of name/value pairs. (In fact, keys will be traversed in alphabetical order as `std::map` with `std::less` is used by default.) Note this behavior conforms to the standard, and we shall not it to any other order. If you do want to preserve the insertion order, you can specialize the object type with containers like [`tsl::ordered_map`](https://github.com/Tessil/ordered-map) or [`nlohmann::fifo_map`](https://github.com/nlohmann/fifo_map). + - We do not preserve the **insertion order of object elements**. The [JSON standard](https://tools.ietf.org/html/rfc7159.html) defines objects as "an unordered collection of zero or more name/value pairs". To this end, this library does not preserve insertion order of name/value pairs. (In fact, keys will be traversed in alphabetical order as `std::map` with `std::less` is used by default.) Note this behavior conforms to the standard, and we shall not change it to any other order. If you do want to preserve the insertion order, you can specialize the object type with containers like [`tsl::ordered_map`](https://github.com/Tessil/ordered-map) or [`nlohmann::fifo_map`](https://github.com/nlohmann/fifo_map). - Please do not open pull requests that address **multiple issues**. From b86d2148ef09f3ec005f39b024f8fff50cedf196 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Fri, 3 Mar 2017 11:56:58 +0100 Subject: [PATCH 024/530] :hammer: added user-defined exceptions 106-108 These exceptions occur when JSON pointers are malformed. --- src/json.hpp | 30 +++++++++++------------ src/json.hpp.re2c | 30 +++++++++++------------ test/src/unit-json_patch.cpp | 44 +++++++++++++++++----------------- test/src/unit-json_pointer.cpp | 35 ++++++++++++++++----------- 4 files changed, 73 insertions(+), 66 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 6451d732..e2380dbd 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -1109,9 +1109,9 @@ class basic_json json.exception.[parse_error](@ref parse_error).103 | `"parse error: code points above 0x10FFFF are invalid"` | Unicode supports code points up to 0x10FFFF. Code points above 0x10FFFF are invalid. json.exception.[parse_error](@ref parse_error).104 | `"parse error: JSON patch must be an array of objects"` | [RFC 6902](https://tools.ietf.org/html/rfc6902) requires a JSON Patch document to be a JSON document that represents an array of objects. json.exception.[parse_error](@ref parse_error).105 | `"parse error: operation must have string member 'op'"` | An operation of a JSON Patch document must contain exactly one "op" member, whose value indicates the operation to perform. Its value must be one of "add", "remove", "replace", "move", "copy", or "test"; other values are errors. - json.exception.[parse_error](@ref parse_error).106 | "parse error: array index must not begin with '0'" | An array index in a JSON Pointer ([RFC 6901](https://tools.ietf.org/html/rfc6901)) may be `0` or any number wihtout a leading `0`. - json.exception.[parse_error](@ref parse_error).107 | "parse error: JSON pointer must be empty or begin with '/'" | A JSON Pointer must be a Unicode string containing a sequence of zero or more reference tokens, each prefixed by a `/` character. - json.exception.[parse_error](@ref parse_error).108 | "parse error: escape character '~' must be followed with '0' or '1'" | In a JSON Pointer, only `~0` and `~1` are valid escape sequences. + json.exception.[parse_error](@ref parse_error).106 | `"parse error: array index '01' must not begin with '0'"` | An array index in a JSON Pointer ([RFC 6901](https://tools.ietf.org/html/rfc6901)) may be `0` or any number wihtout a leading `0`. + json.exception.[parse_error](@ref parse_error).107 | `"parse error: JSON pointer must be empty or begin with '/' - was: 'foo'"` | A JSON Pointer must be a Unicode string containing a sequence of zero or more reference tokens, each prefixed by a `/` character. + json.exception.[parse_error](@ref parse_error).108 | `"parse error: escape character '~' must be followed with '0' or '1'"` | In a JSON Pointer, only `~0` and `~1` are valid escape sequences. json.exception.[parse_error](@ref parse_error).109 | "parse error: array index 'one' is not a number" | A JSON Pointer array index must be a number. json.exception.[invalid_iterator](@ref invalid_iterator).201 | "iterators are not compatible" | The iterators passed to constructor @ref basic_json(InputIT first, InputIT last) are not compatible, meaning they do not belong to the same container. Therefore, the range (@a first, @a last) is invalid. json.exception.[invalid_iterator](@ref invalid_iterator).202 | "iterator does not fit current value" | In an erase or insert function, the passed iterator @a pos does not belong to the JSON value for which the function was called. It hence does not define a valid position for the deletion/insertion. @@ -11918,12 +11918,12 @@ basic_json_parser_74: empty string is assumed which references the whole JSON value - @throw std::domain_error if reference token is nonempty and does not + @throw parse_error.107 if reference token is nonempty and does not begin with a slash (`/`); example: `"JSON pointer must be empty or - begin with /"` - @throw std::domain_error if a tilde (`~`) is not followed by `0` - (representing `~`) or `1` (representing `/`); example: `"escape error: - ~ must be followed with 0 or 1"` + begin with / - was: 'foo'"` + @throw parse_error.108 if a tilde (`~`) is not followed by `0` + (representing `~`) or `1` (representing `/`); example: `"escape + character '~' must be followed with '0' or '1'"` @liveexample{The example shows the construction several valid JSON pointers as well as the exceptional behavior.,json_pointer} @@ -12074,7 +12074,7 @@ basic_json_parser_74: @complexity Linear in the length of the JSON pointer. @throw std::out_of_range if the JSON pointer can not be resolved - @throw std::domain_error if an array index begins with '0' + @throw parse_error.106 if an array index begins with '0' @throw std::invalid_argument if an array index was not a number */ reference get_unchecked(pointer ptr) const @@ -12118,7 +12118,7 @@ basic_json_parser_74: // error condition (cf. RFC 6901, Sect. 4) if (reference_token.size() > 1 and reference_token[0] == '0') { - JSON_THROW(std::domain_error("array index must not begin with '0'")); + JSON_THROW(parse_error(106, 0, "array index '" + reference_token + "' must not begin with '0'")); } if (reference_token == "-") @@ -12170,7 +12170,7 @@ basic_json_parser_74: // error condition (cf. RFC 6901, Sect. 4) if (reference_token.size() > 1 and reference_token[0] == '0') { - JSON_THROW(std::domain_error("array index must not begin with '0'")); + JSON_THROW(parse_error(106, 0, "array index '" + reference_token + "' must not begin with '0'")); } // note: at performs range check @@ -12222,7 +12222,7 @@ basic_json_parser_74: // error condition (cf. RFC 6901, Sect. 4) if (reference_token.size() > 1 and reference_token[0] == '0') { - JSON_THROW(std::domain_error("array index must not begin with '0'")); + JSON_THROW(parse_error(106, 0, "array index '" + reference_token + "' must not begin with '0'")); } // use unchecked array access @@ -12266,7 +12266,7 @@ basic_json_parser_74: // error condition (cf. RFC 6901, Sect. 4) if (reference_token.size() > 1 and reference_token[0] == '0') { - JSON_THROW(std::domain_error("array index must not begin with '0'")); + JSON_THROW(parse_error(106, 0, "array index '" + reference_token + "' must not begin with '0'")); } // note: at performs range check @@ -12298,7 +12298,7 @@ basic_json_parser_74: // check if nonempty reference string begins with slash if (reference_string[0] != '/') { - JSON_THROW(std::domain_error("JSON pointer must be empty or begin with '/'")); + JSON_THROW(parse_error(107, 1, "JSON pointer must be empty or begin with '/' - was: '" + reference_string + "'")); } // extract the reference tokens: @@ -12333,7 +12333,7 @@ basic_json_parser_74: (reference_token[pos + 1] != '0' and reference_token[pos + 1] != '1')) { - JSON_THROW(std::domain_error("escape error: '~' must be followed with '0' or '1'")); + JSON_THROW(parse_error(108, 0, "escape character '~' must be followed with '0' or '1'")); } } diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 3ed26c53..d1727df5 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -1109,9 +1109,9 @@ class basic_json json.exception.[parse_error](@ref parse_error).103 | `"parse error: code points above 0x10FFFF are invalid"` | Unicode supports code points up to 0x10FFFF. Code points above 0x10FFFF are invalid. json.exception.[parse_error](@ref parse_error).104 | `"parse error: JSON patch must be an array of objects"` | [RFC 6902](https://tools.ietf.org/html/rfc6902) requires a JSON Patch document to be a JSON document that represents an array of objects. json.exception.[parse_error](@ref parse_error).105 | `"parse error: operation must have string member 'op'"` | An operation of a JSON Patch document must contain exactly one "op" member, whose value indicates the operation to perform. Its value must be one of "add", "remove", "replace", "move", "copy", or "test"; other values are errors. - json.exception.[parse_error](@ref parse_error).106 | "parse error: array index must not begin with '0'" | An array index in a JSON Pointer ([RFC 6901](https://tools.ietf.org/html/rfc6901)) may be `0` or any number wihtout a leading `0`. - json.exception.[parse_error](@ref parse_error).107 | "parse error: JSON pointer must be empty or begin with '/'" | A JSON Pointer must be a Unicode string containing a sequence of zero or more reference tokens, each prefixed by a `/` character. - json.exception.[parse_error](@ref parse_error).108 | "parse error: escape character '~' must be followed with '0' or '1'" | In a JSON Pointer, only `~0` and `~1` are valid escape sequences. + json.exception.[parse_error](@ref parse_error).106 | `"parse error: array index '01' must not begin with '0'"` | An array index in a JSON Pointer ([RFC 6901](https://tools.ietf.org/html/rfc6901)) may be `0` or any number wihtout a leading `0`. + json.exception.[parse_error](@ref parse_error).107 | `"parse error: JSON pointer must be empty or begin with '/' - was: 'foo'"` | A JSON Pointer must be a Unicode string containing a sequence of zero or more reference tokens, each prefixed by a `/` character. + json.exception.[parse_error](@ref parse_error).108 | `"parse error: escape character '~' must be followed with '0' or '1'"` | In a JSON Pointer, only `~0` and `~1` are valid escape sequences. json.exception.[parse_error](@ref parse_error).109 | "parse error: array index 'one' is not a number" | A JSON Pointer array index must be a number. json.exception.[invalid_iterator](@ref invalid_iterator).201 | "iterators are not compatible" | The iterators passed to constructor @ref basic_json(InputIT first, InputIT last) are not compatible, meaning they do not belong to the same container. Therefore, the range (@a first, @a last) is invalid. json.exception.[invalid_iterator](@ref invalid_iterator).202 | "iterator does not fit current value" | In an erase or insert function, the passed iterator @a pos does not belong to the JSON value for which the function was called. It hence does not define a valid position for the deletion/insertion. @@ -10951,12 +10951,12 @@ class basic_json empty string is assumed which references the whole JSON value - @throw std::domain_error if reference token is nonempty and does not + @throw parse_error.107 if reference token is nonempty and does not begin with a slash (`/`); example: `"JSON pointer must be empty or - begin with /"` - @throw std::domain_error if a tilde (`~`) is not followed by `0` - (representing `~`) or `1` (representing `/`); example: `"escape error: - ~ must be followed with 0 or 1"` + begin with / - was: 'foo'"` + @throw parse_error.108 if a tilde (`~`) is not followed by `0` + (representing `~`) or `1` (representing `/`); example: `"escape + character '~' must be followed with '0' or '1'"` @liveexample{The example shows the construction several valid JSON pointers as well as the exceptional behavior.,json_pointer} @@ -11107,7 +11107,7 @@ class basic_json @complexity Linear in the length of the JSON pointer. @throw std::out_of_range if the JSON pointer can not be resolved - @throw std::domain_error if an array index begins with '0' + @throw parse_error.106 if an array index begins with '0' @throw std::invalid_argument if an array index was not a number */ reference get_unchecked(pointer ptr) const @@ -11151,7 +11151,7 @@ class basic_json // error condition (cf. RFC 6901, Sect. 4) if (reference_token.size() > 1 and reference_token[0] == '0') { - JSON_THROW(std::domain_error("array index must not begin with '0'")); + JSON_THROW(parse_error(106, 0, "array index '" + reference_token + "' must not begin with '0'")); } if (reference_token == "-") @@ -11203,7 +11203,7 @@ class basic_json // error condition (cf. RFC 6901, Sect. 4) if (reference_token.size() > 1 and reference_token[0] == '0') { - JSON_THROW(std::domain_error("array index must not begin with '0'")); + JSON_THROW(parse_error(106, 0, "array index '" + reference_token + "' must not begin with '0'")); } // note: at performs range check @@ -11255,7 +11255,7 @@ class basic_json // error condition (cf. RFC 6901, Sect. 4) if (reference_token.size() > 1 and reference_token[0] == '0') { - JSON_THROW(std::domain_error("array index must not begin with '0'")); + JSON_THROW(parse_error(106, 0, "array index '" + reference_token + "' must not begin with '0'")); } // use unchecked array access @@ -11299,7 +11299,7 @@ class basic_json // error condition (cf. RFC 6901, Sect. 4) if (reference_token.size() > 1 and reference_token[0] == '0') { - JSON_THROW(std::domain_error("array index must not begin with '0'")); + JSON_THROW(parse_error(106, 0, "array index '" + reference_token + "' must not begin with '0'")); } // note: at performs range check @@ -11331,7 +11331,7 @@ class basic_json // check if nonempty reference string begins with slash if (reference_string[0] != '/') { - JSON_THROW(std::domain_error("JSON pointer must be empty or begin with '/'")); + JSON_THROW(parse_error(107, 1, "JSON pointer must be empty or begin with '/' - was: '" + reference_string + "'")); } // extract the reference tokens: @@ -11366,7 +11366,7 @@ class basic_json (reference_token[pos + 1] != '0' and reference_token[pos + 1] != '1')) { - JSON_THROW(std::domain_error("escape error: '~' must be followed with '0' or '1'")); + JSON_THROW(parse_error(108, 0, "escape character '~' must be followed with '0' or '1'")); } } diff --git a/test/src/unit-json_patch.cpp b/test/src/unit-json_patch.cpp index 4dd89b75..1d905bb5 100644 --- a/test/src/unit-json_patch.cpp +++ b/test/src/unit-json_patch.cpp @@ -686,7 +686,7 @@ TEST_CASE("JSON patch") json patch = {{{"foo", "bar"}}}; CHECK_THROWS_AS(j.patch(patch), json::parse_error); CHECK_THROWS_WITH(j.patch(patch), - "[json.exception.parse_error.105] parse error: operation must have member 'op'"); + "[json.exception.parse_error.105] parse error: operation must have member 'op'"); } SECTION("non-string 'op'") @@ -695,7 +695,7 @@ TEST_CASE("JSON patch") json patch = {{{"op", 1}}}; CHECK_THROWS_AS(j.patch(patch), json::parse_error); CHECK_THROWS_WITH(j.patch(patch), - "[json.exception.parse_error.105] parse error: operation must have string member 'op'"); + "[json.exception.parse_error.105] parse error: operation must have string member 'op'"); } SECTION("invalid operation") @@ -704,7 +704,7 @@ TEST_CASE("JSON patch") json patch = {{{"op", "foo"}, {"path", ""}}}; CHECK_THROWS_AS(j.patch(patch), json::parse_error); CHECK_THROWS_WITH(j.patch(patch), - "[json.exception.parse_error.105] parse error: operation value 'foo' is invalid"); + "[json.exception.parse_error.105] parse error: operation value 'foo' is invalid"); } } @@ -716,7 +716,7 @@ TEST_CASE("JSON patch") json patch = {{{"op", "add"}}}; CHECK_THROWS_AS(j.patch(patch), json::parse_error); CHECK_THROWS_WITH(j.patch(patch), - "[json.exception.parse_error.105] parse error: operation 'add' must have member 'path'"); + "[json.exception.parse_error.105] parse error: operation 'add' must have member 'path'"); } SECTION("non-string 'path'") @@ -725,7 +725,7 @@ TEST_CASE("JSON patch") json patch = {{{"op", "add"}, {"path", 1}}}; CHECK_THROWS_AS(j.patch(patch), json::parse_error); CHECK_THROWS_WITH(j.patch(patch), - "[json.exception.parse_error.105] parse error: operation 'add' must have string member 'path'"); + "[json.exception.parse_error.105] parse error: operation 'add' must have string member 'path'"); } SECTION("missing 'value'") @@ -734,7 +734,7 @@ TEST_CASE("JSON patch") json patch = {{{"op", "add"}, {"path", ""}}}; CHECK_THROWS_AS(j.patch(patch), json::parse_error); CHECK_THROWS_WITH(j.patch(patch), - "[json.exception.parse_error.105] parse error: operation 'add' must have member 'value'"); + "[json.exception.parse_error.105] parse error: operation 'add' must have member 'value'"); } SECTION("invalid array index") @@ -754,7 +754,7 @@ TEST_CASE("JSON patch") json patch = {{{"op", "remove"}}}; CHECK_THROWS_AS(j.patch(patch), json::parse_error); CHECK_THROWS_WITH(j.patch(patch), - "[json.exception.parse_error.105] parse error: operation 'remove' must have member 'path'"); + "[json.exception.parse_error.105] parse error: operation 'remove' must have member 'path'"); } SECTION("non-string 'path'") @@ -763,7 +763,7 @@ TEST_CASE("JSON patch") json patch = {{{"op", "remove"}, {"path", 1}}}; CHECK_THROWS_AS(j.patch(patch), json::parse_error); CHECK_THROWS_WITH(j.patch(patch), - "[json.exception.parse_error.105] parse error: operation 'remove' must have string member 'path'"); + "[json.exception.parse_error.105] parse error: operation 'remove' must have string member 'path'"); } SECTION("nonexisting target location (array)") @@ -799,7 +799,7 @@ TEST_CASE("JSON patch") json patch = {{{"op", "replace"}}}; CHECK_THROWS_AS(j.patch(patch), json::parse_error); CHECK_THROWS_WITH(j.patch(patch), - "[json.exception.parse_error.105] parse error: operation 'replace' must have member 'path'"); + "[json.exception.parse_error.105] parse error: operation 'replace' must have member 'path'"); } SECTION("non-string 'path'") @@ -808,7 +808,7 @@ TEST_CASE("JSON patch") json patch = {{{"op", "replace"}, {"path", 1}}}; CHECK_THROWS_AS(j.patch(patch), json::parse_error); CHECK_THROWS_WITH(j.patch(patch), - "[json.exception.parse_error.105] parse error: operation 'replace' must have string member 'path'"); + "[json.exception.parse_error.105] parse error: operation 'replace' must have string member 'path'"); } SECTION("missing 'value'") @@ -817,7 +817,7 @@ TEST_CASE("JSON patch") json patch = {{{"op", "replace"}, {"path", ""}}}; CHECK_THROWS_AS(j.patch(patch), json::parse_error); CHECK_THROWS_WITH(j.patch(patch), - "[json.exception.parse_error.105] parse error: operation 'replace' must have member 'value'"); + "[json.exception.parse_error.105] parse error: operation 'replace' must have member 'value'"); } SECTION("nonexisting target location (array)") @@ -845,7 +845,7 @@ TEST_CASE("JSON patch") json patch = {{{"op", "move"}}}; CHECK_THROWS_AS(j.patch(patch), json::parse_error); CHECK_THROWS_WITH(j.patch(patch), - "[json.exception.parse_error.105] parse error: operation 'move' must have member 'path'"); + "[json.exception.parse_error.105] parse error: operation 'move' must have member 'path'"); } SECTION("non-string 'path'") @@ -854,7 +854,7 @@ TEST_CASE("JSON patch") json patch = {{{"op", "move"}, {"path", 1}}}; CHECK_THROWS_AS(j.patch(patch), json::parse_error); CHECK_THROWS_WITH(j.patch(patch), - "[json.exception.parse_error.105] parse error: operation 'move' must have string member 'path'"); + "[json.exception.parse_error.105] parse error: operation 'move' must have string member 'path'"); } SECTION("missing 'from'") @@ -863,7 +863,7 @@ TEST_CASE("JSON patch") json patch = {{{"op", "move"}, {"path", ""}}}; CHECK_THROWS_AS(j.patch(patch), json::parse_error); CHECK_THROWS_WITH(j.patch(patch), - "[json.exception.parse_error.105] parse error: operation 'move' must have member 'from'"); + "[json.exception.parse_error.105] parse error: operation 'move' must have member 'from'"); } SECTION("non-string 'from'") @@ -872,7 +872,7 @@ TEST_CASE("JSON patch") json patch = {{{"op", "move"}, {"path", ""}, {"from", 1}}}; CHECK_THROWS_AS(j.patch(patch), json::parse_error); CHECK_THROWS_WITH(j.patch(patch), - "[json.exception.parse_error.105] parse error: operation 'move' must have string member 'from'"); + "[json.exception.parse_error.105] parse error: operation 'move' must have string member 'from'"); } SECTION("nonexisting from location (array)") @@ -900,7 +900,7 @@ TEST_CASE("JSON patch") json patch = {{{"op", "copy"}}}; CHECK_THROWS_AS(j.patch(patch), json::parse_error); CHECK_THROWS_WITH(j.patch(patch), - "[json.exception.parse_error.105] parse error: operation 'copy' must have member 'path'"); + "[json.exception.parse_error.105] parse error: operation 'copy' must have member 'path'"); } SECTION("non-string 'path'") @@ -909,7 +909,7 @@ TEST_CASE("JSON patch") json patch = {{{"op", "copy"}, {"path", 1}}}; CHECK_THROWS_AS(j.patch(patch), json::parse_error); CHECK_THROWS_WITH(j.patch(patch), - "[json.exception.parse_error.105] parse error: operation 'copy' must have string member 'path'"); + "[json.exception.parse_error.105] parse error: operation 'copy' must have string member 'path'"); } SECTION("missing 'from'") @@ -918,7 +918,7 @@ TEST_CASE("JSON patch") json patch = {{{"op", "copy"}, {"path", ""}}}; CHECK_THROWS_AS(j.patch(patch), json::parse_error); CHECK_THROWS_WITH(j.patch(patch), - "[json.exception.parse_error.105] parse error: operation 'copy' must have member 'from'"); + "[json.exception.parse_error.105] parse error: operation 'copy' must have member 'from'"); } SECTION("non-string 'from'") @@ -927,7 +927,7 @@ TEST_CASE("JSON patch") json patch = {{{"op", "copy"}, {"path", ""}, {"from", 1}}}; CHECK_THROWS_AS(j.patch(patch), json::parse_error); CHECK_THROWS_WITH(j.patch(patch), - "[json.exception.parse_error.105] parse error: operation 'copy' must have string member 'from'"); + "[json.exception.parse_error.105] parse error: operation 'copy' must have string member 'from'"); } SECTION("nonexisting from location (array)") @@ -955,7 +955,7 @@ TEST_CASE("JSON patch") json patch = {{{"op", "test"}}}; CHECK_THROWS_AS(j.patch(patch), json::parse_error); CHECK_THROWS_WITH(j.patch(patch), - "[json.exception.parse_error.105] parse error: operation 'test' must have member 'path'"); + "[json.exception.parse_error.105] parse error: operation 'test' must have member 'path'"); } SECTION("non-string 'path'") @@ -964,7 +964,7 @@ TEST_CASE("JSON patch") json patch = {{{"op", "test"}, {"path", 1}}}; CHECK_THROWS_AS(j.patch(patch), json::parse_error); CHECK_THROWS_WITH(j.patch(patch), - "[json.exception.parse_error.105] parse error: operation 'test' must have string member 'path'"); + "[json.exception.parse_error.105] parse error: operation 'test' must have string member 'path'"); } SECTION("missing 'value'") @@ -973,7 +973,7 @@ TEST_CASE("JSON patch") json patch = {{{"op", "test"}, {"path", ""}}}; CHECK_THROWS_AS(j.patch(patch), json::parse_error); CHECK_THROWS_WITH(j.patch(patch), - "[json.exception.parse_error.105] parse error: operation 'test' must have member 'value'"); + "[json.exception.parse_error.105] parse error: operation 'test' must have member 'value'"); } } } diff --git a/test/src/unit-json_pointer.cpp b/test/src/unit-json_pointer.cpp index 495d7066..917f8c09 100644 --- a/test/src/unit-json_pointer.cpp +++ b/test/src/unit-json_pointer.cpp @@ -36,14 +36,17 @@ TEST_CASE("JSON pointers") { SECTION("errors") { - CHECK_THROWS_AS(json::json_pointer("foo"), std::domain_error); - CHECK_THROWS_WITH(json::json_pointer("foo"), "JSON pointer must be empty or begin with '/'"); + CHECK_THROWS_AS(json::json_pointer("foo"), json::parse_error); + CHECK_THROWS_WITH(json::json_pointer("foo"), + "[json.exception.parse_error.107] parse error at 1: JSON pointer must be empty or begin with '/' - was: 'foo'"); - CHECK_THROWS_AS(json::json_pointer("/~~"), std::domain_error); - CHECK_THROWS_WITH(json::json_pointer("/~~"), "escape error: '~' must be followed with '0' or '1'"); + CHECK_THROWS_AS(json::json_pointer("/~~"), json::parse_error); + CHECK_THROWS_WITH(json::json_pointer("/~~"), + "[json.exception.parse_error.108] parse error: escape character '~' must be followed with '0' or '1'"); - CHECK_THROWS_AS(json::json_pointer("/~"), std::domain_error); - CHECK_THROWS_WITH(json::json_pointer("/~"), "escape error: '~' must be followed with '0' or '1'"); + CHECK_THROWS_AS(json::json_pointer("/~"), json::parse_error); + CHECK_THROWS_WITH(json::json_pointer("/~"), + "[json.exception.parse_error.108] parse error: escape character '~' must be followed with '0' or '1'"); json::json_pointer p; CHECK_THROWS_AS(p.top(), std::domain_error); @@ -245,14 +248,18 @@ TEST_CASE("JSON pointers") CHECK(j == json({1, 13, 3, 33, nullptr, 55})); // error with leading 0 - CHECK_THROWS_AS(j["/01"_json_pointer], std::domain_error); - CHECK_THROWS_WITH(j["/01"_json_pointer], "array index must not begin with '0'"); - CHECK_THROWS_AS(j_const["/01"_json_pointer], std::domain_error); - CHECK_THROWS_WITH(j_const["/01"_json_pointer], "array index must not begin with '0'"); - CHECK_THROWS_AS(j.at("/01"_json_pointer), std::domain_error); - CHECK_THROWS_WITH(j.at("/01"_json_pointer), "array index must not begin with '0'"); - CHECK_THROWS_AS(j_const.at("/01"_json_pointer), std::domain_error); - CHECK_THROWS_WITH(j_const.at("/01"_json_pointer), "array index must not begin with '0'"); + CHECK_THROWS_AS(j["/01"_json_pointer], json::parse_error); + CHECK_THROWS_WITH(j["/01"_json_pointer], + "[json.exception.parse_error.106] parse error: array index '01' must not begin with '0'"); + CHECK_THROWS_AS(j_const["/01"_json_pointer], json::parse_error); + CHECK_THROWS_WITH(j_const["/01"_json_pointer], + "[json.exception.parse_error.106] parse error: array index '01' must not begin with '0'"); + CHECK_THROWS_AS(j.at("/01"_json_pointer), json::parse_error); + CHECK_THROWS_WITH(j.at("/01"_json_pointer), + "[json.exception.parse_error.106] parse error: array index '01' must not begin with '0'"); + CHECK_THROWS_AS(j_const.at("/01"_json_pointer), json::parse_error); + CHECK_THROWS_WITH(j_const.at("/01"_json_pointer), + "[json.exception.parse_error.106] parse error: array index '01' must not begin with '0'"); // error with incorrect numbers CHECK_THROWS_AS(j["/one"_json_pointer] = 1, std::invalid_argument); From 776758d7a26e720731ed2cefcf33e87b8abba158 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Fri, 3 Mar 2017 12:54:37 +0100 Subject: [PATCH 025/530] :hammer: added user-defined exception 109 This exception occurs when a JSON pointer contains an array index that can not be converted into a number. --- src/json.hpp | 61 ++++++++++++++++++++++++++-------- src/json.hpp.re2c | 61 ++++++++++++++++++++++++++-------- test/src/unit-json_pointer.cpp | 4 ++- 3 files changed, 99 insertions(+), 27 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index e2380dbd..284a6400 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -1104,14 +1104,14 @@ class basic_json name / id | example massage | description ------------------------------ | --------------- | ------------------------- - json.exception.[parse_error](@ref parse_error).101 | `"parse error at 2: unexpected end of input; expected string literal"` | This error indicates a syntax error while deserializing a JSON text. The error message describes that an unexpected token (character) was encountered, and the member @ref parse_error::byte indicates the error position. - json.exception.[parse_error](@ref parse_error).102 | `"parse error at 14: missing or wrong low surrogate"` | JSON uses the `\uxxxx` format to describe Unicode characters. Code points above above 0xFFFF are split into two `\uxxxx` entries ("surrogate pairs"). This error indicates that the surrogate pair is incomplete or contains an invalid code point. - json.exception.[parse_error](@ref parse_error).103 | `"parse error: code points above 0x10FFFF are invalid"` | Unicode supports code points up to 0x10FFFF. Code points above 0x10FFFF are invalid. - json.exception.[parse_error](@ref parse_error).104 | `"parse error: JSON patch must be an array of objects"` | [RFC 6902](https://tools.ietf.org/html/rfc6902) requires a JSON Patch document to be a JSON document that represents an array of objects. - json.exception.[parse_error](@ref parse_error).105 | `"parse error: operation must have string member 'op'"` | An operation of a JSON Patch document must contain exactly one "op" member, whose value indicates the operation to perform. Its value must be one of "add", "remove", "replace", "move", "copy", or "test"; other values are errors. - json.exception.[parse_error](@ref parse_error).106 | `"parse error: array index '01' must not begin with '0'"` | An array index in a JSON Pointer ([RFC 6901](https://tools.ietf.org/html/rfc6901)) may be `0` or any number wihtout a leading `0`. - json.exception.[parse_error](@ref parse_error).107 | `"parse error: JSON pointer must be empty or begin with '/' - was: 'foo'"` | A JSON Pointer must be a Unicode string containing a sequence of zero or more reference tokens, each prefixed by a `/` character. - json.exception.[parse_error](@ref parse_error).108 | `"parse error: escape character '~' must be followed with '0' or '1'"` | In a JSON Pointer, only `~0` and `~1` are valid escape sequences. + json.exception.[parse_error](@ref parse_error).101 | "parse error at 2: unexpected end of input; expected string literal" | This error indicates a syntax error while deserializing a JSON text. The error message describes that an unexpected token (character) was encountered, and the member @ref parse_error::byte indicates the error position. + json.exception.[parse_error](@ref parse_error).102 | "parse error at 14: missing or wrong low surrogate" | JSON uses the `\uxxxx` format to describe Unicode characters. Code points above above 0xFFFF are split into two `\uxxxx` entries ("surrogate pairs"). This error indicates that the surrogate pair is incomplete or contains an invalid code point. + json.exception.[parse_error](@ref parse_error).103 | "parse error: code points above 0x10FFFF are invalid" | Unicode supports code points up to 0x10FFFF. Code points above 0x10FFFF are invalid. + json.exception.[parse_error](@ref parse_error).104 | "parse error: JSON patch must be an array of objects" | [RFC 6902](https://tools.ietf.org/html/rfc6902) requires a JSON Patch document to be a JSON document that represents an array of objects. + json.exception.[parse_error](@ref parse_error).105 | "parse error: operation must have string member 'op'" | An operation of a JSON Patch document must contain exactly one "op" member, whose value indicates the operation to perform. Its value must be one of "add", "remove", "replace", "move", "copy", or "test"; other values are errors. + json.exception.[parse_error](@ref parse_error).106 | "parse error: array index '01' must not begin with '0'" | An array index in a JSON Pointer ([RFC 6901](https://tools.ietf.org/html/rfc6901)) may be `0` or any number wihtout a leading `0`. + json.exception.[parse_error](@ref parse_error).107 | "parse error: JSON pointer must be empty or begin with '/' - was: 'foo'" | A JSON Pointer must be a Unicode string containing a sequence of zero or more reference tokens, each prefixed by a `/` character. + json.exception.[parse_error](@ref parse_error).108 | "parse error: escape character '~' must be followed with '0' or '1'" | In a JSON Pointer, only `~0` and `~1` are valid escape sequences. json.exception.[parse_error](@ref parse_error).109 | "parse error: array index 'one' is not a number" | A JSON Pointer array index must be a number. json.exception.[invalid_iterator](@ref invalid_iterator).201 | "iterators are not compatible" | The iterators passed to constructor @ref basic_json(InputIT first, InputIT last) are not compatible, meaning they do not belong to the same container. Therefore, the range (@a first, @a last) is invalid. json.exception.[invalid_iterator](@ref invalid_iterator).202 | "iterator does not fit current value" | In an erase or insert function, the passed iterator @a pos does not belong to the JSON value for which the function was called. It hence does not define a valid position for the deletion/insertion. @@ -12037,7 +12037,14 @@ basic_json_parser_74: case value_t::array: { // create an entry in the array - result = &result->operator[](static_cast(std::stoi(reference_token))); + JSON_TRY + { + result = &result->operator[](static_cast(std::stoi(reference_token))); + } + JSON_CATCH(std::invalid_argument&) + { + JSON_THROW(parse_error(109, 0, "array index '" + reference_token + "' is not a number")); + } break; } @@ -12129,7 +12136,14 @@ basic_json_parser_74: else { // convert array index to number; unchecked access - ptr = &ptr->operator[](static_cast(std::stoi(reference_token))); + JSON_TRY + { + ptr = &ptr->operator[](static_cast(std::stoi(reference_token))); + } + JSON_CATCH(std::invalid_argument&) + { + JSON_THROW(parse_error(109, 0, "array index '" + reference_token + "' is not a number")); + } } break; } @@ -12174,7 +12188,14 @@ basic_json_parser_74: } // note: at performs range check - ptr = &ptr->at(static_cast(std::stoi(reference_token))); + JSON_TRY + { + ptr = &ptr->at(static_cast(std::stoi(reference_token))); + } + JSON_CATCH(std::invalid_argument&) + { + JSON_THROW(parse_error(109, 0, "array index '" + reference_token + "' is not a number")); + } break; } @@ -12226,7 +12247,14 @@ basic_json_parser_74: } // use unchecked array access - ptr = &ptr->operator[](static_cast(std::stoi(reference_token))); + JSON_TRY + { + ptr = &ptr->operator[](static_cast(std::stoi(reference_token))); + } + JSON_CATCH(std::invalid_argument&) + { + JSON_THROW(parse_error(109, 0, "array index '" + reference_token + "' is not a number")); + } break; } @@ -12270,7 +12298,14 @@ basic_json_parser_74: } // note: at performs range check - ptr = &ptr->at(static_cast(std::stoi(reference_token))); + JSON_TRY + { + ptr = &ptr->at(static_cast(std::stoi(reference_token))); + } + JSON_CATCH(std::invalid_argument&) + { + JSON_THROW(parse_error(109, 0, "array index '" + reference_token + "' is not a number")); + } break; } diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index d1727df5..55204136 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -1104,14 +1104,14 @@ class basic_json name / id | example massage | description ------------------------------ | --------------- | ------------------------- - json.exception.[parse_error](@ref parse_error).101 | `"parse error at 2: unexpected end of input; expected string literal"` | This error indicates a syntax error while deserializing a JSON text. The error message describes that an unexpected token (character) was encountered, and the member @ref parse_error::byte indicates the error position. - json.exception.[parse_error](@ref parse_error).102 | `"parse error at 14: missing or wrong low surrogate"` | JSON uses the `\uxxxx` format to describe Unicode characters. Code points above above 0xFFFF are split into two `\uxxxx` entries ("surrogate pairs"). This error indicates that the surrogate pair is incomplete or contains an invalid code point. - json.exception.[parse_error](@ref parse_error).103 | `"parse error: code points above 0x10FFFF are invalid"` | Unicode supports code points up to 0x10FFFF. Code points above 0x10FFFF are invalid. - json.exception.[parse_error](@ref parse_error).104 | `"parse error: JSON patch must be an array of objects"` | [RFC 6902](https://tools.ietf.org/html/rfc6902) requires a JSON Patch document to be a JSON document that represents an array of objects. - json.exception.[parse_error](@ref parse_error).105 | `"parse error: operation must have string member 'op'"` | An operation of a JSON Patch document must contain exactly one "op" member, whose value indicates the operation to perform. Its value must be one of "add", "remove", "replace", "move", "copy", or "test"; other values are errors. - json.exception.[parse_error](@ref parse_error).106 | `"parse error: array index '01' must not begin with '0'"` | An array index in a JSON Pointer ([RFC 6901](https://tools.ietf.org/html/rfc6901)) may be `0` or any number wihtout a leading `0`. - json.exception.[parse_error](@ref parse_error).107 | `"parse error: JSON pointer must be empty or begin with '/' - was: 'foo'"` | A JSON Pointer must be a Unicode string containing a sequence of zero or more reference tokens, each prefixed by a `/` character. - json.exception.[parse_error](@ref parse_error).108 | `"parse error: escape character '~' must be followed with '0' or '1'"` | In a JSON Pointer, only `~0` and `~1` are valid escape sequences. + json.exception.[parse_error](@ref parse_error).101 | "parse error at 2: unexpected end of input; expected string literal" | This error indicates a syntax error while deserializing a JSON text. The error message describes that an unexpected token (character) was encountered, and the member @ref parse_error::byte indicates the error position. + json.exception.[parse_error](@ref parse_error).102 | "parse error at 14: missing or wrong low surrogate" | JSON uses the `\uxxxx` format to describe Unicode characters. Code points above above 0xFFFF are split into two `\uxxxx` entries ("surrogate pairs"). This error indicates that the surrogate pair is incomplete or contains an invalid code point. + json.exception.[parse_error](@ref parse_error).103 | "parse error: code points above 0x10FFFF are invalid" | Unicode supports code points up to 0x10FFFF. Code points above 0x10FFFF are invalid. + json.exception.[parse_error](@ref parse_error).104 | "parse error: JSON patch must be an array of objects" | [RFC 6902](https://tools.ietf.org/html/rfc6902) requires a JSON Patch document to be a JSON document that represents an array of objects. + json.exception.[parse_error](@ref parse_error).105 | "parse error: operation must have string member 'op'" | An operation of a JSON Patch document must contain exactly one "op" member, whose value indicates the operation to perform. Its value must be one of "add", "remove", "replace", "move", "copy", or "test"; other values are errors. + json.exception.[parse_error](@ref parse_error).106 | "parse error: array index '01' must not begin with '0'" | An array index in a JSON Pointer ([RFC 6901](https://tools.ietf.org/html/rfc6901)) may be `0` or any number wihtout a leading `0`. + json.exception.[parse_error](@ref parse_error).107 | "parse error: JSON pointer must be empty or begin with '/' - was: 'foo'" | A JSON Pointer must be a Unicode string containing a sequence of zero or more reference tokens, each prefixed by a `/` character. + json.exception.[parse_error](@ref parse_error).108 | "parse error: escape character '~' must be followed with '0' or '1'" | In a JSON Pointer, only `~0` and `~1` are valid escape sequences. json.exception.[parse_error](@ref parse_error).109 | "parse error: array index 'one' is not a number" | A JSON Pointer array index must be a number. json.exception.[invalid_iterator](@ref invalid_iterator).201 | "iterators are not compatible" | The iterators passed to constructor @ref basic_json(InputIT first, InputIT last) are not compatible, meaning they do not belong to the same container. Therefore, the range (@a first, @a last) is invalid. json.exception.[invalid_iterator](@ref invalid_iterator).202 | "iterator does not fit current value" | In an erase or insert function, the passed iterator @a pos does not belong to the JSON value for which the function was called. It hence does not define a valid position for the deletion/insertion. @@ -11070,7 +11070,14 @@ class basic_json case value_t::array: { // create an entry in the array - result = &result->operator[](static_cast(std::stoi(reference_token))); + JSON_TRY + { + result = &result->operator[](static_cast(std::stoi(reference_token))); + } + JSON_CATCH(std::invalid_argument&) + { + JSON_THROW(parse_error(109, 0, "array index '" + reference_token + "' is not a number")); + } break; } @@ -11162,7 +11169,14 @@ class basic_json else { // convert array index to number; unchecked access - ptr = &ptr->operator[](static_cast(std::stoi(reference_token))); + JSON_TRY + { + ptr = &ptr->operator[](static_cast(std::stoi(reference_token))); + } + JSON_CATCH(std::invalid_argument&) + { + JSON_THROW(parse_error(109, 0, "array index '" + reference_token + "' is not a number")); + } } break; } @@ -11207,7 +11221,14 @@ class basic_json } // note: at performs range check - ptr = &ptr->at(static_cast(std::stoi(reference_token))); + JSON_TRY + { + ptr = &ptr->at(static_cast(std::stoi(reference_token))); + } + JSON_CATCH(std::invalid_argument&) + { + JSON_THROW(parse_error(109, 0, "array index '" + reference_token + "' is not a number")); + } break; } @@ -11259,7 +11280,14 @@ class basic_json } // use unchecked array access - ptr = &ptr->operator[](static_cast(std::stoi(reference_token))); + JSON_TRY + { + ptr = &ptr->operator[](static_cast(std::stoi(reference_token))); + } + JSON_CATCH(std::invalid_argument&) + { + JSON_THROW(parse_error(109, 0, "array index '" + reference_token + "' is not a number")); + } break; } @@ -11303,7 +11331,14 @@ class basic_json } // note: at performs range check - ptr = &ptr->at(static_cast(std::stoi(reference_token))); + JSON_TRY + { + ptr = &ptr->at(static_cast(std::stoi(reference_token))); + } + JSON_CATCH(std::invalid_argument&) + { + JSON_THROW(parse_error(109, 0, "array index '" + reference_token + "' is not a number")); + } break; } diff --git a/test/src/unit-json_pointer.cpp b/test/src/unit-json_pointer.cpp index 917f8c09..3c05a278 100644 --- a/test/src/unit-json_pointer.cpp +++ b/test/src/unit-json_pointer.cpp @@ -262,7 +262,9 @@ TEST_CASE("JSON pointers") "[json.exception.parse_error.106] parse error: array index '01' must not begin with '0'"); // error with incorrect numbers - CHECK_THROWS_AS(j["/one"_json_pointer] = 1, std::invalid_argument); + CHECK_THROWS_AS(j["/one"_json_pointer] = 1, json::parse_error); + CHECK_THROWS_WITH(j["/one"_json_pointer] = 1, + "[json.exception.parse_error.109] parse error: array index 'one' is not a number"); // assign to "-" j["/-"_json_pointer] = 99; From a1354184ceadb72669cecbb83ece091180fb74ab Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Fri, 3 Mar 2017 12:56:54 +0100 Subject: [PATCH 026/530] :memo: fixed typo #481 --- README.md | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/README.md b/README.md index 1ada712c..94120db6 100644 --- a/README.md +++ b/README.md @@ -825,9 +825,10 @@ I deeply appreciate the help of the following people. - [EnricoBilla](https://github.com/EnricoBilla) noted a typo in an example. - [Martin Hořeňovský](https://github.com/horenmar) found a way for a 2x speedup for the compilation time of the test suite. - [ukhegg](https://github.com/ukhegg) found proposed an improvement for the examples section. -- [rswanson-ihi](https://github.com/rswanson-ihi) noted a type in the README. +- [rswanson-ihi](https://github.com/rswanson-ihi) noted a typo in the README. - [Mihai Stan](https://github.com/stanmihai4) fixed a bug in the comparison with `nullptr`s. - [Tushar Maheshwari](https://github.com/tusharpm) added [cotire](https://github.com/sakra/cotire) support to speed up the compilation. +- [TedLyngmo](https://github.com/TedLyngmo) noted a typo in the README. Thanks a lot for helping out! Please [let me know](mailto:mail@nlohmann.me) if I forgot someone. @@ -887,7 +888,7 @@ To compile and run the tests, you need to execute ```sh $ make json_unit -Ctest -$ ./test/json_unit "*"" +$ ./test/json_unit "*" =============================================================================== All tests passed (11202597 assertions in 47 test cases) From 9381f6c4dae9d1a163e5a660646ce0363c886753 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Fri, 3 Mar 2017 13:19:45 +0100 Subject: [PATCH 027/530] :hammer: added user-defined exceptions 201-202 Started implementing exceptions for invalid iterators. --- src/json.hpp | 38 ++++++++++++++++--------------- src/json.hpp.re2c | 38 ++++++++++++++++--------------- test/src/unit-constructor1.cpp | 32 +++++++++++++------------- test/src/unit-element_access1.cpp | 10 ++++---- test/src/unit-element_access2.cpp | 10 ++++---- test/src/unit-modifiers.cpp | 24 +++++++++---------- 6 files changed, 80 insertions(+), 72 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 284a6400..21601fcd 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -2415,8 +2415,9 @@ class basic_json @pre Iterators @a first and @a last must be initialized. **This precondition is enforced with an assertion.** - @throw std::domain_error if iterators are not compatible; that is, do not - belong to the same JSON value; example: `"iterators are not compatible"` + @throw invalid_iterator.201 if iterators are not compatible; that is, do + not belong to the same JSON value; example: `"iterators are not + compatible"` @throw std::out_of_range if iterators are for a primitive type (number, boolean, or string) where an out of range error can be detected easily; example: `"iterators out of range"` @@ -2442,7 +2443,7 @@ class basic_json // make sure iterator fits the current value if (first.m_object != last.m_object) { - JSON_THROW(std::domain_error("iterators are not compatible")); + JSON_THROW(invalid_iterator(201, "iterators are not compatible")); } // copy type from first iterator @@ -4398,8 +4399,9 @@ class basic_json @throw std::domain_error if called on a `null` value; example: `"cannot use erase() with null"` - @throw std::domain_error if called on an iterator which does not belong to - the current JSON value; example: `"iterator does not fit current value"` + @throw invalid_iterator.202 if called on an iterator which does not belong + to the current JSON value; example: `"iterator does not fit current + value"` @throw std::out_of_range if called on a primitive type with invalid iterator (i.e., any iterator which is not `begin()`); example: `"iterator out of range"` @@ -4431,7 +4433,7 @@ class basic_json // make sure iterator fits the current value if (this != pos.m_object) { - JSON_THROW(std::domain_error("iterator does not fit current value")); + JSON_THROW(invalid_iterator(202, "iterator does not fit current value")); } IteratorType result = end(); @@ -5655,8 +5657,8 @@ class basic_json @throw std::domain_error if called on JSON values other than arrays; example: `"cannot use insert() with string"` - @throw std::domain_error if @a pos is not an iterator of *this; example: - `"iterator does not fit current value"` + @throw invalid_iterator.202 if @a pos is not an iterator of *this; + example: `"iterator does not fit current value"` @complexity Constant plus linear in the distance between @a pos and end of the container. @@ -5673,7 +5675,7 @@ class basic_json // check if iterator pos fits to this JSON value if (pos.m_object != this) { - JSON_THROW(std::domain_error("iterator does not fit current value")); + JSON_THROW(invalid_iterator(202, "iterator does not fit current value")); } // insert to array and return iterator @@ -5708,8 +5710,8 @@ class basic_json @throw std::domain_error if called on JSON values other than arrays; example: `"cannot use insert() with string"` - @throw std::domain_error if @a pos is not an iterator of *this; example: - `"iterator does not fit current value"` + @throw invalid_iterator.202 if @a pos is not an iterator of *this; + example: `"iterator does not fit current value"` @complexity Linear in @a cnt plus linear in the distance between @a pos and end of the container. @@ -5726,7 +5728,7 @@ class basic_json // check if iterator pos fits to this JSON value if (pos.m_object != this) { - JSON_THROW(std::domain_error("iterator does not fit current value")); + JSON_THROW(invalid_iterator(202, "iterator does not fit current value")); } // insert to array and return iterator @@ -5750,8 +5752,8 @@ class basic_json @throw std::domain_error if called on JSON values other than arrays; example: `"cannot use insert() with string"` - @throw std::domain_error if @a pos is not an iterator of *this; example: - `"iterator does not fit current value"` + @throw invalid_iterator.202 if @a pos is not an iterator of *this; + example: `"iterator does not fit current value"` @throw std::domain_error if @a first and @a last do not belong to the same JSON value; example: `"iterators do not fit"` @throw std::domain_error if @a first or @a last are iterators into @@ -5779,7 +5781,7 @@ class basic_json // check if iterator pos fits to this JSON value if (pos.m_object != this) { - JSON_THROW(std::domain_error("iterator does not fit current value")); + JSON_THROW(invalid_iterator(202, "iterator does not fit current value")); } // check if range iterators belong to the same JSON object @@ -5813,8 +5815,8 @@ class basic_json @throw std::domain_error if called on JSON values other than arrays; example: `"cannot use insert() with string"` - @throw std::domain_error if @a pos is not an iterator of *this; example: - `"iterator does not fit current value"` + @throw invalid_iterator.202 if @a pos is not an iterator of *this; + example: `"iterator does not fit current value"` @return iterator pointing to the first element inserted, or @a pos if `ilist` is empty @@ -5837,7 +5839,7 @@ class basic_json // check if iterator pos fits to this JSON value if (pos.m_object != this) { - JSON_THROW(std::domain_error("iterator does not fit current value")); + JSON_THROW(invalid_iterator(202, "iterator does not fit current value")); } // insert to array and return iterator diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 55204136..72b073f0 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -2415,8 +2415,9 @@ class basic_json @pre Iterators @a first and @a last must be initialized. **This precondition is enforced with an assertion.** - @throw std::domain_error if iterators are not compatible; that is, do not - belong to the same JSON value; example: `"iterators are not compatible"` + @throw invalid_iterator.201 if iterators are not compatible; that is, do + not belong to the same JSON value; example: `"iterators are not + compatible"` @throw std::out_of_range if iterators are for a primitive type (number, boolean, or string) where an out of range error can be detected easily; example: `"iterators out of range"` @@ -2442,7 +2443,7 @@ class basic_json // make sure iterator fits the current value if (first.m_object != last.m_object) { - JSON_THROW(std::domain_error("iterators are not compatible")); + JSON_THROW(invalid_iterator(201, "iterators are not compatible")); } // copy type from first iterator @@ -4398,8 +4399,9 @@ class basic_json @throw std::domain_error if called on a `null` value; example: `"cannot use erase() with null"` - @throw std::domain_error if called on an iterator which does not belong to - the current JSON value; example: `"iterator does not fit current value"` + @throw invalid_iterator.202 if called on an iterator which does not belong + to the current JSON value; example: `"iterator does not fit current + value"` @throw std::out_of_range if called on a primitive type with invalid iterator (i.e., any iterator which is not `begin()`); example: `"iterator out of range"` @@ -4431,7 +4433,7 @@ class basic_json // make sure iterator fits the current value if (this != pos.m_object) { - JSON_THROW(std::domain_error("iterator does not fit current value")); + JSON_THROW(invalid_iterator(202, "iterator does not fit current value")); } IteratorType result = end(); @@ -5655,8 +5657,8 @@ class basic_json @throw std::domain_error if called on JSON values other than arrays; example: `"cannot use insert() with string"` - @throw std::domain_error if @a pos is not an iterator of *this; example: - `"iterator does not fit current value"` + @throw invalid_iterator.202 if @a pos is not an iterator of *this; + example: `"iterator does not fit current value"` @complexity Constant plus linear in the distance between @a pos and end of the container. @@ -5673,7 +5675,7 @@ class basic_json // check if iterator pos fits to this JSON value if (pos.m_object != this) { - JSON_THROW(std::domain_error("iterator does not fit current value")); + JSON_THROW(invalid_iterator(202, "iterator does not fit current value")); } // insert to array and return iterator @@ -5708,8 +5710,8 @@ class basic_json @throw std::domain_error if called on JSON values other than arrays; example: `"cannot use insert() with string"` - @throw std::domain_error if @a pos is not an iterator of *this; example: - `"iterator does not fit current value"` + @throw invalid_iterator.202 if @a pos is not an iterator of *this; + example: `"iterator does not fit current value"` @complexity Linear in @a cnt plus linear in the distance between @a pos and end of the container. @@ -5726,7 +5728,7 @@ class basic_json // check if iterator pos fits to this JSON value if (pos.m_object != this) { - JSON_THROW(std::domain_error("iterator does not fit current value")); + JSON_THROW(invalid_iterator(202, "iterator does not fit current value")); } // insert to array and return iterator @@ -5750,8 +5752,8 @@ class basic_json @throw std::domain_error if called on JSON values other than arrays; example: `"cannot use insert() with string"` - @throw std::domain_error if @a pos is not an iterator of *this; example: - `"iterator does not fit current value"` + @throw invalid_iterator.202 if @a pos is not an iterator of *this; + example: `"iterator does not fit current value"` @throw std::domain_error if @a first and @a last do not belong to the same JSON value; example: `"iterators do not fit"` @throw std::domain_error if @a first or @a last are iterators into @@ -5779,7 +5781,7 @@ class basic_json // check if iterator pos fits to this JSON value if (pos.m_object != this) { - JSON_THROW(std::domain_error("iterator does not fit current value")); + JSON_THROW(invalid_iterator(202, "iterator does not fit current value")); } // check if range iterators belong to the same JSON object @@ -5813,8 +5815,8 @@ class basic_json @throw std::domain_error if called on JSON values other than arrays; example: `"cannot use insert() with string"` - @throw std::domain_error if @a pos is not an iterator of *this; example: - `"iterator does not fit current value"` + @throw invalid_iterator.202 if @a pos is not an iterator of *this; + example: `"iterator does not fit current value"` @return iterator pointing to the first element inserted, or @a pos if `ilist` is empty @@ -5837,7 +5839,7 @@ class basic_json // check if iterator pos fits to this JSON value if (pos.m_object != this) { - JSON_THROW(std::domain_error("iterator does not fit current value")); + JSON_THROW(invalid_iterator(202, "iterator does not fit current value")); } // insert to array and return iterator diff --git a/test/src/unit-constructor1.cpp b/test/src/unit-constructor1.cpp index 9363f0ba..3179863e 100644 --- a/test/src/unit-constructor1.cpp +++ b/test/src/unit-constructor1.cpp @@ -1017,18 +1017,18 @@ TEST_CASE("constructors") { json jobject = {{"a", "a"}, {"b", 1}, {"c", 17u}, {"d", false}, {"e", true}}; json jobject2 = {{"a", "a"}, {"b", 1}, {"c", 17u}}; - CHECK_THROWS_AS(json(jobject.begin(), jobject2.end()), std::domain_error); - CHECK_THROWS_AS(json(jobject2.begin(), jobject.end()), std::domain_error); - CHECK_THROWS_WITH(json(jobject.begin(), jobject2.end()), "iterators are not compatible"); - CHECK_THROWS_WITH(json(jobject2.begin(), jobject.end()), "iterators are not compatible"); + CHECK_THROWS_AS(json(jobject.begin(), jobject2.end()), json::invalid_iterator); + CHECK_THROWS_AS(json(jobject2.begin(), jobject.end()), json::invalid_iterator); + CHECK_THROWS_WITH(json(jobject.begin(), jobject2.end()), "[json.exception.invalid_iterator.201] iterators are not compatible"); + CHECK_THROWS_WITH(json(jobject2.begin(), jobject.end()), "[json.exception.invalid_iterator.201] iterators are not compatible"); } { json jobject = {{"a", "a"}, {"b", 1}, {"c", 17u}, {"d", false}, {"e", true}}; json jobject2 = {{"a", "a"}, {"b", 1}, {"c", 17u}}; - CHECK_THROWS_AS(json(jobject.cbegin(), jobject2.cend()), std::domain_error); - CHECK_THROWS_AS(json(jobject2.cbegin(), jobject.cend()), std::domain_error); - CHECK_THROWS_WITH(json(jobject.cbegin(), jobject2.cend()), "iterators are not compatible"); - CHECK_THROWS_WITH(json(jobject2.cbegin(), jobject.cend()), "iterators are not compatible"); + CHECK_THROWS_AS(json(jobject.cbegin(), jobject2.cend()), json::invalid_iterator); + CHECK_THROWS_AS(json(jobject2.cbegin(), jobject.cend()), json::invalid_iterator); + CHECK_THROWS_WITH(json(jobject.cbegin(), jobject2.cend()), "[json.exception.invalid_iterator.201] iterators are not compatible"); + CHECK_THROWS_WITH(json(jobject2.cbegin(), jobject.cend()), "[json.exception.invalid_iterator.201] iterators are not compatible"); } } } @@ -1082,18 +1082,18 @@ TEST_CASE("constructors") { json jarray = {1, 2, 3, 4}; json jarray2 = {2, 3, 4, 5}; - CHECK_THROWS_AS(json(jarray.begin(), jarray2.end()), std::domain_error); - CHECK_THROWS_AS(json(jarray2.begin(), jarray.end()), std::domain_error); - CHECK_THROWS_WITH(json(jarray.begin(), jarray2.end()), "iterators are not compatible"); - CHECK_THROWS_WITH(json(jarray2.begin(), jarray.end()), "iterators are not compatible"); + CHECK_THROWS_AS(json(jarray.begin(), jarray2.end()), json::invalid_iterator); + CHECK_THROWS_AS(json(jarray2.begin(), jarray.end()), json::invalid_iterator); + CHECK_THROWS_WITH(json(jarray.begin(), jarray2.end()), "[json.exception.invalid_iterator.201] iterators are not compatible"); + CHECK_THROWS_WITH(json(jarray2.begin(), jarray.end()), "[json.exception.invalid_iterator.201] iterators are not compatible"); } { json jarray = {1, 2, 3, 4}; json jarray2 = {2, 3, 4, 5}; - CHECK_THROWS_AS(json(jarray.cbegin(), jarray2.cend()), std::domain_error); - CHECK_THROWS_AS(json(jarray2.cbegin(), jarray.cend()), std::domain_error); - CHECK_THROWS_WITH(json(jarray.cbegin(), jarray2.cend()), "iterators are not compatible"); - CHECK_THROWS_WITH(json(jarray2.cbegin(), jarray.cend()), "iterators are not compatible"); + CHECK_THROWS_AS(json(jarray.cbegin(), jarray2.cend()), json::invalid_iterator); + CHECK_THROWS_AS(json(jarray2.cbegin(), jarray.cend()), json::invalid_iterator); + CHECK_THROWS_WITH(json(jarray.cbegin(), jarray2.cend()), "[json.exception.invalid_iterator.201] iterators are not compatible"); + CHECK_THROWS_WITH(json(jarray2.cbegin(), jarray.cend()), "[json.exception.invalid_iterator.201] iterators are not compatible"); } } } diff --git a/test/src/unit-element_access1.cpp b/test/src/unit-element_access1.cpp index f0763c5d..d4ec045e 100644 --- a/test/src/unit-element_access1.cpp +++ b/test/src/unit-element_access1.cpp @@ -405,12 +405,13 @@ TEST_CASE("element access 1") { json jarray = {1, 1u, true, nullptr, "string", 42.23, json::object(), {1, 2, 3}}; json jarray2 = {"foo", "bar"}; - CHECK_THROWS_AS(jarray.erase(jarray2.begin()), std::domain_error); + CHECK_THROWS_AS(jarray.erase(jarray2.begin()), json::invalid_iterator); CHECK_THROWS_AS(jarray.erase(jarray.begin(), jarray2.end()), std::domain_error); CHECK_THROWS_AS(jarray.erase(jarray2.begin(), jarray.end()), std::domain_error); CHECK_THROWS_AS(jarray.erase(jarray2.begin(), jarray2.end()), std::domain_error); - CHECK_THROWS_WITH(jarray.erase(jarray2.begin()), "iterator does not fit current value"); + CHECK_THROWS_WITH(jarray.erase(jarray2.begin()), + "[json.exception.invalid_iterator.202] iterator does not fit current value"); CHECK_THROWS_WITH(jarray.erase(jarray.begin(), jarray2.end()), "iterators do not fit current value"); CHECK_THROWS_WITH(jarray.erase(jarray2.begin(), jarray.end()), @@ -421,12 +422,13 @@ TEST_CASE("element access 1") { json jarray = {1, 1u, true, nullptr, "string", 42.23, json::object(), {1, 2, 3}}; json jarray2 = {"foo", "bar"}; - CHECK_THROWS_AS(jarray.erase(jarray2.cbegin()), std::domain_error); + CHECK_THROWS_AS(jarray.erase(jarray2.cbegin()), json::invalid_iterator); CHECK_THROWS_AS(jarray.erase(jarray.cbegin(), jarray2.cend()), std::domain_error); CHECK_THROWS_AS(jarray.erase(jarray2.cbegin(), jarray.cend()), std::domain_error); CHECK_THROWS_AS(jarray.erase(jarray2.cbegin(), jarray2.cend()), std::domain_error); - CHECK_THROWS_WITH(jarray.erase(jarray2.cbegin()), "iterator does not fit current value"); + CHECK_THROWS_WITH(jarray.erase(jarray2.cbegin()), + "[json.exception.invalid_iterator.202] iterator does not fit current value"); CHECK_THROWS_WITH(jarray.erase(jarray.cbegin(), jarray2.cend()), "iterators do not fit current value"); CHECK_THROWS_WITH(jarray.erase(jarray2.cbegin(), jarray.cend()), diff --git a/test/src/unit-element_access2.cpp b/test/src/unit-element_access2.cpp index 8e91e89d..d7e2cadc 100644 --- a/test/src/unit-element_access2.cpp +++ b/test/src/unit-element_access2.cpp @@ -685,11 +685,12 @@ TEST_CASE("element access 2") { json jobject = {{"a", "a"}, {"b", 1}, {"c", 17u}, {"d", false}, {"e", true}}; json jobject2 = {{"a", "a"}, {"b", 1}, {"c", 17u}}; - CHECK_THROWS_AS(jobject.erase(jobject2.begin()), std::domain_error); + CHECK_THROWS_AS(jobject.erase(jobject2.begin()), json::invalid_iterator); CHECK_THROWS_AS(jobject.erase(jobject.begin(), jobject2.end()), std::domain_error); CHECK_THROWS_AS(jobject.erase(jobject2.begin(), jobject.end()), std::domain_error); CHECK_THROWS_AS(jobject.erase(jobject2.begin(), jobject2.end()), std::domain_error); - CHECK_THROWS_WITH(jobject.erase(jobject2.begin()), "iterator does not fit current value"); + CHECK_THROWS_WITH(jobject.erase(jobject2.begin()), + "[json.exception.invalid_iterator.202] iterator does not fit current value"); CHECK_THROWS_WITH(jobject.erase(jobject.begin(), jobject2.end()), "iterators do not fit current value"); CHECK_THROWS_WITH(jobject.erase(jobject2.begin(), jobject.end()), @@ -700,11 +701,12 @@ TEST_CASE("element access 2") { json jobject = {{"a", "a"}, {"b", 1}, {"c", 17u}, {"d", false}, {"e", true}}; json jobject2 = {{"a", "a"}, {"b", 1}, {"c", 17u}}; - CHECK_THROWS_AS(jobject.erase(jobject2.cbegin()), std::domain_error); + CHECK_THROWS_AS(jobject.erase(jobject2.cbegin()), json::invalid_iterator); CHECK_THROWS_AS(jobject.erase(jobject.cbegin(), jobject2.cend()), std::domain_error); CHECK_THROWS_AS(jobject.erase(jobject2.cbegin(), jobject.cend()), std::domain_error); CHECK_THROWS_AS(jobject.erase(jobject2.cbegin(), jobject2.cend()), std::domain_error); - CHECK_THROWS_WITH(jobject.erase(jobject2.cbegin()), "iterator does not fit current value"); + CHECK_THROWS_WITH(jobject.erase(jobject2.cbegin()), + "[json.exception.invalid_iterator.202] iterator does not fit current value"); CHECK_THROWS_WITH(jobject.erase(jobject.cbegin(), jobject2.cend()), "iterators do not fit current value"); CHECK_THROWS_WITH(jobject.erase(jobject2.cbegin(), jobject.cend()), diff --git a/test/src/unit-modifiers.cpp b/test/src/unit-modifiers.cpp index 0edc9a12..00521535 100644 --- a/test/src/unit-modifiers.cpp +++ b/test/src/unit-modifiers.cpp @@ -662,22 +662,22 @@ TEST_CASE("modifiers") // pass iterator to a different array json j_another_array = {1, 2}; json j_yet_another_array = {"first", "second"}; - CHECK_THROWS_AS(j_array.insert(j_another_array.end(), 10), std::domain_error); - CHECK_THROWS_AS(j_array.insert(j_another_array.end(), j_value), std::domain_error); - CHECK_THROWS_AS(j_array.insert(j_another_array.end(), 10, 11), std::domain_error); - CHECK_THROWS_AS(j_array.insert(j_another_array.end(), j_yet_another_array.begin(), - j_yet_another_array.end()), std::domain_error); - CHECK_THROWS_AS(j_array.insert(j_another_array.end(), {1, 2, 3, 4}), std::domain_error); + CHECK_THROWS_AS(j_array.insert(j_another_array.end(), 10), json::invalid_iterator); + CHECK_THROWS_AS(j_array.insert(j_another_array.end(), j_value), json::invalid_iterator); + CHECK_THROWS_AS(j_array.insert(j_another_array.end(), 10, 11), json::invalid_iterator); + CHECK_THROWS_AS(j_array.insert(j_another_array.end(), j_yet_another_array.begin(), j_yet_another_array.end()), json::invalid_iterator); + CHECK_THROWS_AS(j_array.insert(j_another_array.end(), {1, 2, 3, 4}), json::invalid_iterator); - CHECK_THROWS_WITH(j_array.insert(j_another_array.end(), 10), "iterator does not fit current value"); + CHECK_THROWS_WITH(j_array.insert(j_another_array.end(), 10), + "[json.exception.invalid_iterator.202] iterator does not fit current value"); CHECK_THROWS_WITH(j_array.insert(j_another_array.end(), j_value), - "iterator does not fit current value"); + "[json.exception.invalid_iterator.202] iterator does not fit current value"); CHECK_THROWS_WITH(j_array.insert(j_another_array.end(), 10, 11), - "iterator does not fit current value"); - CHECK_THROWS_WITH(j_array.insert(j_another_array.end(), j_yet_another_array.begin(), - j_yet_another_array.end()), "iterator does not fit current value"); + "[json.exception.invalid_iterator.202] iterator does not fit current value"); + CHECK_THROWS_WITH(j_array.insert(j_another_array.end(), j_yet_another_array.begin(), j_yet_another_array.end()), + "[json.exception.invalid_iterator.202] iterator does not fit current value"); CHECK_THROWS_WITH(j_array.insert(j_another_array.end(), {1, 2, 3, 4}), - "iterator does not fit current value"); + "[json.exception.invalid_iterator.202] iterator does not fit current value"); } SECTION("non-array type") From 875b2da95d5a0bbd0af8d16631de071ea61a6e7c Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Fri, 3 Mar 2017 13:38:14 +0100 Subject: [PATCH 028/530] :hammer: added user-defined exceptions 203-204 --- src/json.hpp | 14 ++-- src/json.hpp.re2c | 14 ++-- test/src/unit-constructor1.cpp | 80 +++++++++++------------ test/src/unit-element_access1.cpp | 104 +++++++++++++++--------------- test/src/unit-element_access2.cpp | 24 +++---- 5 files changed, 118 insertions(+), 118 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 21601fcd..89bcc2ff 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -2418,7 +2418,7 @@ class basic_json @throw invalid_iterator.201 if iterators are not compatible; that is, do not belong to the same JSON value; example: `"iterators are not compatible"` - @throw std::out_of_range if iterators are for a primitive type (number, + @throw invalid_iterator.204 if iterators are for a primitive type (number, boolean, or string) where an out of range error can be detected easily; example: `"iterators out of range"` @throw std::bad_alloc if allocation for object, array, or string fails @@ -2460,7 +2460,7 @@ class basic_json { if (not first.m_it.primitive_iterator.is_begin() or not last.m_it.primitive_iterator.is_end()) { - JSON_THROW(std::out_of_range("iterators out of range")); + JSON_THROW(invalid_iterator(204, "iterators out of range")); } break; } @@ -4507,9 +4507,9 @@ class basic_json @throw std::domain_error if called on a `null` value; example: `"cannot use erase() with null"` - @throw std::domain_error if called on iterators which does not belong to - the current JSON value; example: `"iterators do not fit current value"` - @throw std::out_of_range if called on a primitive type with invalid + @throw invalid_iterator.203 if called on iterators which does not belong + to the current JSON value; example: `"iterators do not fit current value"` + @throw invalid_iterator.204 if called on a primitive type with invalid iterators (i.e., if `first != begin()` and `last != end()`); example: `"iterators out of range"` @@ -4540,7 +4540,7 @@ class basic_json // make sure iterator fits the current value if (this != first.m_object or this != last.m_object) { - JSON_THROW(std::domain_error("iterators do not fit current value")); + JSON_THROW(invalid_iterator(203, "iterators do not fit current value")); } IteratorType result = end(); @@ -4555,7 +4555,7 @@ class basic_json { if (not first.m_it.primitive_iterator.is_begin() or not last.m_it.primitive_iterator.is_end()) { - JSON_THROW(std::out_of_range("iterators out of range")); + JSON_THROW(invalid_iterator(204, "iterators out of range")); } if (is_string()) diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 72b073f0..5ad644b6 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -2418,7 +2418,7 @@ class basic_json @throw invalid_iterator.201 if iterators are not compatible; that is, do not belong to the same JSON value; example: `"iterators are not compatible"` - @throw std::out_of_range if iterators are for a primitive type (number, + @throw invalid_iterator.204 if iterators are for a primitive type (number, boolean, or string) where an out of range error can be detected easily; example: `"iterators out of range"` @throw std::bad_alloc if allocation for object, array, or string fails @@ -2460,7 +2460,7 @@ class basic_json { if (not first.m_it.primitive_iterator.is_begin() or not last.m_it.primitive_iterator.is_end()) { - JSON_THROW(std::out_of_range("iterators out of range")); + JSON_THROW(invalid_iterator(204, "iterators out of range")); } break; } @@ -4507,9 +4507,9 @@ class basic_json @throw std::domain_error if called on a `null` value; example: `"cannot use erase() with null"` - @throw std::domain_error if called on iterators which does not belong to - the current JSON value; example: `"iterators do not fit current value"` - @throw std::out_of_range if called on a primitive type with invalid + @throw invalid_iterator.203 if called on iterators which does not belong + to the current JSON value; example: `"iterators do not fit current value"` + @throw invalid_iterator.204 if called on a primitive type with invalid iterators (i.e., if `first != begin()` and `last != end()`); example: `"iterators out of range"` @@ -4540,7 +4540,7 @@ class basic_json // make sure iterator fits the current value if (this != first.m_object or this != last.m_object) { - JSON_THROW(std::domain_error("iterators do not fit current value")); + JSON_THROW(invalid_iterator(203, "iterators do not fit current value")); } IteratorType result = end(); @@ -4555,7 +4555,7 @@ class basic_json { if (not first.m_it.primitive_iterator.is_begin() or not last.m_it.primitive_iterator.is_end()) { - JSON_THROW(std::out_of_range("iterators out of range")); + JSON_THROW(invalid_iterator(204, "iterators out of range")); } if (is_string()) diff --git a/test/src/unit-constructor1.cpp b/test/src/unit-constructor1.cpp index 3179863e..3f7d9bbf 100644 --- a/test/src/unit-constructor1.cpp +++ b/test/src/unit-constructor1.cpp @@ -1193,17 +1193,17 @@ TEST_CASE("constructors") { { json j = "foo"; - CHECK_THROWS_AS(json(j.end(), j.end()), std::out_of_range); - CHECK_THROWS_AS(json(j.begin(), j.begin()), std::out_of_range); - CHECK_THROWS_WITH(json(j.end(), j.end()), "iterators out of range"); - CHECK_THROWS_WITH(json(j.begin(), j.begin()), "iterators out of range"); + CHECK_THROWS_AS(json(j.end(), j.end()), json::invalid_iterator); + CHECK_THROWS_AS(json(j.begin(), j.begin()), json::invalid_iterator); + CHECK_THROWS_WITH(json(j.end(), j.end()), "[json.exception.invalid_iterator.204] iterators out of range"); + CHECK_THROWS_WITH(json(j.begin(), j.begin()), "[json.exception.invalid_iterator.204] iterators out of range"); } { json j = "bar"; - CHECK_THROWS_AS(json(j.cend(), j.cend()), std::out_of_range); - CHECK_THROWS_AS(json(j.cbegin(), j.cbegin()), std::out_of_range); - CHECK_THROWS_WITH(json(j.cend(), j.cend()), "iterators out of range"); - CHECK_THROWS_WITH(json(j.cbegin(), j.cbegin()), "iterators out of range"); + CHECK_THROWS_AS(json(j.cend(), j.cend()), json::invalid_iterator); + CHECK_THROWS_AS(json(j.cbegin(), j.cbegin()), json::invalid_iterator); + CHECK_THROWS_WITH(json(j.cend(), j.cend()), "[json.exception.invalid_iterator.204] iterators out of range"); + CHECK_THROWS_WITH(json(j.cbegin(), j.cbegin()), "[json.exception.invalid_iterator.204] iterators out of range"); } } @@ -1211,17 +1211,17 @@ TEST_CASE("constructors") { { json j = false; - CHECK_THROWS_AS(json(j.end(), j.end()), std::out_of_range); - CHECK_THROWS_AS(json(j.begin(), j.begin()), std::out_of_range); - CHECK_THROWS_WITH(json(j.end(), j.end()), "iterators out of range"); - CHECK_THROWS_WITH(json(j.begin(), j.begin()), "iterators out of range"); + CHECK_THROWS_AS(json(j.end(), j.end()), json::invalid_iterator); + CHECK_THROWS_AS(json(j.begin(), j.begin()), json::invalid_iterator); + CHECK_THROWS_WITH(json(j.end(), j.end()), "[json.exception.invalid_iterator.204] iterators out of range"); + CHECK_THROWS_WITH(json(j.begin(), j.begin()), "[json.exception.invalid_iterator.204] iterators out of range"); } { json j = true; - CHECK_THROWS_AS(json(j.cend(), j.cend()), std::out_of_range); - CHECK_THROWS_AS(json(j.cbegin(), j.cbegin()), std::out_of_range); - CHECK_THROWS_WITH(json(j.cend(), j.cend()), "iterators out of range"); - CHECK_THROWS_WITH(json(j.cbegin(), j.cbegin()), "iterators out of range"); + CHECK_THROWS_AS(json(j.cend(), j.cend()), json::invalid_iterator); + CHECK_THROWS_AS(json(j.cbegin(), j.cbegin()), json::invalid_iterator); + CHECK_THROWS_WITH(json(j.cend(), j.cend()), "[json.exception.invalid_iterator.204] iterators out of range"); + CHECK_THROWS_WITH(json(j.cbegin(), j.cbegin()), "[json.exception.invalid_iterator.204] iterators out of range"); } } @@ -1229,17 +1229,17 @@ TEST_CASE("constructors") { { json j = 17; - CHECK_THROWS_AS(json(j.end(), j.end()), std::out_of_range); - CHECK_THROWS_AS(json(j.begin(), j.begin()), std::out_of_range); - CHECK_THROWS_WITH(json(j.end(), j.end()), "iterators out of range"); - CHECK_THROWS_WITH(json(j.begin(), j.begin()), "iterators out of range"); + CHECK_THROWS_AS(json(j.end(), j.end()), json::invalid_iterator); + CHECK_THROWS_AS(json(j.begin(), j.begin()), json::invalid_iterator); + CHECK_THROWS_WITH(json(j.end(), j.end()), "[json.exception.invalid_iterator.204] iterators out of range"); + CHECK_THROWS_WITH(json(j.begin(), j.begin()), "[json.exception.invalid_iterator.204] iterators out of range"); } { json j = 17; - CHECK_THROWS_AS(json(j.cend(), j.cend()), std::out_of_range); - CHECK_THROWS_AS(json(j.cbegin(), j.cbegin()), std::out_of_range); - CHECK_THROWS_WITH(json(j.cend(), j.cend()), "iterators out of range"); - CHECK_THROWS_WITH(json(j.cbegin(), j.cbegin()), "iterators out of range"); + CHECK_THROWS_AS(json(j.cend(), j.cend()), json::invalid_iterator); + CHECK_THROWS_AS(json(j.cbegin(), j.cbegin()), json::invalid_iterator); + CHECK_THROWS_WITH(json(j.cend(), j.cend()), "[json.exception.invalid_iterator.204] iterators out of range"); + CHECK_THROWS_WITH(json(j.cbegin(), j.cbegin()), "[json.exception.invalid_iterator.204] iterators out of range"); } } @@ -1247,17 +1247,17 @@ TEST_CASE("constructors") { { json j = 17u; - CHECK_THROWS_AS(json(j.end(), j.end()), std::out_of_range); - CHECK_THROWS_AS(json(j.begin(), j.begin()), std::out_of_range); - CHECK_THROWS_WITH(json(j.end(), j.end()), "iterators out of range"); - CHECK_THROWS_WITH(json(j.begin(), j.begin()), "iterators out of range"); + CHECK_THROWS_AS(json(j.end(), j.end()), json::invalid_iterator); + CHECK_THROWS_AS(json(j.begin(), j.begin()), json::invalid_iterator); + CHECK_THROWS_WITH(json(j.end(), j.end()), "[json.exception.invalid_iterator.204] iterators out of range"); + CHECK_THROWS_WITH(json(j.begin(), j.begin()), "[json.exception.invalid_iterator.204] iterators out of range"); } { json j = 17u; - CHECK_THROWS_AS(json(j.cend(), j.cend()), std::out_of_range); - CHECK_THROWS_AS(json(j.cbegin(), j.cbegin()), std::out_of_range); - CHECK_THROWS_WITH(json(j.cend(), j.cend()), "iterators out of range"); - CHECK_THROWS_WITH(json(j.cbegin(), j.cbegin()), "iterators out of range"); + CHECK_THROWS_AS(json(j.cend(), j.cend()), json::invalid_iterator); + CHECK_THROWS_AS(json(j.cbegin(), j.cbegin()), json::invalid_iterator); + CHECK_THROWS_WITH(json(j.cend(), j.cend()), "[json.exception.invalid_iterator.204] iterators out of range"); + CHECK_THROWS_WITH(json(j.cbegin(), j.cbegin()), "[json.exception.invalid_iterator.204] iterators out of range"); } } @@ -1265,17 +1265,17 @@ TEST_CASE("constructors") { { json j = 23.42; - CHECK_THROWS_AS(json(j.end(), j.end()), std::out_of_range); - CHECK_THROWS_AS(json(j.begin(), j.begin()), std::out_of_range); - CHECK_THROWS_WITH(json(j.end(), j.end()), "iterators out of range"); - CHECK_THROWS_WITH(json(j.begin(), j.begin()), "iterators out of range"); + CHECK_THROWS_AS(json(j.end(), j.end()), json::invalid_iterator); + CHECK_THROWS_AS(json(j.begin(), j.begin()), json::invalid_iterator); + CHECK_THROWS_WITH(json(j.end(), j.end()), "[json.exception.invalid_iterator.204] iterators out of range"); + CHECK_THROWS_WITH(json(j.begin(), j.begin()), "[json.exception.invalid_iterator.204] iterators out of range"); } { json j = 23.42; - CHECK_THROWS_AS(json(j.cend(), j.cend()), std::out_of_range); - CHECK_THROWS_AS(json(j.cbegin(), j.cbegin()), std::out_of_range); - CHECK_THROWS_WITH(json(j.cend(), j.cend()), "iterators out of range"); - CHECK_THROWS_WITH(json(j.cbegin(), j.cbegin()), "iterators out of range"); + CHECK_THROWS_AS(json(j.cend(), j.cend()), json::invalid_iterator); + CHECK_THROWS_AS(json(j.cbegin(), j.cbegin()), json::invalid_iterator); + CHECK_THROWS_WITH(json(j.cend(), j.cend()), "[json.exception.invalid_iterator.204] iterators out of range"); + CHECK_THROWS_WITH(json(j.cbegin(), j.cbegin()), "[json.exception.invalid_iterator.204] iterators out of range"); } } } diff --git a/test/src/unit-element_access1.cpp b/test/src/unit-element_access1.cpp index d4ec045e..c84afd38 100644 --- a/test/src/unit-element_access1.cpp +++ b/test/src/unit-element_access1.cpp @@ -406,35 +406,35 @@ TEST_CASE("element access 1") json jarray = {1, 1u, true, nullptr, "string", 42.23, json::object(), {1, 2, 3}}; json jarray2 = {"foo", "bar"}; CHECK_THROWS_AS(jarray.erase(jarray2.begin()), json::invalid_iterator); - CHECK_THROWS_AS(jarray.erase(jarray.begin(), jarray2.end()), std::domain_error); - CHECK_THROWS_AS(jarray.erase(jarray2.begin(), jarray.end()), std::domain_error); - CHECK_THROWS_AS(jarray.erase(jarray2.begin(), jarray2.end()), std::domain_error); + CHECK_THROWS_AS(jarray.erase(jarray.begin(), jarray2.end()), json::invalid_iterator); + CHECK_THROWS_AS(jarray.erase(jarray2.begin(), jarray.end()), json::invalid_iterator); + CHECK_THROWS_AS(jarray.erase(jarray2.begin(), jarray2.end()), json::invalid_iterator); CHECK_THROWS_WITH(jarray.erase(jarray2.begin()), "[json.exception.invalid_iterator.202] iterator does not fit current value"); CHECK_THROWS_WITH(jarray.erase(jarray.begin(), jarray2.end()), - "iterators do not fit current value"); + "[json.exception.invalid_iterator.203] iterators do not fit current value"); CHECK_THROWS_WITH(jarray.erase(jarray2.begin(), jarray.end()), - "iterators do not fit current value"); + "[json.exception.invalid_iterator.203] iterators do not fit current value"); CHECK_THROWS_WITH(jarray.erase(jarray2.begin(), jarray2.end()), - "iterators do not fit current value"); + "[json.exception.invalid_iterator.203] iterators do not fit current value"); } { json jarray = {1, 1u, true, nullptr, "string", 42.23, json::object(), {1, 2, 3}}; json jarray2 = {"foo", "bar"}; CHECK_THROWS_AS(jarray.erase(jarray2.cbegin()), json::invalid_iterator); - CHECK_THROWS_AS(jarray.erase(jarray.cbegin(), jarray2.cend()), std::domain_error); - CHECK_THROWS_AS(jarray.erase(jarray2.cbegin(), jarray.cend()), std::domain_error); - CHECK_THROWS_AS(jarray.erase(jarray2.cbegin(), jarray2.cend()), std::domain_error); + CHECK_THROWS_AS(jarray.erase(jarray.cbegin(), jarray2.cend()), json::invalid_iterator); + CHECK_THROWS_AS(jarray.erase(jarray2.cbegin(), jarray.cend()), json::invalid_iterator); + CHECK_THROWS_AS(jarray.erase(jarray2.cbegin(), jarray2.cend()), json::invalid_iterator); CHECK_THROWS_WITH(jarray.erase(jarray2.cbegin()), "[json.exception.invalid_iterator.202] iterator does not fit current value"); CHECK_THROWS_WITH(jarray.erase(jarray.cbegin(), jarray2.cend()), - "iterators do not fit current value"); + "[json.exception.invalid_iterator.203] iterators do not fit current value"); CHECK_THROWS_WITH(jarray.erase(jarray2.cbegin(), jarray.cend()), - "iterators do not fit current value"); + "[json.exception.invalid_iterator.203] iterators do not fit current value"); CHECK_THROWS_WITH(jarray.erase(jarray2.cbegin(), jarray2.cend()), - "iterators do not fit current value"); + "[json.exception.invalid_iterator.203] iterators do not fit current value"); } } } @@ -859,17 +859,17 @@ TEST_CASE("element access 1") { { json j = "foo"; - CHECK_THROWS_AS(j.erase(j.end(), j.end()), std::out_of_range); - CHECK_THROWS_AS(j.erase(j.begin(), j.begin()), std::out_of_range); - CHECK_THROWS_WITH(j.erase(j.end(), j.end()), "iterators out of range"); - CHECK_THROWS_WITH(j.erase(j.begin(), j.begin()), "iterators out of range"); + CHECK_THROWS_AS(j.erase(j.end(), j.end()), json::invalid_iterator); + CHECK_THROWS_AS(j.erase(j.begin(), j.begin()), json::invalid_iterator); + CHECK_THROWS_WITH(j.erase(j.end(), j.end()), "[json.exception.invalid_iterator.204] iterators out of range"); + CHECK_THROWS_WITH(j.erase(j.begin(), j.begin()), "[json.exception.invalid_iterator.204] iterators out of range"); } { json j = "bar"; - CHECK_THROWS_AS(j.erase(j.cend(), j.cend()), std::out_of_range); - CHECK_THROWS_AS(j.erase(j.cbegin(), j.cbegin()), std::out_of_range); - CHECK_THROWS_WITH(j.erase(j.cend(), j.cend()), "iterators out of range"); - CHECK_THROWS_WITH(j.erase(j.cbegin(), j.cbegin()), "iterators out of range"); + CHECK_THROWS_AS(j.erase(j.cend(), j.cend()), json::invalid_iterator); + CHECK_THROWS_AS(j.erase(j.cbegin(), j.cbegin()), json::invalid_iterator); + CHECK_THROWS_WITH(j.erase(j.cend(), j.cend()), "[json.exception.invalid_iterator.204] iterators out of range"); + CHECK_THROWS_WITH(j.erase(j.cbegin(), j.cbegin()), "[json.exception.invalid_iterator.204] iterators out of range"); } } @@ -877,17 +877,17 @@ TEST_CASE("element access 1") { { json j = false; - CHECK_THROWS_AS(j.erase(j.end(), j.end()), std::out_of_range); - CHECK_THROWS_AS(j.erase(j.begin(), j.begin()), std::out_of_range); - CHECK_THROWS_WITH(j.erase(j.end(), j.end()), "iterators out of range"); - CHECK_THROWS_WITH(j.erase(j.begin(), j.begin()), "iterators out of range"); + CHECK_THROWS_AS(j.erase(j.end(), j.end()), json::invalid_iterator); + CHECK_THROWS_AS(j.erase(j.begin(), j.begin()), json::invalid_iterator); + CHECK_THROWS_WITH(j.erase(j.end(), j.end()), "[json.exception.invalid_iterator.204] iterators out of range"); + CHECK_THROWS_WITH(j.erase(j.begin(), j.begin()), "[json.exception.invalid_iterator.204] iterators out of range"); } { json j = true; - CHECK_THROWS_AS(j.erase(j.cend(), j.cend()), std::out_of_range); - CHECK_THROWS_AS(j.erase(j.cbegin(), j.cbegin()), std::out_of_range); - CHECK_THROWS_WITH(j.erase(j.cend(), j.cend()), "iterators out of range"); - CHECK_THROWS_WITH(j.erase(j.cbegin(), j.cbegin()), "iterators out of range"); + CHECK_THROWS_AS(j.erase(j.cend(), j.cend()), json::invalid_iterator); + CHECK_THROWS_AS(j.erase(j.cbegin(), j.cbegin()), json::invalid_iterator); + CHECK_THROWS_WITH(j.erase(j.cend(), j.cend()), "[json.exception.invalid_iterator.204] iterators out of range"); + CHECK_THROWS_WITH(j.erase(j.cbegin(), j.cbegin()), "[json.exception.invalid_iterator.204] iterators out of range"); } } @@ -895,17 +895,17 @@ TEST_CASE("element access 1") { { json j = 17; - CHECK_THROWS_AS(j.erase(j.end(), j.end()), std::out_of_range); - CHECK_THROWS_AS(j.erase(j.begin(), j.begin()), std::out_of_range); - CHECK_THROWS_WITH(j.erase(j.end(), j.end()), "iterators out of range"); - CHECK_THROWS_WITH(j.erase(j.begin(), j.begin()), "iterators out of range"); + CHECK_THROWS_AS(j.erase(j.end(), j.end()), json::invalid_iterator); + CHECK_THROWS_AS(j.erase(j.begin(), j.begin()), json::invalid_iterator); + CHECK_THROWS_WITH(j.erase(j.end(), j.end()), "[json.exception.invalid_iterator.204] iterators out of range"); + CHECK_THROWS_WITH(j.erase(j.begin(), j.begin()), "[json.exception.invalid_iterator.204] iterators out of range"); } { json j = 17; - CHECK_THROWS_AS(j.erase(j.cend(), j.cend()), std::out_of_range); - CHECK_THROWS_AS(j.erase(j.cbegin(), j.cbegin()), std::out_of_range); - CHECK_THROWS_WITH(j.erase(j.cend(), j.cend()), "iterators out of range"); - CHECK_THROWS_WITH(j.erase(j.cbegin(), j.cbegin()), "iterators out of range"); + CHECK_THROWS_AS(j.erase(j.cend(), j.cend()), json::invalid_iterator); + CHECK_THROWS_AS(j.erase(j.cbegin(), j.cbegin()), json::invalid_iterator); + CHECK_THROWS_WITH(j.erase(j.cend(), j.cend()), "[json.exception.invalid_iterator.204] iterators out of range"); + CHECK_THROWS_WITH(j.erase(j.cbegin(), j.cbegin()), "[json.exception.invalid_iterator.204] iterators out of range"); } } @@ -913,17 +913,17 @@ TEST_CASE("element access 1") { { json j = 17u; - CHECK_THROWS_AS(j.erase(j.end(), j.end()), std::out_of_range); - CHECK_THROWS_AS(j.erase(j.begin(), j.begin()), std::out_of_range); - CHECK_THROWS_WITH(j.erase(j.end(), j.end()), "iterators out of range"); - CHECK_THROWS_WITH(j.erase(j.begin(), j.begin()), "iterators out of range"); + CHECK_THROWS_AS(j.erase(j.end(), j.end()), json::invalid_iterator); + CHECK_THROWS_AS(j.erase(j.begin(), j.begin()), json::invalid_iterator); + CHECK_THROWS_WITH(j.erase(j.end(), j.end()), "[json.exception.invalid_iterator.204] iterators out of range"); + CHECK_THROWS_WITH(j.erase(j.begin(), j.begin()), "[json.exception.invalid_iterator.204] iterators out of range"); } { json j = 17u; - CHECK_THROWS_AS(j.erase(j.cend(), j.cend()), std::out_of_range); - CHECK_THROWS_AS(j.erase(j.cbegin(), j.cbegin()), std::out_of_range); - CHECK_THROWS_WITH(j.erase(j.cend(), j.cend()), "iterators out of range"); - CHECK_THROWS_WITH(j.erase(j.cbegin(), j.cbegin()), "iterators out of range"); + CHECK_THROWS_AS(j.erase(j.cend(), j.cend()), json::invalid_iterator); + CHECK_THROWS_AS(j.erase(j.cbegin(), j.cbegin()), json::invalid_iterator); + CHECK_THROWS_WITH(j.erase(j.cend(), j.cend()), "[json.exception.invalid_iterator.204] iterators out of range"); + CHECK_THROWS_WITH(j.erase(j.cbegin(), j.cbegin()), "[json.exception.invalid_iterator.204] iterators out of range"); } } @@ -931,17 +931,17 @@ TEST_CASE("element access 1") { { json j = 23.42; - CHECK_THROWS_AS(j.erase(j.end(), j.end()), std::out_of_range); - CHECK_THROWS_AS(j.erase(j.begin(), j.begin()), std::out_of_range); - CHECK_THROWS_WITH(j.erase(j.end(), j.end()), "iterators out of range"); - CHECK_THROWS_WITH(j.erase(j.begin(), j.begin()), "iterators out of range"); + CHECK_THROWS_AS(j.erase(j.end(), j.end()), json::invalid_iterator); + CHECK_THROWS_AS(j.erase(j.begin(), j.begin()), json::invalid_iterator); + CHECK_THROWS_WITH(j.erase(j.end(), j.end()), "[json.exception.invalid_iterator.204] iterators out of range"); + CHECK_THROWS_WITH(j.erase(j.begin(), j.begin()), "[json.exception.invalid_iterator.204] iterators out of range"); } { json j = 23.42; - CHECK_THROWS_AS(j.erase(j.cend(), j.cend()), std::out_of_range); - CHECK_THROWS_AS(j.erase(j.cbegin(), j.cbegin()), std::out_of_range); - CHECK_THROWS_WITH(j.erase(j.cend(), j.cend()), "iterators out of range"); - CHECK_THROWS_WITH(j.erase(j.cbegin(), j.cbegin()), "iterators out of range"); + CHECK_THROWS_AS(j.erase(j.cend(), j.cend()), json::invalid_iterator); + CHECK_THROWS_AS(j.erase(j.cbegin(), j.cbegin()), json::invalid_iterator); + CHECK_THROWS_WITH(j.erase(j.cend(), j.cend()), "[json.exception.invalid_iterator.204] iterators out of range"); + CHECK_THROWS_WITH(j.erase(j.cbegin(), j.cbegin()), "[json.exception.invalid_iterator.204] iterators out of range"); } } } diff --git a/test/src/unit-element_access2.cpp b/test/src/unit-element_access2.cpp index d7e2cadc..3c3122f1 100644 --- a/test/src/unit-element_access2.cpp +++ b/test/src/unit-element_access2.cpp @@ -686,33 +686,33 @@ TEST_CASE("element access 2") json jobject = {{"a", "a"}, {"b", 1}, {"c", 17u}, {"d", false}, {"e", true}}; json jobject2 = {{"a", "a"}, {"b", 1}, {"c", 17u}}; CHECK_THROWS_AS(jobject.erase(jobject2.begin()), json::invalid_iterator); - CHECK_THROWS_AS(jobject.erase(jobject.begin(), jobject2.end()), std::domain_error); - CHECK_THROWS_AS(jobject.erase(jobject2.begin(), jobject.end()), std::domain_error); - CHECK_THROWS_AS(jobject.erase(jobject2.begin(), jobject2.end()), std::domain_error); + CHECK_THROWS_AS(jobject.erase(jobject.begin(), jobject2.end()), json::invalid_iterator); + CHECK_THROWS_AS(jobject.erase(jobject2.begin(), jobject.end()), json::invalid_iterator); + CHECK_THROWS_AS(jobject.erase(jobject2.begin(), jobject2.end()), json::invalid_iterator); CHECK_THROWS_WITH(jobject.erase(jobject2.begin()), "[json.exception.invalid_iterator.202] iterator does not fit current value"); CHECK_THROWS_WITH(jobject.erase(jobject.begin(), jobject2.end()), - "iterators do not fit current value"); + "[json.exception.invalid_iterator.203] iterators do not fit current value"); CHECK_THROWS_WITH(jobject.erase(jobject2.begin(), jobject.end()), - "iterators do not fit current value"); + "[json.exception.invalid_iterator.203] iterators do not fit current value"); CHECK_THROWS_WITH(jobject.erase(jobject2.begin(), jobject2.end()), - "iterators do not fit current value"); + "[json.exception.invalid_iterator.203] iterators do not fit current value"); } { json jobject = {{"a", "a"}, {"b", 1}, {"c", 17u}, {"d", false}, {"e", true}}; json jobject2 = {{"a", "a"}, {"b", 1}, {"c", 17u}}; CHECK_THROWS_AS(jobject.erase(jobject2.cbegin()), json::invalid_iterator); - CHECK_THROWS_AS(jobject.erase(jobject.cbegin(), jobject2.cend()), std::domain_error); - CHECK_THROWS_AS(jobject.erase(jobject2.cbegin(), jobject.cend()), std::domain_error); - CHECK_THROWS_AS(jobject.erase(jobject2.cbegin(), jobject2.cend()), std::domain_error); + CHECK_THROWS_AS(jobject.erase(jobject.cbegin(), jobject2.cend()), json::invalid_iterator); + CHECK_THROWS_AS(jobject.erase(jobject2.cbegin(), jobject.cend()), json::invalid_iterator); + CHECK_THROWS_AS(jobject.erase(jobject2.cbegin(), jobject2.cend()), json::invalid_iterator); CHECK_THROWS_WITH(jobject.erase(jobject2.cbegin()), "[json.exception.invalid_iterator.202] iterator does not fit current value"); CHECK_THROWS_WITH(jobject.erase(jobject.cbegin(), jobject2.cend()), - "iterators do not fit current value"); + "[json.exception.invalid_iterator.203] iterators do not fit current value"); CHECK_THROWS_WITH(jobject.erase(jobject2.cbegin(), jobject.cend()), - "iterators do not fit current value"); + "[json.exception.invalid_iterator.203] iterators do not fit current value"); CHECK_THROWS_WITH(jobject.erase(jobject2.cbegin(), jobject2.cend()), - "iterators do not fit current value"); + "[json.exception.invalid_iterator.203] iterators do not fit current value"); } } } From a4274d776683f592469bad9463639f1b2ce483f9 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Fri, 3 Mar 2017 14:00:42 +0100 Subject: [PATCH 029/530] :hammer: added user-defined exceptions 205-206 --- src/json.hpp | 15 +++-- src/json.hpp.re2c | 15 +++-- test/src/unit-constructor1.cpp | 10 +-- test/src/unit-element_access1.cpp | 40 ++++++------ test/src/unit-iterators1.cpp | 104 +++++++++++++++--------------- test/src/unit-iterators2.cpp | 16 ++--- 6 files changed, 102 insertions(+), 98 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 89bcc2ff..039223dc 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -2422,8 +2422,8 @@ class basic_json boolean, or string) where an out of range error can be detected easily; example: `"iterators out of range"` @throw std::bad_alloc if allocation for object, array, or string fails - @throw std::domain_error if called with a null value; example: `"cannot - use construct with iterators from null"` + @throw invalid_iterator.206 if called with a null value; example: `"cannot + construct with iterators from null"` @complexity Linear in distance between @a first and @a last. @@ -2519,7 +2519,8 @@ class basic_json default: { - JSON_THROW(std::domain_error("cannot use construct with iterators from " + first.m_object->type_name())); + JSON_THROW(invalid_iterator(206, "cannot construct with iterators from " + + first.m_object->type_name())); } } @@ -4402,7 +4403,7 @@ class basic_json @throw invalid_iterator.202 if called on an iterator which does not belong to the current JSON value; example: `"iterator does not fit current value"` - @throw std::out_of_range if called on a primitive type with invalid + @throw invalid_iterator.205 if called on a primitive type with invalid iterator (i.e., any iterator which is not `begin()`); example: `"iterator out of range"` @@ -4448,7 +4449,7 @@ class basic_json { if (not pos.m_it.primitive_iterator.is_begin()) { - JSON_THROW(std::out_of_range("iterator out of range")); + JSON_THROW(invalid_iterator(205, "iterator out of range")); } if (is_string()) @@ -9625,7 +9626,7 @@ class basic_json { case basic_json::value_t::object: { - JSON_THROW(std::domain_error("cannot use operator[] for object iterators")); + JSON_THROW(invalid_iterator(208, "cannot use operator[] for object iterators")); } case basic_json::value_t::array: @@ -9663,7 +9664,7 @@ class basic_json return m_it.object_iterator->first; } - JSON_THROW(std::domain_error("cannot use key() for non-object iterators")); + JSON_THROW(invalid_iterator(207, "cannot use key() for non-object iterators")); } /*! diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 5ad644b6..b4e596a2 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -2422,8 +2422,8 @@ class basic_json boolean, or string) where an out of range error can be detected easily; example: `"iterators out of range"` @throw std::bad_alloc if allocation for object, array, or string fails - @throw std::domain_error if called with a null value; example: `"cannot - use construct with iterators from null"` + @throw invalid_iterator.206 if called with a null value; example: `"cannot + construct with iterators from null"` @complexity Linear in distance between @a first and @a last. @@ -2519,7 +2519,8 @@ class basic_json default: { - JSON_THROW(std::domain_error("cannot use construct with iterators from " + first.m_object->type_name())); + JSON_THROW(invalid_iterator(206, "cannot construct with iterators from " + + first.m_object->type_name())); } } @@ -4402,7 +4403,7 @@ class basic_json @throw invalid_iterator.202 if called on an iterator which does not belong to the current JSON value; example: `"iterator does not fit current value"` - @throw std::out_of_range if called on a primitive type with invalid + @throw invalid_iterator.205 if called on a primitive type with invalid iterator (i.e., any iterator which is not `begin()`); example: `"iterator out of range"` @@ -4448,7 +4449,7 @@ class basic_json { if (not pos.m_it.primitive_iterator.is_begin()) { - JSON_THROW(std::out_of_range("iterator out of range")); + JSON_THROW(invalid_iterator(205, "iterator out of range")); } if (is_string()) @@ -9625,7 +9626,7 @@ class basic_json { case basic_json::value_t::object: { - JSON_THROW(std::domain_error("cannot use operator[] for object iterators")); + JSON_THROW(invalid_iterator(208, "cannot use operator[] for object iterators")); } case basic_json::value_t::array: @@ -9663,7 +9664,7 @@ class basic_json return m_it.object_iterator->first; } - JSON_THROW(std::domain_error("cannot use key() for non-object iterators")); + JSON_THROW(invalid_iterator(207, "cannot use key() for non-object iterators")); } /*! diff --git a/test/src/unit-constructor1.cpp b/test/src/unit-constructor1.cpp index 3f7d9bbf..d99855d8 100644 --- a/test/src/unit-constructor1.cpp +++ b/test/src/unit-constructor1.cpp @@ -1106,13 +1106,15 @@ TEST_CASE("constructors") { { json j; - CHECK_THROWS_AS(json(j.begin(), j.end()), std::domain_error); - CHECK_THROWS_WITH(json(j.begin(), j.end()), "cannot use construct with iterators from null"); + CHECK_THROWS_AS(json(j.begin(), j.end()), json::invalid_iterator); + CHECK_THROWS_WITH(json(j.begin(), j.end()), + "[json.exception.invalid_iterator.206] cannot construct with iterators from null"); } { json j; - CHECK_THROWS_AS(json(j.cbegin(), j.cend()), std::domain_error); - CHECK_THROWS_WITH(json(j.cbegin(), j.cend()), "cannot use construct with iterators from null"); + CHECK_THROWS_AS(json(j.cbegin(), j.cend()), json::invalid_iterator); + CHECK_THROWS_WITH(json(j.cbegin(), j.cend()), + "[json.exception.invalid_iterator.206] cannot construct with iterators from null"); } } diff --git a/test/src/unit-element_access1.cpp b/test/src/unit-element_access1.cpp index c84afd38..265f28e1 100644 --- a/test/src/unit-element_access1.cpp +++ b/test/src/unit-element_access1.cpp @@ -689,13 +689,13 @@ TEST_CASE("element access 1") { { json j = "foo"; - CHECK_THROWS_AS(j.erase(j.end()), std::out_of_range); - CHECK_THROWS_WITH(j.erase(j.end()), "iterator out of range"); + CHECK_THROWS_AS(j.erase(j.end()), json::invalid_iterator); + CHECK_THROWS_WITH(j.erase(j.end()), "[json.exception.invalid_iterator.205] iterator out of range"); } { json j = "bar"; - CHECK_THROWS_AS(j.erase(j.cend()), std::out_of_range); - CHECK_THROWS_WITH(j.erase(j.cend()), "iterator out of range"); + CHECK_THROWS_AS(j.erase(j.cend()), json::invalid_iterator); + CHECK_THROWS_WITH(j.erase(j.cend()), "[json.exception.invalid_iterator.205] iterator out of range"); } } @@ -703,13 +703,13 @@ TEST_CASE("element access 1") { { json j = false; - CHECK_THROWS_AS(j.erase(j.end()), std::out_of_range); - CHECK_THROWS_WITH(j.erase(j.end()), "iterator out of range"); + CHECK_THROWS_AS(j.erase(j.end()), json::invalid_iterator); + CHECK_THROWS_WITH(j.erase(j.end()), "[json.exception.invalid_iterator.205] iterator out of range"); } { json j = true; - CHECK_THROWS_AS(j.erase(j.cend()), std::out_of_range); - CHECK_THROWS_WITH(j.erase(j.cend()), "iterator out of range"); + CHECK_THROWS_AS(j.erase(j.cend()), json::invalid_iterator); + CHECK_THROWS_WITH(j.erase(j.cend()), "[json.exception.invalid_iterator.205] iterator out of range"); } } @@ -717,13 +717,13 @@ TEST_CASE("element access 1") { { json j = 17; - CHECK_THROWS_AS(j.erase(j.end()), std::out_of_range); - CHECK_THROWS_WITH(j.erase(j.end()), "iterator out of range"); + CHECK_THROWS_AS(j.erase(j.end()), json::invalid_iterator); + CHECK_THROWS_WITH(j.erase(j.end()), "[json.exception.invalid_iterator.205] iterator out of range"); } { json j = 17; - CHECK_THROWS_AS(j.erase(j.cend()), std::out_of_range); - CHECK_THROWS_WITH(j.erase(j.cend()), "iterator out of range"); + CHECK_THROWS_AS(j.erase(j.cend()), json::invalid_iterator); + CHECK_THROWS_WITH(j.erase(j.cend()), "[json.exception.invalid_iterator.205] iterator out of range"); } } @@ -731,13 +731,13 @@ TEST_CASE("element access 1") { { json j = 17u; - CHECK_THROWS_AS(j.erase(j.end()), std::out_of_range); - CHECK_THROWS_WITH(j.erase(j.end()), "iterator out of range"); + CHECK_THROWS_AS(j.erase(j.end()), json::invalid_iterator); + CHECK_THROWS_WITH(j.erase(j.end()), "[json.exception.invalid_iterator.205] iterator out of range"); } { json j = 17u; - CHECK_THROWS_AS(j.erase(j.cend()), std::out_of_range); - CHECK_THROWS_WITH(j.erase(j.cend()), "iterator out of range"); + CHECK_THROWS_AS(j.erase(j.cend()), json::invalid_iterator); + CHECK_THROWS_WITH(j.erase(j.cend()), "[json.exception.invalid_iterator.205] iterator out of range"); } } @@ -745,13 +745,13 @@ TEST_CASE("element access 1") { { json j = 23.42; - CHECK_THROWS_AS(j.erase(j.end()), std::out_of_range); - CHECK_THROWS_WITH(j.erase(j.end()), "iterator out of range"); + CHECK_THROWS_AS(j.erase(j.end()), json::invalid_iterator); + CHECK_THROWS_WITH(j.erase(j.end()), "[json.exception.invalid_iterator.205] iterator out of range"); } { json j = 23.42; - CHECK_THROWS_AS(j.erase(j.cend()), std::out_of_range); - CHECK_THROWS_WITH(j.erase(j.cend()), "iterator out of range"); + CHECK_THROWS_AS(j.erase(j.cend()), json::invalid_iterator); + CHECK_THROWS_WITH(j.erase(j.cend()), "[json.exception.invalid_iterator.205] iterator out of range"); } } } diff --git a/test/src/unit-iterators1.cpp b/test/src/unit-iterators1.cpp index eebee110..c1f179fd 100644 --- a/test/src/unit-iterators1.cpp +++ b/test/src/unit-iterators1.cpp @@ -229,22 +229,22 @@ TEST_CASE("iterators 1") { auto it = j.begin(); auto cit = j_const.cbegin(); - CHECK_THROWS_AS(it.key(), std::domain_error); - CHECK_THROWS_WITH(it.key(), "cannot use key() for non-object iterators"); + CHECK_THROWS_AS(it.key(), json::invalid_iterator); + CHECK_THROWS_WITH(it.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); CHECK(it.value() == json(true)); - CHECK_THROWS_AS(cit.key(), std::domain_error); - CHECK_THROWS_WITH(cit.key(), "cannot use key() for non-object iterators"); + CHECK_THROWS_AS(cit.key(), json::invalid_iterator); + CHECK_THROWS_WITH(cit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); CHECK(cit.value() == json(true)); auto rit = j.rend(); auto crit = j.crend(); - CHECK_THROWS_AS(rit.key(), std::domain_error); + CHECK_THROWS_AS(rit.key(), json::invalid_iterator); CHECK_THROWS_AS(rit.value(), std::out_of_range); - CHECK_THROWS_AS(crit.key(), std::domain_error); + CHECK_THROWS_AS(crit.key(), json::invalid_iterator); CHECK_THROWS_AS(crit.value(), std::out_of_range); - CHECK_THROWS_WITH(rit.key(), "cannot use key() for non-object iterators"); + CHECK_THROWS_WITH(rit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); CHECK_THROWS_WITH(rit.value(), "cannot get value"); - CHECK_THROWS_WITH(crit.key(), "cannot use key() for non-object iterators"); + CHECK_THROWS_WITH(crit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); CHECK_THROWS_WITH(crit.value(), "cannot get value"); } } @@ -433,22 +433,22 @@ TEST_CASE("iterators 1") { auto it = j.begin(); auto cit = j_const.cbegin(); - CHECK_THROWS_AS(it.key(), std::domain_error); - CHECK_THROWS_WITH(it.key(), "cannot use key() for non-object iterators"); + CHECK_THROWS_AS(it.key(), json::invalid_iterator); + CHECK_THROWS_WITH(it.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); CHECK(it.value() == json("hello world")); - CHECK_THROWS_AS(cit.key(), std::domain_error); - CHECK_THROWS_WITH(cit.key(), "cannot use key() for non-object iterators"); + CHECK_THROWS_AS(cit.key(), json::invalid_iterator); + CHECK_THROWS_WITH(cit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); CHECK(cit.value() == json("hello world")); auto rit = j.rend(); auto crit = j.crend(); - CHECK_THROWS_AS(rit.key(), std::domain_error); + CHECK_THROWS_AS(rit.key(), json::invalid_iterator); CHECK_THROWS_AS(rit.value(), std::out_of_range); - CHECK_THROWS_AS(crit.key(), std::domain_error); + CHECK_THROWS_AS(crit.key(), json::invalid_iterator); CHECK_THROWS_AS(crit.value(), std::out_of_range); - CHECK_THROWS_WITH(rit.key(), "cannot use key() for non-object iterators"); + CHECK_THROWS_WITH(rit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); CHECK_THROWS_WITH(rit.value(), "cannot get value"); - CHECK_THROWS_WITH(crit.key(), "cannot use key() for non-object iterators"); + CHECK_THROWS_WITH(crit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); CHECK_THROWS_WITH(crit.value(), "cannot get value"); } } @@ -630,11 +630,11 @@ TEST_CASE("iterators 1") { auto it = j.begin(); auto cit = j_const.cbegin(); - CHECK_THROWS_AS(it.key(), std::domain_error); - CHECK_THROWS_WITH(it.key(), "cannot use key() for non-object iterators"); + CHECK_THROWS_AS(it.key(), json::invalid_iterator); + CHECK_THROWS_WITH(it.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); CHECK(it.value() == json(1)); - CHECK_THROWS_AS(cit.key(), std::domain_error); - CHECK_THROWS_WITH(cit.key(), "cannot use key() for non-object iterators"); + CHECK_THROWS_AS(cit.key(), json::invalid_iterator); + CHECK_THROWS_WITH(cit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); CHECK(cit.value() == json(1)); } } @@ -1007,22 +1007,22 @@ TEST_CASE("iterators 1") { auto it = j.begin(); auto cit = j_const.cbegin(); - CHECK_THROWS_AS(it.key(), std::domain_error); - CHECK_THROWS_WITH(it.key(), "cannot use key() for non-object iterators"); + CHECK_THROWS_AS(it.key(), json::invalid_iterator); + CHECK_THROWS_WITH(it.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); CHECK(it.value() == json(23)); - CHECK_THROWS_AS(cit.key(), std::domain_error); - CHECK_THROWS_WITH(cit.key(), "cannot use key() for non-object iterators"); + CHECK_THROWS_AS(cit.key(), json::invalid_iterator); + CHECK_THROWS_WITH(cit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); CHECK(cit.value() == json(23)); auto rit = j.rend(); auto crit = j.crend(); - CHECK_THROWS_AS(rit.key(), std::domain_error); + CHECK_THROWS_AS(rit.key(), json::invalid_iterator); CHECK_THROWS_AS(rit.value(), std::out_of_range); - CHECK_THROWS_AS(crit.key(), std::domain_error); + CHECK_THROWS_AS(crit.key(), json::invalid_iterator); CHECK_THROWS_AS(crit.value(), std::out_of_range); - CHECK_THROWS_WITH(rit.key(), "cannot use key() for non-object iterators"); + CHECK_THROWS_WITH(rit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); CHECK_THROWS_WITH(rit.value(), "cannot get value"); - CHECK_THROWS_WITH(crit.key(), "cannot use key() for non-object iterators"); + CHECK_THROWS_WITH(crit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); CHECK_THROWS_WITH(crit.value(), "cannot get value"); } } @@ -1211,22 +1211,22 @@ TEST_CASE("iterators 1") { auto it = j.begin(); auto cit = j_const.cbegin(); - CHECK_THROWS_AS(it.key(), std::domain_error); - CHECK_THROWS_WITH(it.key(), "cannot use key() for non-object iterators"); + CHECK_THROWS_AS(it.key(), json::invalid_iterator); + CHECK_THROWS_WITH(it.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); CHECK(it.value() == json(23)); - CHECK_THROWS_AS(cit.key(), std::domain_error); - CHECK_THROWS_WITH(cit.key(), "cannot use key() for non-object iterators"); + CHECK_THROWS_AS(cit.key(), json::invalid_iterator); + CHECK_THROWS_WITH(cit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); CHECK(cit.value() == json(23)); auto rit = j.rend(); auto crit = j.crend(); - CHECK_THROWS_AS(rit.key(), std::domain_error); + CHECK_THROWS_AS(rit.key(), json::invalid_iterator); CHECK_THROWS_AS(rit.value(), std::out_of_range); - CHECK_THROWS_AS(crit.key(), std::domain_error); + CHECK_THROWS_AS(crit.key(), json::invalid_iterator); CHECK_THROWS_AS(crit.value(), std::out_of_range); - CHECK_THROWS_WITH(rit.key(), "cannot use key() for non-object iterators"); + CHECK_THROWS_WITH(rit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); CHECK_THROWS_WITH(rit.value(), "cannot get value"); - CHECK_THROWS_WITH(crit.key(), "cannot use key() for non-object iterators"); + CHECK_THROWS_WITH(crit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); CHECK_THROWS_WITH(crit.value(), "cannot get value"); } } @@ -1415,22 +1415,22 @@ TEST_CASE("iterators 1") { auto it = j.begin(); auto cit = j_const.cbegin(); - CHECK_THROWS_AS(it.key(), std::domain_error); - CHECK_THROWS_WITH(it.key(), "cannot use key() for non-object iterators"); + CHECK_THROWS_AS(it.key(), json::invalid_iterator); + CHECK_THROWS_WITH(it.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); CHECK(it.value() == json(23.42)); - CHECK_THROWS_AS(cit.key(), std::domain_error); - CHECK_THROWS_WITH(cit.key(), "cannot use key() for non-object iterators"); + CHECK_THROWS_AS(cit.key(), json::invalid_iterator); + CHECK_THROWS_WITH(cit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); CHECK(cit.value() == json(23.42)); auto rit = j.rend(); auto crit = j.crend(); - CHECK_THROWS_AS(rit.key(), std::domain_error); + CHECK_THROWS_AS(rit.key(), json::invalid_iterator); CHECK_THROWS_AS(rit.value(), std::out_of_range); - CHECK_THROWS_AS(crit.key(), std::domain_error); + CHECK_THROWS_AS(crit.key(), json::invalid_iterator); CHECK_THROWS_AS(crit.value(), std::out_of_range); - CHECK_THROWS_WITH(rit.key(), "cannot use key() for non-object iterators"); + CHECK_THROWS_WITH(rit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); CHECK_THROWS_WITH(rit.value(), "cannot get value"); - CHECK_THROWS_WITH(crit.key(), "cannot use key() for non-object iterators"); + CHECK_THROWS_WITH(crit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); CHECK_THROWS_WITH(crit.value(), "cannot get value"); } } @@ -1489,24 +1489,24 @@ TEST_CASE("iterators 1") { auto it = j.begin(); auto cit = j_const.cbegin(); - CHECK_THROWS_AS(it.key(), std::domain_error); + CHECK_THROWS_AS(it.key(), json::invalid_iterator); CHECK_THROWS_AS(it.value(), std::out_of_range); - CHECK_THROWS_AS(cit.key(), std::domain_error); + CHECK_THROWS_AS(cit.key(), json::invalid_iterator); CHECK_THROWS_AS(cit.value(), std::out_of_range); - CHECK_THROWS_WITH(it.key(), "cannot use key() for non-object iterators"); + CHECK_THROWS_WITH(it.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); CHECK_THROWS_WITH(it.value(), "cannot get value"); - CHECK_THROWS_WITH(cit.key(), "cannot use key() for non-object iterators"); + CHECK_THROWS_WITH(cit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); CHECK_THROWS_WITH(cit.value(), "cannot get value"); auto rit = j.rend(); auto crit = j.crend(); - CHECK_THROWS_AS(rit.key(), std::domain_error); + CHECK_THROWS_AS(rit.key(), json::invalid_iterator); CHECK_THROWS_AS(rit.value(), std::out_of_range); - CHECK_THROWS_AS(crit.key(), std::domain_error); + CHECK_THROWS_AS(crit.key(), json::invalid_iterator); CHECK_THROWS_AS(crit.value(), std::out_of_range); - CHECK_THROWS_WITH(rit.key(), "cannot use key() for non-object iterators"); + CHECK_THROWS_WITH(rit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); CHECK_THROWS_WITH(rit.value(), "cannot get value"); - CHECK_THROWS_WITH(crit.key(), "cannot use key() for non-object iterators"); + CHECK_THROWS_WITH(crit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); CHECK_THROWS_WITH(crit.value(), "cannot get value"); } } diff --git a/test/src/unit-iterators2.cpp b/test/src/unit-iterators2.cpp index b2bc4a38..63242f1e 100644 --- a/test/src/unit-iterators2.cpp +++ b/test/src/unit-iterators2.cpp @@ -380,17 +380,17 @@ TEST_CASE("iterators 2") { { auto it = j_object.begin(); - CHECK_THROWS_AS(it[0], std::domain_error); - CHECK_THROWS_AS(it[1], std::domain_error); - CHECK_THROWS_WITH(it[0], "cannot use operator[] for object iterators"); - CHECK_THROWS_WITH(it[1], "cannot use operator[] for object iterators"); + CHECK_THROWS_AS(it[0], json::invalid_iterator); + CHECK_THROWS_AS(it[1], json::invalid_iterator); + CHECK_THROWS_WITH(it[0], "[json.exception.invalid_iterator.208] cannot use operator[] for object iterators"); + CHECK_THROWS_WITH(it[1], "[json.exception.invalid_iterator.208] cannot use operator[] for object iterators"); } { auto it = j_object.cbegin(); - CHECK_THROWS_AS(it[0], std::domain_error); - CHECK_THROWS_AS(it[1], std::domain_error); - CHECK_THROWS_WITH(it[0], "cannot use operator[] for object iterators"); - CHECK_THROWS_WITH(it[1], "cannot use operator[] for object iterators"); + CHECK_THROWS_AS(it[0], json::invalid_iterator); + CHECK_THROWS_AS(it[1], json::invalid_iterator); + CHECK_THROWS_WITH(it[0], "[json.exception.invalid_iterator.208] cannot use operator[] for object iterators"); + CHECK_THROWS_WITH(it[1], "[json.exception.invalid_iterator.208] cannot use operator[] for object iterators"); } } From 0c40c8e3be75e0ebb74ec9394812f798eef8f7a8 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Fri, 3 Mar 2017 14:34:57 +0100 Subject: [PATCH 030/530] :hammer: added user-defined exceptions 2xx --- src/json.hpp | 30 +- src/json.hpp.re2c | 30 +- test/src/unit-algorithms.cpp | 5 +- test/src/unit-class_const_iterator.cpp | 16 +- test/src/unit-class_iterator.cpp | 16 +- test/src/unit-element_access1.cpp | 16 +- test/src/unit-iterators1.cpp | 56 ++-- test/src/unit-iterators2.cpp | 432 ++++++++++++------------- test/src/unit-modifiers.cpp | 9 +- 9 files changed, 306 insertions(+), 304 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 039223dc..59010a13 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -5755,9 +5755,9 @@ class basic_json example: `"cannot use insert() with string"` @throw invalid_iterator.202 if @a pos is not an iterator of *this; example: `"iterator does not fit current value"` - @throw std::domain_error if @a first and @a last do not belong to the same - JSON value; example: `"iterators do not fit"` - @throw std::domain_error if @a first or @a last are iterators into + @throw invalid_iterator.210 if @a first and @a last do not belong to the + same JSON value; example: `"iterators do not fit"` + @throw invalid_iterator.211 if @a first or @a last are iterators into container for which insert is called; example: `"passed iterators may not belong to container"` @@ -5788,12 +5788,12 @@ class basic_json // check if range iterators belong to the same JSON object if (first.m_object != last.m_object) { - JSON_THROW(std::domain_error("iterators do not fit")); + JSON_THROW(invalid_iterator(210, "iterators do not fit")); } if (first.m_object == this or last.m_object == this) { - JSON_THROW(std::domain_error("passed iterators may not belong to container")); + JSON_THROW(invalid_iterator(211, "passed iterators may not belong to container")); } // insert to array and return iterator @@ -9288,7 +9288,7 @@ class basic_json case basic_json::value_t::null: { - JSON_THROW(std::out_of_range("cannot get value")); + JSON_THROW(invalid_iterator(214, "cannot get value")); } default: @@ -9298,7 +9298,7 @@ class basic_json return *m_object; } - JSON_THROW(std::out_of_range("cannot get value")); + JSON_THROW(invalid_iterator(214, "cannot get value")); } } } @@ -9332,7 +9332,7 @@ class basic_json return m_object; } - JSON_THROW(std::out_of_range("cannot get value")); + JSON_THROW(invalid_iterator(214, "cannot get value")); } } } @@ -9432,7 +9432,7 @@ class basic_json // if objects are not the same, the comparison is undefined if (m_object != other.m_object) { - JSON_THROW(std::domain_error("cannot compare iterators of different containers")); + JSON_THROW(invalid_iterator(212, "cannot compare iterators of different containers")); } assert(m_object != nullptr); @@ -9474,7 +9474,7 @@ class basic_json // if objects are not the same, the comparison is undefined if (m_object != other.m_object) { - JSON_THROW(std::domain_error("cannot compare iterators of different containers")); + JSON_THROW(invalid_iterator(212, "cannot compare iterators of different containers")); } assert(m_object != nullptr); @@ -9483,7 +9483,7 @@ class basic_json { case basic_json::value_t::object: { - JSON_THROW(std::domain_error("cannot compare order of object iterators")); + JSON_THROW(invalid_iterator(213, "cannot compare order of object iterators")); } case basic_json::value_t::array: @@ -9537,7 +9537,7 @@ class basic_json { case basic_json::value_t::object: { - JSON_THROW(std::domain_error("cannot use offsets with object iterators")); + JSON_THROW(invalid_iterator(209, "cannot use offsets with object iterators")); } case basic_json::value_t::array: @@ -9599,7 +9599,7 @@ class basic_json { case basic_json::value_t::object: { - JSON_THROW(std::domain_error("cannot use offsets with object iterators")); + JSON_THROW(invalid_iterator(209, "cannot use offsets with object iterators")); } case basic_json::value_t::array: @@ -9636,7 +9636,7 @@ class basic_json case basic_json::value_t::null: { - JSON_THROW(std::out_of_range("cannot get value")); + JSON_THROW(invalid_iterator(214, "cannot get value")); } default: @@ -9646,7 +9646,7 @@ class basic_json return *m_object; } - JSON_THROW(std::out_of_range("cannot get value")); + JSON_THROW(invalid_iterator(214, "cannot get value")); } } } diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index b4e596a2..e87feb48 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -5755,9 +5755,9 @@ class basic_json example: `"cannot use insert() with string"` @throw invalid_iterator.202 if @a pos is not an iterator of *this; example: `"iterator does not fit current value"` - @throw std::domain_error if @a first and @a last do not belong to the same - JSON value; example: `"iterators do not fit"` - @throw std::domain_error if @a first or @a last are iterators into + @throw invalid_iterator.210 if @a first and @a last do not belong to the + same JSON value; example: `"iterators do not fit"` + @throw invalid_iterator.211 if @a first or @a last are iterators into container for which insert is called; example: `"passed iterators may not belong to container"` @@ -5788,12 +5788,12 @@ class basic_json // check if range iterators belong to the same JSON object if (first.m_object != last.m_object) { - JSON_THROW(std::domain_error("iterators do not fit")); + JSON_THROW(invalid_iterator(210, "iterators do not fit")); } if (first.m_object == this or last.m_object == this) { - JSON_THROW(std::domain_error("passed iterators may not belong to container")); + JSON_THROW(invalid_iterator(211, "passed iterators may not belong to container")); } // insert to array and return iterator @@ -9288,7 +9288,7 @@ class basic_json case basic_json::value_t::null: { - JSON_THROW(std::out_of_range("cannot get value")); + JSON_THROW(invalid_iterator(214, "cannot get value")); } default: @@ -9298,7 +9298,7 @@ class basic_json return *m_object; } - JSON_THROW(std::out_of_range("cannot get value")); + JSON_THROW(invalid_iterator(214, "cannot get value")); } } } @@ -9332,7 +9332,7 @@ class basic_json return m_object; } - JSON_THROW(std::out_of_range("cannot get value")); + JSON_THROW(invalid_iterator(214, "cannot get value")); } } } @@ -9432,7 +9432,7 @@ class basic_json // if objects are not the same, the comparison is undefined if (m_object != other.m_object) { - JSON_THROW(std::domain_error("cannot compare iterators of different containers")); + JSON_THROW(invalid_iterator(212, "cannot compare iterators of different containers")); } assert(m_object != nullptr); @@ -9474,7 +9474,7 @@ class basic_json // if objects are not the same, the comparison is undefined if (m_object != other.m_object) { - JSON_THROW(std::domain_error("cannot compare iterators of different containers")); + JSON_THROW(invalid_iterator(212, "cannot compare iterators of different containers")); } assert(m_object != nullptr); @@ -9483,7 +9483,7 @@ class basic_json { case basic_json::value_t::object: { - JSON_THROW(std::domain_error("cannot compare order of object iterators")); + JSON_THROW(invalid_iterator(213, "cannot compare order of object iterators")); } case basic_json::value_t::array: @@ -9537,7 +9537,7 @@ class basic_json { case basic_json::value_t::object: { - JSON_THROW(std::domain_error("cannot use offsets with object iterators")); + JSON_THROW(invalid_iterator(209, "cannot use offsets with object iterators")); } case basic_json::value_t::array: @@ -9599,7 +9599,7 @@ class basic_json { case basic_json::value_t::object: { - JSON_THROW(std::domain_error("cannot use offsets with object iterators")); + JSON_THROW(invalid_iterator(209, "cannot use offsets with object iterators")); } case basic_json::value_t::array: @@ -9636,7 +9636,7 @@ class basic_json case basic_json::value_t::null: { - JSON_THROW(std::out_of_range("cannot get value")); + JSON_THROW(invalid_iterator(214, "cannot get value")); } default: @@ -9646,7 +9646,7 @@ class basic_json return *m_object; } - JSON_THROW(std::out_of_range("cannot get value")); + JSON_THROW(invalid_iterator(214, "cannot get value")); } } } diff --git a/test/src/unit-algorithms.cpp b/test/src/unit-algorithms.cpp index 8386238a..494ea83e 100644 --- a/test/src/unit-algorithms.cpp +++ b/test/src/unit-algorithms.cpp @@ -240,8 +240,9 @@ TEST_CASE("algorithms") SECTION("sorting an object") { json j({{"one", 1}, {"two", 2}}); - CHECK_THROWS_AS(std::sort(j.begin(), j.end()), std::domain_error); - CHECK_THROWS_WITH(std::sort(j.begin(), j.end()), "cannot use offsets with object iterators"); + CHECK_THROWS_AS(std::sort(j.begin(), j.end()), json::invalid_iterator); + CHECK_THROWS_WITH(std::sort(j.begin(), j.end()), + "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } } diff --git a/test/src/unit-class_const_iterator.cpp b/test/src/unit-class_const_iterator.cpp index 840549a3..26b31265 100644 --- a/test/src/unit-class_const_iterator.cpp +++ b/test/src/unit-class_const_iterator.cpp @@ -147,8 +147,8 @@ TEST_CASE("const_iterator class") { json j(json::value_t::null); json::const_iterator it = j.cbegin(); - CHECK_THROWS_AS(*it, std::out_of_range); - CHECK_THROWS_WITH(*it, "cannot get value"); + CHECK_THROWS_AS(*it, json::invalid_iterator); + CHECK_THROWS_WITH(*it, "[json.exception.invalid_iterator.214] cannot get value"); } SECTION("number") @@ -157,8 +157,8 @@ TEST_CASE("const_iterator class") json::const_iterator it = j.cbegin(); CHECK(*it == json(17)); it = j.cend(); - CHECK_THROWS_AS(*it, std::out_of_range); - CHECK_THROWS_WITH(*it, "cannot get value"); + CHECK_THROWS_AS(*it, json::invalid_iterator); + CHECK_THROWS_WITH(*it, "[json.exception.invalid_iterator.214] cannot get value"); } SECTION("object") @@ -182,8 +182,8 @@ TEST_CASE("const_iterator class") { json j(json::value_t::null); json::const_iterator it = j.cbegin(); - CHECK_THROWS_AS(it->type_name(), std::out_of_range); - CHECK_THROWS_WITH(it->type_name(), "cannot get value"); + CHECK_THROWS_AS(it->type_name(), json::invalid_iterator); + CHECK_THROWS_WITH(it->type_name(), "[json.exception.invalid_iterator.214] cannot get value"); } SECTION("number") @@ -192,8 +192,8 @@ TEST_CASE("const_iterator class") json::const_iterator it = j.cbegin(); CHECK(it->type_name() == "number"); it = j.cend(); - CHECK_THROWS_AS(it->type_name(), std::out_of_range); - CHECK_THROWS_WITH(it->type_name(), "cannot get value"); + CHECK_THROWS_AS(it->type_name(), json::invalid_iterator); + CHECK_THROWS_WITH(it->type_name(), "[json.exception.invalid_iterator.214] cannot get value"); } SECTION("object") diff --git a/test/src/unit-class_iterator.cpp b/test/src/unit-class_iterator.cpp index 0238c337..6ff3c369 100644 --- a/test/src/unit-class_iterator.cpp +++ b/test/src/unit-class_iterator.cpp @@ -131,8 +131,8 @@ TEST_CASE("iterator class") { json j(json::value_t::null); json::iterator it = j.begin(); - CHECK_THROWS_AS(*it, std::out_of_range); - CHECK_THROWS_WITH(*it, "cannot get value"); + CHECK_THROWS_AS(*it, json::invalid_iterator); + CHECK_THROWS_WITH(*it, "[json.exception.invalid_iterator.214] cannot get value"); } SECTION("number") @@ -141,8 +141,8 @@ TEST_CASE("iterator class") json::iterator it = j.begin(); CHECK(*it == json(17)); it = j.end(); - CHECK_THROWS_AS(*it, std::out_of_range); - CHECK_THROWS_WITH(*it, "cannot get value"); + CHECK_THROWS_AS(*it, json::invalid_iterator); + CHECK_THROWS_WITH(*it, "[json.exception.invalid_iterator.214] cannot get value"); } SECTION("object") @@ -166,8 +166,8 @@ TEST_CASE("iterator class") { json j(json::value_t::null); json::iterator it = j.begin(); - CHECK_THROWS_AS(it->type_name(), std::out_of_range); - CHECK_THROWS_WITH(it->type_name(), "cannot get value"); + CHECK_THROWS_AS(it->type_name(), json::invalid_iterator); + CHECK_THROWS_WITH(it->type_name(), "[json.exception.invalid_iterator.214] cannot get value"); } SECTION("number") @@ -176,8 +176,8 @@ TEST_CASE("iterator class") json::iterator it = j.begin(); CHECK(it->type_name() == "number"); it = j.end(); - CHECK_THROWS_AS(it->type_name(), std::out_of_range); - CHECK_THROWS_WITH(it->type_name(), "cannot get value"); + CHECK_THROWS_AS(it->type_name(), json::invalid_iterator); + CHECK_THROWS_WITH(it->type_name(), "[json.exception.invalid_iterator.214] cannot get value"); } SECTION("object") diff --git a/test/src/unit-element_access1.cpp b/test/src/unit-element_access1.cpp index 265f28e1..a521e3c4 100644 --- a/test/src/unit-element_access1.cpp +++ b/test/src/unit-element_access1.cpp @@ -501,17 +501,17 @@ TEST_CASE("element access 1") { { json j; - CHECK_THROWS_AS(j.front(), std::out_of_range); - CHECK_THROWS_AS(j.back(), std::out_of_range); - CHECK_THROWS_WITH(j.front(), "cannot get value"); - CHECK_THROWS_WITH(j.back(), "cannot get value"); + CHECK_THROWS_AS(j.front(), json::invalid_iterator); + CHECK_THROWS_AS(j.back(), json::invalid_iterator); + CHECK_THROWS_WITH(j.front(), "[json.exception.invalid_iterator.214] cannot get value"); + CHECK_THROWS_WITH(j.back(), "[json.exception.invalid_iterator.214] cannot get value"); } { const json j{}; - CHECK_THROWS_AS(j.front(), std::out_of_range); - CHECK_THROWS_AS(j.back(), std::out_of_range); - CHECK_THROWS_WITH(j.front(), "cannot get value"); - CHECK_THROWS_WITH(j.back(), "cannot get value"); + CHECK_THROWS_AS(j.front(), json::invalid_iterator); + CHECK_THROWS_AS(j.back(), json::invalid_iterator); + CHECK_THROWS_WITH(j.front(), "[json.exception.invalid_iterator.214] cannot get value"); + CHECK_THROWS_WITH(j.back(), "[json.exception.invalid_iterator.214] cannot get value"); } } diff --git a/test/src/unit-iterators1.cpp b/test/src/unit-iterators1.cpp index c1f179fd..6605076a 100644 --- a/test/src/unit-iterators1.cpp +++ b/test/src/unit-iterators1.cpp @@ -239,13 +239,13 @@ TEST_CASE("iterators 1") auto rit = j.rend(); auto crit = j.crend(); CHECK_THROWS_AS(rit.key(), json::invalid_iterator); - CHECK_THROWS_AS(rit.value(), std::out_of_range); + CHECK_THROWS_AS(rit.value(), json::invalid_iterator); CHECK_THROWS_AS(crit.key(), json::invalid_iterator); - CHECK_THROWS_AS(crit.value(), std::out_of_range); + CHECK_THROWS_AS(crit.value(), json::invalid_iterator); CHECK_THROWS_WITH(rit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); - CHECK_THROWS_WITH(rit.value(), "cannot get value"); + CHECK_THROWS_WITH(rit.value(), "[json.exception.invalid_iterator.214] cannot get value"); CHECK_THROWS_WITH(crit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); - CHECK_THROWS_WITH(crit.value(), "cannot get value"); + CHECK_THROWS_WITH(crit.value(), "[json.exception.invalid_iterator.214] cannot get value"); } } @@ -443,13 +443,13 @@ TEST_CASE("iterators 1") auto rit = j.rend(); auto crit = j.crend(); CHECK_THROWS_AS(rit.key(), json::invalid_iterator); - CHECK_THROWS_AS(rit.value(), std::out_of_range); + CHECK_THROWS_AS(rit.value(), json::invalid_iterator); CHECK_THROWS_AS(crit.key(), json::invalid_iterator); - CHECK_THROWS_AS(crit.value(), std::out_of_range); + CHECK_THROWS_AS(crit.value(), json::invalid_iterator); CHECK_THROWS_WITH(rit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); - CHECK_THROWS_WITH(rit.value(), "cannot get value"); + CHECK_THROWS_WITH(rit.value(), "[json.exception.invalid_iterator.214] cannot get value"); CHECK_THROWS_WITH(crit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); - CHECK_THROWS_WITH(crit.value(), "cannot get value"); + CHECK_THROWS_WITH(crit.value(), "[json.exception.invalid_iterator.214] cannot get value"); } } @@ -1017,13 +1017,13 @@ TEST_CASE("iterators 1") auto rit = j.rend(); auto crit = j.crend(); CHECK_THROWS_AS(rit.key(), json::invalid_iterator); - CHECK_THROWS_AS(rit.value(), std::out_of_range); + CHECK_THROWS_AS(rit.value(), json::invalid_iterator); CHECK_THROWS_AS(crit.key(), json::invalid_iterator); - CHECK_THROWS_AS(crit.value(), std::out_of_range); + CHECK_THROWS_AS(crit.value(), json::invalid_iterator); CHECK_THROWS_WITH(rit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); - CHECK_THROWS_WITH(rit.value(), "cannot get value"); + CHECK_THROWS_WITH(rit.value(), "[json.exception.invalid_iterator.214] cannot get value"); CHECK_THROWS_WITH(crit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); - CHECK_THROWS_WITH(crit.value(), "cannot get value"); + CHECK_THROWS_WITH(crit.value(), "[json.exception.invalid_iterator.214] cannot get value"); } } @@ -1221,13 +1221,13 @@ TEST_CASE("iterators 1") auto rit = j.rend(); auto crit = j.crend(); CHECK_THROWS_AS(rit.key(), json::invalid_iterator); - CHECK_THROWS_AS(rit.value(), std::out_of_range); + CHECK_THROWS_AS(rit.value(), json::invalid_iterator); CHECK_THROWS_AS(crit.key(), json::invalid_iterator); - CHECK_THROWS_AS(crit.value(), std::out_of_range); + CHECK_THROWS_AS(crit.value(), json::invalid_iterator); CHECK_THROWS_WITH(rit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); - CHECK_THROWS_WITH(rit.value(), "cannot get value"); + CHECK_THROWS_WITH(rit.value(), "[json.exception.invalid_iterator.214] cannot get value"); CHECK_THROWS_WITH(crit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); - CHECK_THROWS_WITH(crit.value(), "cannot get value"); + CHECK_THROWS_WITH(crit.value(), "[json.exception.invalid_iterator.214] cannot get value"); } } @@ -1425,13 +1425,13 @@ TEST_CASE("iterators 1") auto rit = j.rend(); auto crit = j.crend(); CHECK_THROWS_AS(rit.key(), json::invalid_iterator); - CHECK_THROWS_AS(rit.value(), std::out_of_range); + CHECK_THROWS_AS(rit.value(), json::invalid_iterator); CHECK_THROWS_AS(crit.key(), json::invalid_iterator); - CHECK_THROWS_AS(crit.value(), std::out_of_range); + CHECK_THROWS_AS(crit.value(), json::invalid_iterator); CHECK_THROWS_WITH(rit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); - CHECK_THROWS_WITH(rit.value(), "cannot get value"); + CHECK_THROWS_WITH(rit.value(), "[json.exception.invalid_iterator.214] cannot get value"); CHECK_THROWS_WITH(crit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); - CHECK_THROWS_WITH(crit.value(), "cannot get value"); + CHECK_THROWS_WITH(crit.value(), "[json.exception.invalid_iterator.214] cannot get value"); } } @@ -1490,24 +1490,24 @@ TEST_CASE("iterators 1") auto it = j.begin(); auto cit = j_const.cbegin(); CHECK_THROWS_AS(it.key(), json::invalid_iterator); - CHECK_THROWS_AS(it.value(), std::out_of_range); + CHECK_THROWS_AS(it.value(), json::invalid_iterator); CHECK_THROWS_AS(cit.key(), json::invalid_iterator); - CHECK_THROWS_AS(cit.value(), std::out_of_range); + CHECK_THROWS_AS(cit.value(), json::invalid_iterator); CHECK_THROWS_WITH(it.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); - CHECK_THROWS_WITH(it.value(), "cannot get value"); + CHECK_THROWS_WITH(it.value(), "[json.exception.invalid_iterator.214] cannot get value"); CHECK_THROWS_WITH(cit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); - CHECK_THROWS_WITH(cit.value(), "cannot get value"); + CHECK_THROWS_WITH(cit.value(), "[json.exception.invalid_iterator.214] cannot get value"); auto rit = j.rend(); auto crit = j.crend(); CHECK_THROWS_AS(rit.key(), json::invalid_iterator); - CHECK_THROWS_AS(rit.value(), std::out_of_range); + CHECK_THROWS_AS(rit.value(), json::invalid_iterator); CHECK_THROWS_AS(crit.key(), json::invalid_iterator); - CHECK_THROWS_AS(crit.value(), std::out_of_range); + CHECK_THROWS_AS(crit.value(), json::invalid_iterator); CHECK_THROWS_WITH(rit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); - CHECK_THROWS_WITH(rit.value(), "cannot get value"); + CHECK_THROWS_WITH(rit.value(), "[json.exception.invalid_iterator.214] cannot get value"); CHECK_THROWS_WITH(crit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); - CHECK_THROWS_WITH(crit.value(), "cannot get value"); + CHECK_THROWS_WITH(crit.value(), "[json.exception.invalid_iterator.214] cannot get value"); } } } diff --git a/test/src/unit-iterators2.cpp b/test/src/unit-iterators2.cpp index 63242f1e..e20417a1 100644 --- a/test/src/unit-iterators2.cpp +++ b/test/src/unit-iterators2.cpp @@ -81,22 +81,22 @@ TEST_CASE("iterators 2") { if (j.type() == json::value_t::object) { - CHECK_THROWS_AS(it1 < it1, std::domain_error); - CHECK_THROWS_AS(it1 < it2, std::domain_error); - CHECK_THROWS_AS(it2 < it3, std::domain_error); - CHECK_THROWS_AS(it1 < it3, std::domain_error); - CHECK_THROWS_AS(it1_c < it1_c, std::domain_error); - CHECK_THROWS_AS(it1_c < it2_c, std::domain_error); - CHECK_THROWS_AS(it2_c < it3_c, std::domain_error); - CHECK_THROWS_AS(it1_c < it3_c, std::domain_error); - CHECK_THROWS_WITH(it1 < it1, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it1 < it2, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it2 < it3, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it1 < it3, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it1_c < it1_c, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it1_c < it2_c, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it2_c < it3_c, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it1_c < it3_c, "cannot compare order of object iterators"); + CHECK_THROWS_AS(it1 < it1, json::invalid_iterator); + CHECK_THROWS_AS(it1 < it2, json::invalid_iterator); + CHECK_THROWS_AS(it2 < it3, json::invalid_iterator); + CHECK_THROWS_AS(it1 < it3, json::invalid_iterator); + CHECK_THROWS_AS(it1_c < it1_c, json::invalid_iterator); + CHECK_THROWS_AS(it1_c < it2_c, json::invalid_iterator); + CHECK_THROWS_AS(it2_c < it3_c, json::invalid_iterator); + CHECK_THROWS_AS(it1_c < it3_c, json::invalid_iterator); + CHECK_THROWS_WITH(it1 < it1, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it1 < it2, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it2 < it3, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it1 < it3, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it1_c < it1_c, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it1_c < it2_c, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it2_c < it3_c, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it1_c < it3_c, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); } else { @@ -115,22 +115,22 @@ TEST_CASE("iterators 2") { if (j.type() == json::value_t::object) { - CHECK_THROWS_AS(it1 <= it1, std::domain_error); - CHECK_THROWS_AS(it1 <= it2, std::domain_error); - CHECK_THROWS_AS(it2 <= it3, std::domain_error); - CHECK_THROWS_AS(it1 <= it3, std::domain_error); - CHECK_THROWS_AS(it1_c <= it1_c, std::domain_error); - CHECK_THROWS_AS(it1_c <= it2_c, std::domain_error); - CHECK_THROWS_AS(it2_c <= it3_c, std::domain_error); - CHECK_THROWS_AS(it1_c <= it3_c, std::domain_error); - CHECK_THROWS_WITH(it1 <= it1, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it1 <= it2, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it2 <= it3, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it1 <= it3, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it1_c <= it1_c, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it1_c <= it2_c, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it2_c <= it3_c, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it1_c <= it3_c, "cannot compare order of object iterators"); + CHECK_THROWS_AS(it1 <= it1, json::invalid_iterator); + CHECK_THROWS_AS(it1 <= it2, json::invalid_iterator); + CHECK_THROWS_AS(it2 <= it3, json::invalid_iterator); + CHECK_THROWS_AS(it1 <= it3, json::invalid_iterator); + CHECK_THROWS_AS(it1_c <= it1_c, json::invalid_iterator); + CHECK_THROWS_AS(it1_c <= it2_c, json::invalid_iterator); + CHECK_THROWS_AS(it2_c <= it3_c, json::invalid_iterator); + CHECK_THROWS_AS(it1_c <= it3_c, json::invalid_iterator); + CHECK_THROWS_WITH(it1 <= it1, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it1 <= it2, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it2 <= it3, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it1 <= it3, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it1_c <= it1_c, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it1_c <= it2_c, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it2_c <= it3_c, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it1_c <= it3_c, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); } else { @@ -150,22 +150,22 @@ TEST_CASE("iterators 2") { if (j.type() == json::value_t::object) { - CHECK_THROWS_AS(it1 > it1, std::domain_error); - CHECK_THROWS_AS(it1 > it2, std::domain_error); - CHECK_THROWS_AS(it2 > it3, std::domain_error); - CHECK_THROWS_AS(it1 > it3, std::domain_error); - CHECK_THROWS_AS(it1_c > it1_c, std::domain_error); - CHECK_THROWS_AS(it1_c > it2_c, std::domain_error); - CHECK_THROWS_AS(it2_c > it3_c, std::domain_error); - CHECK_THROWS_AS(it1_c > it3_c, std::domain_error); - CHECK_THROWS_WITH(it1 > it1, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it1 > it2, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it2 > it3, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it1 > it3, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it1_c > it1_c, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it1_c > it2_c, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it2_c > it3_c, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it1_c > it3_c, "cannot compare order of object iterators"); + CHECK_THROWS_AS(it1 > it1, json::invalid_iterator); + CHECK_THROWS_AS(it1 > it2, json::invalid_iterator); + CHECK_THROWS_AS(it2 > it3, json::invalid_iterator); + CHECK_THROWS_AS(it1 > it3, json::invalid_iterator); + CHECK_THROWS_AS(it1_c > it1_c, json::invalid_iterator); + CHECK_THROWS_AS(it1_c > it2_c, json::invalid_iterator); + CHECK_THROWS_AS(it2_c > it3_c, json::invalid_iterator); + CHECK_THROWS_AS(it1_c > it3_c, json::invalid_iterator); + CHECK_THROWS_WITH(it1 > it1, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it1 > it2, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it2 > it3, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it1 > it3, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it1_c > it1_c, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it1_c > it2_c, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it2_c > it3_c, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it1_c > it3_c, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); } else { @@ -185,22 +185,22 @@ TEST_CASE("iterators 2") { if (j.type() == json::value_t::object) { - CHECK_THROWS_AS(it1 >= it1, std::domain_error); - CHECK_THROWS_AS(it1 >= it2, std::domain_error); - CHECK_THROWS_AS(it2 >= it3, std::domain_error); - CHECK_THROWS_AS(it1 >= it3, std::domain_error); - CHECK_THROWS_AS(it1_c >= it1_c, std::domain_error); - CHECK_THROWS_AS(it1_c >= it2_c, std::domain_error); - CHECK_THROWS_AS(it2_c >= it3_c, std::domain_error); - CHECK_THROWS_AS(it1_c >= it3_c, std::domain_error); - CHECK_THROWS_WITH(it1 >= it1, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it1 >= it2, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it2 >= it3, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it1 >= it3, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it1_c >= it1_c, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it1_c >= it2_c, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it2_c >= it3_c, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it1_c >= it3_c, "cannot compare order of object iterators"); + CHECK_THROWS_AS(it1 >= it1, json::invalid_iterator); + CHECK_THROWS_AS(it1 >= it2, json::invalid_iterator); + CHECK_THROWS_AS(it2 >= it3, json::invalid_iterator); + CHECK_THROWS_AS(it1 >= it3, json::invalid_iterator); + CHECK_THROWS_AS(it1_c >= it1_c, json::invalid_iterator); + CHECK_THROWS_AS(it1_c >= it2_c, json::invalid_iterator); + CHECK_THROWS_AS(it2_c >= it3_c, json::invalid_iterator); + CHECK_THROWS_AS(it1_c >= it3_c, json::invalid_iterator); + CHECK_THROWS_WITH(it1 >= it1, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it1 >= it2, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it2 >= it3, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it1 >= it3, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it1_c >= it1_c, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it1_c >= it2_c, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it2_c >= it3_c, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it1_c >= it3_c, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); } else { @@ -224,15 +224,15 @@ TEST_CASE("iterators 2") { if (j != k) { - CHECK_THROWS_AS(j.begin() == k.begin(), std::domain_error); - CHECK_THROWS_AS(j.cbegin() == k.cbegin(), std::domain_error); - CHECK_THROWS_WITH(j.begin() == k.begin(), "cannot compare iterators of different containers"); - CHECK_THROWS_WITH(j.cbegin() == k.cbegin(), "cannot compare iterators of different containers"); + CHECK_THROWS_AS(j.begin() == k.begin(), json::invalid_iterator); + CHECK_THROWS_AS(j.cbegin() == k.cbegin(), json::invalid_iterator); + CHECK_THROWS_WITH(j.begin() == k.begin(), "[json.exception.invalid_iterator.212] cannot compare iterators of different containers"); + CHECK_THROWS_WITH(j.cbegin() == k.cbegin(), "[json.exception.invalid_iterator.212] cannot compare iterators of different containers"); - CHECK_THROWS_AS(j.begin() < k.begin(), std::domain_error); - CHECK_THROWS_AS(j.cbegin() < k.cbegin(), std::domain_error); - CHECK_THROWS_WITH(j.begin() < k.begin(), "cannot compare iterators of different containers"); - CHECK_THROWS_WITH(j.cbegin() < k.cbegin(), "cannot compare iterators of different containers"); + CHECK_THROWS_AS(j.begin() < k.begin(), json::invalid_iterator); + CHECK_THROWS_AS(j.cbegin() < k.cbegin(), json::invalid_iterator); + CHECK_THROWS_WITH(j.begin() < k.begin(), "[json.exception.invalid_iterator.212] cannot compare iterators of different containers"); + CHECK_THROWS_WITH(j.cbegin() < k.cbegin(), "[json.exception.invalid_iterator.212] cannot compare iterators of different containers"); } } } @@ -251,53 +251,53 @@ TEST_CASE("iterators 2") { { auto it = j_object.begin(); - CHECK_THROWS_AS(it += 1, std::domain_error); - CHECK_THROWS_WITH(it += 1, "cannot use offsets with object iterators"); + CHECK_THROWS_AS(it += 1, json::invalid_iterator); + CHECK_THROWS_WITH(it += 1, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } { auto it = j_object.cbegin(); - CHECK_THROWS_AS(it += 1, std::domain_error); - CHECK_THROWS_WITH(it += 1, "cannot use offsets with object iterators"); + CHECK_THROWS_AS(it += 1, json::invalid_iterator); + CHECK_THROWS_WITH(it += 1, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } { auto it = j_object.begin(); - CHECK_THROWS_AS(it + 1, std::domain_error); - CHECK_THROWS_WITH(it + 1, "cannot use offsets with object iterators"); + CHECK_THROWS_AS(it + 1, json::invalid_iterator); + CHECK_THROWS_WITH(it + 1, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } { auto it = j_object.cbegin(); - CHECK_THROWS_AS(it + 1, std::domain_error); - CHECK_THROWS_WITH(it + 1, "cannot use offsets with object iterators"); + CHECK_THROWS_AS(it + 1, json::invalid_iterator); + CHECK_THROWS_WITH(it + 1, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } { auto it = j_object.begin(); - CHECK_THROWS_AS(it -= 1, std::domain_error); - CHECK_THROWS_WITH(it -= 1, "cannot use offsets with object iterators"); + CHECK_THROWS_AS(it -= 1, json::invalid_iterator); + CHECK_THROWS_WITH(it -= 1, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } { auto it = j_object.cbegin(); - CHECK_THROWS_AS(it -= 1, std::domain_error); - CHECK_THROWS_WITH(it -= 1, "cannot use offsets with object iterators"); + CHECK_THROWS_AS(it -= 1, json::invalid_iterator); + CHECK_THROWS_WITH(it -= 1, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } { auto it = j_object.begin(); - CHECK_THROWS_AS(it - 1, std::domain_error); - CHECK_THROWS_WITH(it - 1, "cannot use offsets with object iterators"); + CHECK_THROWS_AS(it - 1, json::invalid_iterator); + CHECK_THROWS_WITH(it - 1, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } { auto it = j_object.cbegin(); - CHECK_THROWS_AS(it - 1, std::domain_error); - CHECK_THROWS_WITH(it - 1, "cannot use offsets with object iterators"); + CHECK_THROWS_AS(it - 1, json::invalid_iterator); + CHECK_THROWS_WITH(it - 1, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } { auto it = j_object.begin(); - CHECK_THROWS_AS(it - it, std::domain_error); - CHECK_THROWS_WITH(it - it, "cannot use offsets with object iterators"); + CHECK_THROWS_AS(it - it, json::invalid_iterator); + CHECK_THROWS_WITH(it - it, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } { auto it = j_object.cbegin(); - CHECK_THROWS_AS(it - it, std::domain_error); - CHECK_THROWS_WITH(it - it, "cannot use offsets with object iterators"); + CHECK_THROWS_AS(it - it, json::invalid_iterator); + CHECK_THROWS_WITH(it - it, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } } @@ -420,17 +420,17 @@ TEST_CASE("iterators 2") { { auto it = j_null.begin(); - CHECK_THROWS_AS(it[0], std::out_of_range); - CHECK_THROWS_AS(it[1], std::out_of_range); - CHECK_THROWS_WITH(it[0], "cannot get value"); - CHECK_THROWS_WITH(it[1], "cannot get value"); + CHECK_THROWS_AS(it[0], json::invalid_iterator); + CHECK_THROWS_AS(it[1], json::invalid_iterator); + CHECK_THROWS_WITH(it[0], "[json.exception.invalid_iterator.214] cannot get value"); + CHECK_THROWS_WITH(it[1], "[json.exception.invalid_iterator.214] cannot get value"); } { auto it = j_null.cbegin(); - CHECK_THROWS_AS(it[0], std::out_of_range); - CHECK_THROWS_AS(it[1], std::out_of_range); - CHECK_THROWS_WITH(it[0], "cannot get value"); - CHECK_THROWS_WITH(it[1], "cannot get value"); + CHECK_THROWS_AS(it[0], json::invalid_iterator); + CHECK_THROWS_AS(it[1], json::invalid_iterator); + CHECK_THROWS_WITH(it[0], "[json.exception.invalid_iterator.214] cannot get value"); + CHECK_THROWS_WITH(it[1], "[json.exception.invalid_iterator.214] cannot get value"); } } @@ -439,14 +439,14 @@ TEST_CASE("iterators 2") { auto it = j_value.begin(); CHECK(it[0] == json(42)); - CHECK_THROWS_AS(it[1], std::out_of_range); - CHECK_THROWS_WITH(it[1], "cannot get value"); + CHECK_THROWS_AS(it[1], json::invalid_iterator); + CHECK_THROWS_WITH(it[1], "[json.exception.invalid_iterator.214] cannot get value"); } { auto it = j_value.cbegin(); CHECK(it[0] == json(42)); - CHECK_THROWS_AS(it[1], std::out_of_range); - CHECK_THROWS_WITH(it[1], "cannot get value"); + CHECK_THROWS_AS(it[1], json::invalid_iterator); + CHECK_THROWS_WITH(it[1], "[json.exception.invalid_iterator.214] cannot get value"); } } } @@ -500,22 +500,22 @@ TEST_CASE("iterators 2") { if (j.type() == json::value_t::object) { - CHECK_THROWS_AS(it1 < it1, std::domain_error); - CHECK_THROWS_AS(it1 < it2, std::domain_error); - CHECK_THROWS_AS(it2 < it3, std::domain_error); - CHECK_THROWS_AS(it1 < it3, std::domain_error); - CHECK_THROWS_AS(it1_c < it1_c, std::domain_error); - CHECK_THROWS_AS(it1_c < it2_c, std::domain_error); - CHECK_THROWS_AS(it2_c < it3_c, std::domain_error); - CHECK_THROWS_AS(it1_c < it3_c, std::domain_error); - CHECK_THROWS_WITH(it1 < it1, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it1 < it2, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it2 < it3, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it1 < it3, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it1_c < it1_c, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it1_c < it2_c, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it2_c < it3_c, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it1_c < it3_c, "cannot compare order of object iterators"); + CHECK_THROWS_AS(it1 < it1, json::invalid_iterator); + CHECK_THROWS_AS(it1 < it2, json::invalid_iterator); + CHECK_THROWS_AS(it2 < it3, json::invalid_iterator); + CHECK_THROWS_AS(it1 < it3, json::invalid_iterator); + CHECK_THROWS_AS(it1_c < it1_c, json::invalid_iterator); + CHECK_THROWS_AS(it1_c < it2_c, json::invalid_iterator); + CHECK_THROWS_AS(it2_c < it3_c, json::invalid_iterator); + CHECK_THROWS_AS(it1_c < it3_c, json::invalid_iterator); + CHECK_THROWS_WITH(it1 < it1, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it1 < it2, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it2 < it3, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it1 < it3, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it1_c < it1_c, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it1_c < it2_c, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it2_c < it3_c, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it1_c < it3_c, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); } else { @@ -534,22 +534,22 @@ TEST_CASE("iterators 2") { if (j.type() == json::value_t::object) { - CHECK_THROWS_AS(it1 <= it1, std::domain_error); - CHECK_THROWS_AS(it1 <= it2, std::domain_error); - CHECK_THROWS_AS(it2 <= it3, std::domain_error); - CHECK_THROWS_AS(it1 <= it3, std::domain_error); - CHECK_THROWS_AS(it1_c <= it1_c, std::domain_error); - CHECK_THROWS_AS(it1_c <= it2_c, std::domain_error); - CHECK_THROWS_AS(it2_c <= it3_c, std::domain_error); - CHECK_THROWS_AS(it1_c <= it3_c, std::domain_error); - CHECK_THROWS_WITH(it1 <= it1, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it1 <= it2, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it2 <= it3, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it1 <= it3, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it1_c <= it1_c, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it1_c <= it2_c, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it2_c <= it3_c, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it1_c <= it3_c, "cannot compare order of object iterators"); + CHECK_THROWS_AS(it1 <= it1, json::invalid_iterator); + CHECK_THROWS_AS(it1 <= it2, json::invalid_iterator); + CHECK_THROWS_AS(it2 <= it3, json::invalid_iterator); + CHECK_THROWS_AS(it1 <= it3, json::invalid_iterator); + CHECK_THROWS_AS(it1_c <= it1_c, json::invalid_iterator); + CHECK_THROWS_AS(it1_c <= it2_c, json::invalid_iterator); + CHECK_THROWS_AS(it2_c <= it3_c, json::invalid_iterator); + CHECK_THROWS_AS(it1_c <= it3_c, json::invalid_iterator); + CHECK_THROWS_WITH(it1 <= it1, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it1 <= it2, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it2 <= it3, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it1 <= it3, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it1_c <= it1_c, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it1_c <= it2_c, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it2_c <= it3_c, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it1_c <= it3_c, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); } else { @@ -569,22 +569,22 @@ TEST_CASE("iterators 2") { if (j.type() == json::value_t::object) { - CHECK_THROWS_AS(it1 > it1, std::domain_error); - CHECK_THROWS_AS(it1 > it2, std::domain_error); - CHECK_THROWS_AS(it2 > it3, std::domain_error); - CHECK_THROWS_AS(it1 > it3, std::domain_error); - CHECK_THROWS_AS(it1_c > it1_c, std::domain_error); - CHECK_THROWS_AS(it1_c > it2_c, std::domain_error); - CHECK_THROWS_AS(it2_c > it3_c, std::domain_error); - CHECK_THROWS_AS(it1_c > it3_c, std::domain_error); - CHECK_THROWS_WITH(it1 > it1, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it1 > it2, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it2 > it3, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it1 > it3, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it1_c > it1_c, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it1_c > it2_c, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it2_c > it3_c, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it1_c > it3_c, "cannot compare order of object iterators"); + CHECK_THROWS_AS(it1 > it1, json::invalid_iterator); + CHECK_THROWS_AS(it1 > it2, json::invalid_iterator); + CHECK_THROWS_AS(it2 > it3, json::invalid_iterator); + CHECK_THROWS_AS(it1 > it3, json::invalid_iterator); + CHECK_THROWS_AS(it1_c > it1_c, json::invalid_iterator); + CHECK_THROWS_AS(it1_c > it2_c, json::invalid_iterator); + CHECK_THROWS_AS(it2_c > it3_c, json::invalid_iterator); + CHECK_THROWS_AS(it1_c > it3_c, json::invalid_iterator); + CHECK_THROWS_WITH(it1 > it1, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it1 > it2, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it2 > it3, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it1 > it3, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it1_c > it1_c, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it1_c > it2_c, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it2_c > it3_c, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it1_c > it3_c, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); } else { @@ -604,22 +604,22 @@ TEST_CASE("iterators 2") { if (j.type() == json::value_t::object) { - CHECK_THROWS_AS(it1 >= it1, std::domain_error); - CHECK_THROWS_AS(it1 >= it2, std::domain_error); - CHECK_THROWS_AS(it2 >= it3, std::domain_error); - CHECK_THROWS_AS(it1 >= it3, std::domain_error); - CHECK_THROWS_AS(it1_c >= it1_c, std::domain_error); - CHECK_THROWS_AS(it1_c >= it2_c, std::domain_error); - CHECK_THROWS_AS(it2_c >= it3_c, std::domain_error); - CHECK_THROWS_AS(it1_c >= it3_c, std::domain_error); - CHECK_THROWS_WITH(it1 >= it1, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it1 >= it2, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it2 >= it3, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it1 >= it3, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it1_c >= it1_c, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it1_c >= it2_c, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it2_c >= it3_c, "cannot compare order of object iterators"); - CHECK_THROWS_WITH(it1_c >= it3_c, "cannot compare order of object iterators"); + CHECK_THROWS_AS(it1 >= it1, json::invalid_iterator); + CHECK_THROWS_AS(it1 >= it2, json::invalid_iterator); + CHECK_THROWS_AS(it2 >= it3, json::invalid_iterator); + CHECK_THROWS_AS(it1 >= it3, json::invalid_iterator); + CHECK_THROWS_AS(it1_c >= it1_c, json::invalid_iterator); + CHECK_THROWS_AS(it1_c >= it2_c, json::invalid_iterator); + CHECK_THROWS_AS(it2_c >= it3_c, json::invalid_iterator); + CHECK_THROWS_AS(it1_c >= it3_c, json::invalid_iterator); + CHECK_THROWS_WITH(it1 >= it1, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it1 >= it2, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it2 >= it3, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it1 >= it3, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it1_c >= it1_c, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it1_c >= it2_c, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it2_c >= it3_c, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); + CHECK_THROWS_WITH(it1_c >= it3_c, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); } else { @@ -643,15 +643,15 @@ TEST_CASE("iterators 2") { if (j != k) { - CHECK_THROWS_AS(j.rbegin() == k.rbegin(), std::domain_error); - CHECK_THROWS_AS(j.crbegin() == k.crbegin(), std::domain_error); - CHECK_THROWS_WITH(j.rbegin() == k.rbegin(), "cannot compare iterators of different containers"); - CHECK_THROWS_WITH(j.crbegin() == k.crbegin(), "cannot compare iterators of different containers"); + CHECK_THROWS_AS(j.rbegin() == k.rbegin(), json::invalid_iterator); + CHECK_THROWS_AS(j.crbegin() == k.crbegin(), json::invalid_iterator); + CHECK_THROWS_WITH(j.rbegin() == k.rbegin(), "[json.exception.invalid_iterator.212] cannot compare iterators of different containers"); + CHECK_THROWS_WITH(j.crbegin() == k.crbegin(), "[json.exception.invalid_iterator.212] cannot compare iterators of different containers"); - CHECK_THROWS_AS(j.rbegin() < k.rbegin(), std::domain_error); - CHECK_THROWS_AS(j.crbegin() < k.crbegin(), std::domain_error); - CHECK_THROWS_WITH(j.rbegin() < k.rbegin(), "cannot compare iterators of different containers"); - CHECK_THROWS_WITH(j.crbegin() < k.crbegin(), "cannot compare iterators of different containers"); + CHECK_THROWS_AS(j.rbegin() < k.rbegin(), json::invalid_iterator); + CHECK_THROWS_AS(j.crbegin() < k.crbegin(), json::invalid_iterator); + CHECK_THROWS_WITH(j.rbegin() < k.rbegin(), "[json.exception.invalid_iterator.212] cannot compare iterators of different containers"); + CHECK_THROWS_WITH(j.crbegin() < k.crbegin(), "[json.exception.invalid_iterator.212] cannot compare iterators of different containers"); } } } @@ -670,53 +670,53 @@ TEST_CASE("iterators 2") { { auto it = j_object.rbegin(); - CHECK_THROWS_AS(it += 1, std::domain_error); - CHECK_THROWS_WITH(it += 1, "cannot use offsets with object iterators"); + CHECK_THROWS_AS(it += 1, json::invalid_iterator); + CHECK_THROWS_WITH(it += 1, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } { auto it = j_object.crbegin(); - CHECK_THROWS_AS(it += 1, std::domain_error); - CHECK_THROWS_WITH(it += 1, "cannot use offsets with object iterators"); + CHECK_THROWS_AS(it += 1, json::invalid_iterator); + CHECK_THROWS_WITH(it += 1, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } { auto it = j_object.rbegin(); - CHECK_THROWS_AS(it + 1, std::domain_error); - CHECK_THROWS_WITH(it + 1, "cannot use offsets with object iterators"); + CHECK_THROWS_AS(it + 1, json::invalid_iterator); + CHECK_THROWS_WITH(it + 1, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } { auto it = j_object.crbegin(); - CHECK_THROWS_AS(it + 1, std::domain_error); - CHECK_THROWS_WITH(it + 1, "cannot use offsets with object iterators"); + CHECK_THROWS_AS(it + 1, json::invalid_iterator); + CHECK_THROWS_WITH(it + 1, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } { auto it = j_object.rbegin(); - CHECK_THROWS_AS(it -= 1, std::domain_error); - CHECK_THROWS_WITH(it -= 1, "cannot use offsets with object iterators"); + CHECK_THROWS_AS(it -= 1, json::invalid_iterator); + CHECK_THROWS_WITH(it -= 1, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } { auto it = j_object.crbegin(); - CHECK_THROWS_AS(it -= 1, std::domain_error); - CHECK_THROWS_WITH(it -= 1, "cannot use offsets with object iterators"); + CHECK_THROWS_AS(it -= 1, json::invalid_iterator); + CHECK_THROWS_WITH(it -= 1, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } { auto it = j_object.rbegin(); - CHECK_THROWS_AS(it - 1, std::domain_error); - CHECK_THROWS_WITH(it - 1, "cannot use offsets with object iterators"); + CHECK_THROWS_AS(it - 1, json::invalid_iterator); + CHECK_THROWS_WITH(it - 1, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } { auto it = j_object.crbegin(); - CHECK_THROWS_AS(it - 1, std::domain_error); - CHECK_THROWS_WITH(it - 1, "cannot use offsets with object iterators"); + CHECK_THROWS_AS(it - 1, json::invalid_iterator); + CHECK_THROWS_WITH(it - 1, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } { auto it = j_object.rbegin(); - CHECK_THROWS_AS(it - it, std::domain_error); - CHECK_THROWS_WITH(it - it, "cannot use offsets with object iterators"); + CHECK_THROWS_AS(it - it, json::invalid_iterator); + CHECK_THROWS_WITH(it - it, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } { auto it = j_object.crbegin(); - CHECK_THROWS_AS(it - it, std::domain_error); - CHECK_THROWS_WITH(it - it, "cannot use offsets with object iterators"); + CHECK_THROWS_AS(it - it, json::invalid_iterator); + CHECK_THROWS_WITH(it - it, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } } @@ -799,17 +799,17 @@ TEST_CASE("iterators 2") { { auto it = j_object.rbegin(); - CHECK_THROWS_AS(it[0], std::domain_error); - CHECK_THROWS_AS(it[1], std::domain_error); - CHECK_THROWS_WITH(it[0], "cannot use offsets with object iterators"); - CHECK_THROWS_WITH(it[1], "cannot use offsets with object iterators"); + CHECK_THROWS_AS(it[0], json::invalid_iterator); + CHECK_THROWS_AS(it[1], json::invalid_iterator); + CHECK_THROWS_WITH(it[0], "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); + CHECK_THROWS_WITH(it[1], "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } { auto it = j_object.crbegin(); - CHECK_THROWS_AS(it[0], std::domain_error); - CHECK_THROWS_AS(it[1], std::domain_error); - CHECK_THROWS_WITH(it[0], "cannot use offsets with object iterators"); - CHECK_THROWS_WITH(it[1], "cannot use offsets with object iterators"); + CHECK_THROWS_AS(it[0], json::invalid_iterator); + CHECK_THROWS_AS(it[1], json::invalid_iterator); + CHECK_THROWS_WITH(it[0], "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); + CHECK_THROWS_WITH(it[1], "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } } @@ -839,17 +839,17 @@ TEST_CASE("iterators 2") { { auto it = j_null.rbegin(); - CHECK_THROWS_AS(it[0], std::out_of_range); - CHECK_THROWS_AS(it[1], std::out_of_range); - CHECK_THROWS_WITH(it[0], "cannot get value"); - CHECK_THROWS_WITH(it[1], "cannot get value"); + CHECK_THROWS_AS(it[0], json::invalid_iterator); + CHECK_THROWS_AS(it[1], json::invalid_iterator); + CHECK_THROWS_WITH(it[0], "[json.exception.invalid_iterator.214] cannot get value"); + CHECK_THROWS_WITH(it[1], "[json.exception.invalid_iterator.214] cannot get value"); } { auto it = j_null.crbegin(); - CHECK_THROWS_AS(it[0], std::out_of_range); - CHECK_THROWS_AS(it[1], std::out_of_range); - CHECK_THROWS_WITH(it[0], "cannot get value"); - CHECK_THROWS_WITH(it[1], "cannot get value"); + CHECK_THROWS_AS(it[0], json::invalid_iterator); + CHECK_THROWS_AS(it[1], json::invalid_iterator); + CHECK_THROWS_WITH(it[0], "[json.exception.invalid_iterator.214] cannot get value"); + CHECK_THROWS_WITH(it[1], "[json.exception.invalid_iterator.214] cannot get value"); } } @@ -858,14 +858,14 @@ TEST_CASE("iterators 2") { auto it = j_value.rbegin(); CHECK(it[0] == json(42)); - CHECK_THROWS_AS(it[1], std::out_of_range); - CHECK_THROWS_WITH(it[1], "cannot get value"); + CHECK_THROWS_AS(it[1], json::invalid_iterator); + CHECK_THROWS_WITH(it[1], "[json.exception.invalid_iterator.214] cannot get value"); } { auto it = j_value.crbegin(); CHECK(it[0] == json(42)); - CHECK_THROWS_AS(it[1], std::out_of_range); - CHECK_THROWS_WITH(it[1], "cannot get value"); + CHECK_THROWS_AS(it[1], json::invalid_iterator); + CHECK_THROWS_WITH(it[1], "[json.exception.invalid_iterator.214] cannot get value"); } } } diff --git a/test/src/unit-modifiers.cpp b/test/src/unit-modifiers.cpp index 00521535..2a3826c2 100644 --- a/test/src/unit-modifiers.cpp +++ b/test/src/unit-modifiers.cpp @@ -616,14 +616,15 @@ TEST_CASE("modifiers") { json j_other_array2 = {"first", "second"}; - CHECK_THROWS_AS(j_array.insert(j_array.end(), j_array.begin(), j_array.end()), std::domain_error); + CHECK_THROWS_AS(j_array.insert(j_array.end(), j_array.begin(), j_array.end()), + json::invalid_iterator); CHECK_THROWS_AS(j_array.insert(j_array.end(), j_other_array.begin(), j_other_array2.end()), - std::domain_error); + json::invalid_iterator); CHECK_THROWS_WITH(j_array.insert(j_array.end(), j_array.begin(), j_array.end()), - "passed iterators may not belong to container"); + "[json.exception.invalid_iterator.211] passed iterators may not belong to container"); CHECK_THROWS_WITH(j_array.insert(j_array.end(), j_other_array.begin(), j_other_array2.end()), - "iterators do not fit"); + "[json.exception.invalid_iterator.210] iterators do not fit"); } } From 068c7acda9822b8b32edeba09de1567303517ac0 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 5 Mar 2017 18:40:09 +0100 Subject: [PATCH 031/530] :hammer: added user-defined exceptions 301 --- src/json.hpp | 8 ++++---- src/json.hpp.re2c | 8 ++++---- test/src/unit-constructor1.cpp | 4 ++-- 3 files changed, 10 insertions(+), 10 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 59010a13..508dd805 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -2223,8 +2223,8 @@ class basic_json value_t::array and @ref value_t::object are valid); when @a type_deduction is set to `true`, this parameter has no effect - @throw std::domain_error if @a type_deduction is `false`, @a manual_type - is `value_t::object`, but @a init contains an element which is not a pair + @throw type_error.301 if @a type_deduction is `false`, @a manual_type is + `value_t::object`, but @a init contains an element which is not a pair whose first element is a string; example: `"cannot create object from initializer list"` @@ -2264,7 +2264,7 @@ class basic_json // if object is wanted but impossible, throw an exception if (manual_type == value_t::object and not is_an_object) { - JSON_THROW(std::domain_error("cannot create object from initializer list")); + JSON_THROW(type_error(301, "cannot create object from initializer list")); } } @@ -2347,7 +2347,7 @@ class basic_json @return JSON object value - @throw std::domain_error if @a init is not a pair whose first elements are + @throw type_error.301 if @a init is not a pair whose first elements are strings; thrown by @ref basic_json(std::initializer_list, bool, value_t) diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index e87feb48..03d873af 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -2223,8 +2223,8 @@ class basic_json value_t::array and @ref value_t::object are valid); when @a type_deduction is set to `true`, this parameter has no effect - @throw std::domain_error if @a type_deduction is `false`, @a manual_type - is `value_t::object`, but @a init contains an element which is not a pair + @throw type_error.301 if @a type_deduction is `false`, @a manual_type is + `value_t::object`, but @a init contains an element which is not a pair whose first element is a string; example: `"cannot create object from initializer list"` @@ -2264,7 +2264,7 @@ class basic_json // if object is wanted but impossible, throw an exception if (manual_type == value_t::object and not is_an_object) { - JSON_THROW(std::domain_error("cannot create object from initializer list")); + JSON_THROW(type_error(301, "cannot create object from initializer list")); } } @@ -2347,7 +2347,7 @@ class basic_json @return JSON object value - @throw std::domain_error if @a init is not a pair whose first elements are + @throw type_error.301 if @a init is not a pair whose first elements are strings; thrown by @ref basic_json(std::initializer_list, bool, value_t) diff --git a/test/src/unit-constructor1.cpp b/test/src/unit-constructor1.cpp index d99855d8..e71c2c46 100644 --- a/test/src/unit-constructor1.cpp +++ b/test/src/unit-constructor1.cpp @@ -943,9 +943,9 @@ TEST_CASE("constructors") SECTION("object with error") { CHECK_THROWS_AS(json::object({ {"one", 1}, {"two", 1u}, {"three", 2.2}, {"four", false}, 13 }), - std::logic_error); + json::type_error); CHECK_THROWS_WITH(json::object({ {"one", 1}, {"two", 1u}, {"three", 2.2}, {"four", false}, 13 }), - "cannot create object from initializer list"); + "[json.exception.type_error.301] cannot create object from initializer list"); } SECTION("empty array") From f473f74fb086a0a497c6ef24655ada1189649d1b Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 5 Mar 2017 18:56:35 +0100 Subject: [PATCH 032/530] :hammer: added user-defined exceptions 303-304 --- src/json.hpp | 29 ++++++++-------- src/json.hpp.re2c | 29 ++++++++-------- test/src/unit-element_access1.cpp | 56 +++++++++++++++---------------- test/src/unit-element_access2.cpp | 56 +++++++++++++++---------------- 4 files changed, 84 insertions(+), 86 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 508dd805..f51c85ad 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -3255,7 +3255,7 @@ class basic_json @tparam ThisType will be deduced as `basic_json` or `const basic_json` - @throw std::domain_error if ReferenceType does not match underlying value + @throw type_error.303 if ReferenceType does not match underlying value type of the current JSON */ template @@ -3272,8 +3272,7 @@ class basic_json return *ptr; } - JSON_THROW(std::domain_error("incompatible ReferenceType for get_ref, actual type is " + - obj.type_name())); + JSON_THROW(type_error(303, "incompatible ReferenceType for get_ref, actual type is " + obj.type_name())); } public: @@ -3554,8 +3553,8 @@ class basic_json reference type @a ReferenceType fits to the JSON value; throws std::domain_error otherwise - @throw std::domain_error in case passed type @a ReferenceType is - incompatible with the stored JSON value + @throw type_error.303 in case passed type @a ReferenceType is incompatible + with the stored JSON value @complexity Constant. @@ -3646,8 +3645,8 @@ class basic_json @return reference to the element at index @a idx - @throw std::domain_error if the JSON value is not an array; example: - `"cannot use at() with string"` + @throw type_error.304 if the JSON value is not an array; example: `"cannot + use at() with string"` @throw std::out_of_range if the index @a idx is out of range of the array; that is, `idx >= size()`; example: `"array index 7 is out of range"` @@ -3675,7 +3674,7 @@ class basic_json } else { - JSON_THROW(std::domain_error("cannot use at() with " + type_name())); + JSON_THROW(type_error(304, "cannot use at() with " + type_name())); } } @@ -3689,8 +3688,8 @@ class basic_json @return const reference to the element at index @a idx - @throw std::domain_error if the JSON value is not an array; example: - `"cannot use at() with string"` + @throw type_error.304 if the JSON value is not an array; example: `"cannot + use at() with string"` @throw std::out_of_range if the index @a idx is out of range of the array; that is, `idx >= size()`; example: `"array index 7 is out of range"` @@ -3718,7 +3717,7 @@ class basic_json } else { - JSON_THROW(std::domain_error("cannot use at() with " + type_name())); + JSON_THROW(type_error(304, "cannot use at() with " + type_name())); } } @@ -3732,7 +3731,7 @@ class basic_json @return reference to the element at key @a key - @throw std::domain_error if the JSON value is not an object; example: + @throw type_error.304 if the JSON value is not an object; example: `"cannot use at() with boolean"` @throw std::out_of_range if the key @a key is is not stored in the object; that is, `find(key) == end()`; example: `"key "the fast" not found"` @@ -3765,7 +3764,7 @@ class basic_json } else { - JSON_THROW(std::domain_error("cannot use at() with " + type_name())); + JSON_THROW(type_error(304, "cannot use at() with " + type_name())); } } @@ -3779,7 +3778,7 @@ class basic_json @return const reference to the element at key @a key - @throw std::domain_error if the JSON value is not an object; example: + @throw type_error.304 if the JSON value is not an object; example: `"cannot use at() with boolean"` @throw std::out_of_range if the key @a key is is not stored in the object; that is, `find(key) == end()`; example: `"key "the fast" not found"` @@ -3812,7 +3811,7 @@ class basic_json } else { - JSON_THROW(std::domain_error("cannot use at() with " + type_name())); + JSON_THROW(type_error(304, "cannot use at() with " + type_name())); } } diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 03d873af..1e136f2c 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -3255,7 +3255,7 @@ class basic_json @tparam ThisType will be deduced as `basic_json` or `const basic_json` - @throw std::domain_error if ReferenceType does not match underlying value + @throw type_error.303 if ReferenceType does not match underlying value type of the current JSON */ template @@ -3272,8 +3272,7 @@ class basic_json return *ptr; } - JSON_THROW(std::domain_error("incompatible ReferenceType for get_ref, actual type is " + - obj.type_name())); + JSON_THROW(type_error(303, "incompatible ReferenceType for get_ref, actual type is " + obj.type_name())); } public: @@ -3554,8 +3553,8 @@ class basic_json reference type @a ReferenceType fits to the JSON value; throws std::domain_error otherwise - @throw std::domain_error in case passed type @a ReferenceType is - incompatible with the stored JSON value + @throw type_error.303 in case passed type @a ReferenceType is incompatible + with the stored JSON value @complexity Constant. @@ -3646,8 +3645,8 @@ class basic_json @return reference to the element at index @a idx - @throw std::domain_error if the JSON value is not an array; example: - `"cannot use at() with string"` + @throw type_error.304 if the JSON value is not an array; example: `"cannot + use at() with string"` @throw std::out_of_range if the index @a idx is out of range of the array; that is, `idx >= size()`; example: `"array index 7 is out of range"` @@ -3675,7 +3674,7 @@ class basic_json } else { - JSON_THROW(std::domain_error("cannot use at() with " + type_name())); + JSON_THROW(type_error(304, "cannot use at() with " + type_name())); } } @@ -3689,8 +3688,8 @@ class basic_json @return const reference to the element at index @a idx - @throw std::domain_error if the JSON value is not an array; example: - `"cannot use at() with string"` + @throw type_error.304 if the JSON value is not an array; example: `"cannot + use at() with string"` @throw std::out_of_range if the index @a idx is out of range of the array; that is, `idx >= size()`; example: `"array index 7 is out of range"` @@ -3718,7 +3717,7 @@ class basic_json } else { - JSON_THROW(std::domain_error("cannot use at() with " + type_name())); + JSON_THROW(type_error(304, "cannot use at() with " + type_name())); } } @@ -3732,7 +3731,7 @@ class basic_json @return reference to the element at key @a key - @throw std::domain_error if the JSON value is not an object; example: + @throw type_error.304 if the JSON value is not an object; example: `"cannot use at() with boolean"` @throw std::out_of_range if the key @a key is is not stored in the object; that is, `find(key) == end()`; example: `"key "the fast" not found"` @@ -3765,7 +3764,7 @@ class basic_json } else { - JSON_THROW(std::domain_error("cannot use at() with " + type_name())); + JSON_THROW(type_error(304, "cannot use at() with " + type_name())); } } @@ -3779,7 +3778,7 @@ class basic_json @return const reference to the element at key @a key - @throw std::domain_error if the JSON value is not an object; example: + @throw type_error.304 if the JSON value is not an object; example: `"cannot use at() with boolean"` @throw std::out_of_range if the key @a key is is not stored in the object; that is, `find(key) == end()`; example: `"key "the fast" not found"` @@ -3812,7 +3811,7 @@ class basic_json } else { - JSON_THROW(std::domain_error("cannot use at() with " + type_name())); + JSON_THROW(type_error(304, "cannot use at() with " + type_name())); } } diff --git a/test/src/unit-element_access1.cpp b/test/src/unit-element_access1.cpp index a521e3c4..baea8f2a 100644 --- a/test/src/unit-element_access1.cpp +++ b/test/src/unit-element_access1.cpp @@ -76,77 +76,77 @@ TEST_CASE("element access 1") { json j_nonarray(json::value_t::null); const json j_nonarray_const(j_nonarray); - CHECK_THROWS_AS(j_nonarray.at(0), std::domain_error); - CHECK_THROWS_AS(j_nonarray_const.at(0), std::domain_error); + CHECK_THROWS_AS(j_nonarray.at(0), json::type_error); + CHECK_THROWS_AS(j_nonarray_const.at(0), json::type_error); - CHECK_THROWS_WITH(j_nonarray.at(0), "cannot use at() with null"); - CHECK_THROWS_WITH(j_nonarray_const.at(0), "cannot use at() with null"); + CHECK_THROWS_WITH(j_nonarray.at(0), "[json.exception.type_error.304] cannot use at() with null"); + CHECK_THROWS_WITH(j_nonarray_const.at(0), "[json.exception.type_error.304] cannot use at() with null"); } SECTION("boolean") { json j_nonarray(json::value_t::boolean); const json j_nonarray_const(j_nonarray); - CHECK_THROWS_AS(j_nonarray.at(0), std::domain_error); - CHECK_THROWS_AS(j_nonarray_const.at(0), std::domain_error); + CHECK_THROWS_AS(j_nonarray.at(0), json::type_error); + CHECK_THROWS_AS(j_nonarray_const.at(0), json::type_error); - CHECK_THROWS_WITH(j_nonarray.at(0), "cannot use at() with boolean"); - CHECK_THROWS_WITH(j_nonarray_const.at(0), "cannot use at() with boolean"); + CHECK_THROWS_WITH(j_nonarray.at(0), "[json.exception.type_error.304] cannot use at() with boolean"); + CHECK_THROWS_WITH(j_nonarray_const.at(0), "[json.exception.type_error.304] cannot use at() with boolean"); } SECTION("string") { json j_nonarray(json::value_t::string); const json j_nonarray_const(j_nonarray); - CHECK_THROWS_AS(j_nonarray.at(0), std::domain_error); - CHECK_THROWS_AS(j_nonarray_const.at(0), std::domain_error); + CHECK_THROWS_AS(j_nonarray.at(0), json::type_error); + CHECK_THROWS_AS(j_nonarray_const.at(0), json::type_error); - CHECK_THROWS_WITH(j_nonarray.at(0), "cannot use at() with string"); - CHECK_THROWS_WITH(j_nonarray_const.at(0), "cannot use at() with string"); + CHECK_THROWS_WITH(j_nonarray.at(0), "[json.exception.type_error.304] cannot use at() with string"); + CHECK_THROWS_WITH(j_nonarray_const.at(0), "[json.exception.type_error.304] cannot use at() with string"); } SECTION("object") { json j_nonarray(json::value_t::object); const json j_nonarray_const(j_nonarray); - CHECK_THROWS_AS(j_nonarray.at(0), std::domain_error); - CHECK_THROWS_AS(j_nonarray_const.at(0), std::domain_error); + CHECK_THROWS_AS(j_nonarray.at(0), json::type_error); + CHECK_THROWS_AS(j_nonarray_const.at(0), json::type_error); - CHECK_THROWS_WITH(j_nonarray.at(0), "cannot use at() with object"); - CHECK_THROWS_WITH(j_nonarray_const.at(0), "cannot use at() with object"); + CHECK_THROWS_WITH(j_nonarray.at(0), "[json.exception.type_error.304] cannot use at() with object"); + CHECK_THROWS_WITH(j_nonarray_const.at(0), "[json.exception.type_error.304] cannot use at() with object"); } SECTION("number (integer)") { json j_nonarray(json::value_t::number_integer); const json j_nonarray_const(j_nonarray); - CHECK_THROWS_AS(j_nonarray.at(0), std::domain_error); - CHECK_THROWS_AS(j_nonarray_const.at(0), std::domain_error); + CHECK_THROWS_AS(j_nonarray.at(0), json::type_error); + CHECK_THROWS_AS(j_nonarray_const.at(0), json::type_error); - CHECK_THROWS_WITH(j_nonarray.at(0), "cannot use at() with number"); - CHECK_THROWS_WITH(j_nonarray_const.at(0), "cannot use at() with number"); + CHECK_THROWS_WITH(j_nonarray.at(0), "[json.exception.type_error.304] cannot use at() with number"); + CHECK_THROWS_WITH(j_nonarray_const.at(0), "[json.exception.type_error.304] cannot use at() with number"); } SECTION("number (unsigned)") { json j_nonarray(json::value_t::number_unsigned); const json j_nonarray_const(j_nonarray); - CHECK_THROWS_AS(j_nonarray.at(0), std::domain_error); - CHECK_THROWS_AS(j_nonarray_const.at(0), std::domain_error); + CHECK_THROWS_AS(j_nonarray.at(0), json::type_error); + CHECK_THROWS_AS(j_nonarray_const.at(0), json::type_error); - CHECK_THROWS_WITH(j_nonarray.at(0), "cannot use at() with number"); - CHECK_THROWS_WITH(j_nonarray_const.at(0), "cannot use at() with number"); + CHECK_THROWS_WITH(j_nonarray.at(0), "[json.exception.type_error.304] cannot use at() with number"); + CHECK_THROWS_WITH(j_nonarray_const.at(0), "[json.exception.type_error.304] cannot use at() with number"); } SECTION("number (floating-point)") { json j_nonarray(json::value_t::number_float); const json j_nonarray_const(j_nonarray); - CHECK_THROWS_AS(j_nonarray.at(0), std::domain_error); - CHECK_THROWS_AS(j_nonarray_const.at(0), std::domain_error); + CHECK_THROWS_AS(j_nonarray.at(0), json::type_error); + CHECK_THROWS_AS(j_nonarray_const.at(0), json::type_error); - CHECK_THROWS_WITH(j_nonarray.at(0), "cannot use at() with number"); - CHECK_THROWS_WITH(j_nonarray_const.at(0), "cannot use at() with number"); + CHECK_THROWS_WITH(j_nonarray.at(0), "[json.exception.type_error.304] cannot use at() with number"); + CHECK_THROWS_WITH(j_nonarray_const.at(0), "[json.exception.type_error.304] cannot use at() with number"); } } } diff --git a/test/src/unit-element_access2.cpp b/test/src/unit-element_access2.cpp index 3c3122f1..c4de9d43 100644 --- a/test/src/unit-element_access2.cpp +++ b/test/src/unit-element_access2.cpp @@ -75,70 +75,70 @@ TEST_CASE("element access 2") { json j_nonobject(json::value_t::null); const json j_nonobject_const(j_nonobject); - CHECK_THROWS_AS(j_nonobject.at("foo"), std::domain_error); - CHECK_THROWS_AS(j_nonobject_const.at("foo"), std::domain_error); - CHECK_THROWS_WITH(j_nonobject.at("foo"), "cannot use at() with null"); - CHECK_THROWS_WITH(j_nonobject_const.at("foo"), "cannot use at() with null"); + CHECK_THROWS_AS(j_nonobject.at("foo"), json::type_error); + CHECK_THROWS_AS(j_nonobject_const.at("foo"), json::type_error); + CHECK_THROWS_WITH(j_nonobject.at("foo"), "[json.exception.type_error.304] cannot use at() with null"); + CHECK_THROWS_WITH(j_nonobject_const.at("foo"), "[json.exception.type_error.304] cannot use at() with null"); } SECTION("boolean") { json j_nonobject(json::value_t::boolean); const json j_nonobject_const(j_nonobject); - CHECK_THROWS_AS(j_nonobject.at("foo"), std::domain_error); - CHECK_THROWS_AS(j_nonobject_const.at("foo"), std::domain_error); - CHECK_THROWS_WITH(j_nonobject.at("foo"), "cannot use at() with boolean"); - CHECK_THROWS_WITH(j_nonobject_const.at("foo"), "cannot use at() with boolean"); + CHECK_THROWS_AS(j_nonobject.at("foo"), json::type_error); + CHECK_THROWS_AS(j_nonobject_const.at("foo"), json::type_error); + CHECK_THROWS_WITH(j_nonobject.at("foo"), "[json.exception.type_error.304] cannot use at() with boolean"); + CHECK_THROWS_WITH(j_nonobject_const.at("foo"), "[json.exception.type_error.304] cannot use at() with boolean"); } SECTION("string") { json j_nonobject(json::value_t::string); const json j_nonobject_const(j_nonobject); - CHECK_THROWS_AS(j_nonobject.at("foo"), std::domain_error); - CHECK_THROWS_AS(j_nonobject_const.at("foo"), std::domain_error); - CHECK_THROWS_WITH(j_nonobject.at("foo"), "cannot use at() with string"); - CHECK_THROWS_WITH(j_nonobject_const.at("foo"), "cannot use at() with string"); + CHECK_THROWS_AS(j_nonobject.at("foo"), json::type_error); + CHECK_THROWS_AS(j_nonobject_const.at("foo"), json::type_error); + CHECK_THROWS_WITH(j_nonobject.at("foo"), "[json.exception.type_error.304] cannot use at() with string"); + CHECK_THROWS_WITH(j_nonobject_const.at("foo"), "[json.exception.type_error.304] cannot use at() with string"); } SECTION("array") { json j_nonobject(json::value_t::array); const json j_nonobject_const(j_nonobject); - CHECK_THROWS_AS(j_nonobject.at("foo"), std::domain_error); - CHECK_THROWS_AS(j_nonobject_const.at("foo"), std::domain_error); - CHECK_THROWS_WITH(j_nonobject.at("foo"), "cannot use at() with array"); - CHECK_THROWS_WITH(j_nonobject_const.at("foo"), "cannot use at() with array"); + CHECK_THROWS_AS(j_nonobject.at("foo"), json::type_error); + CHECK_THROWS_AS(j_nonobject_const.at("foo"), json::type_error); + CHECK_THROWS_WITH(j_nonobject.at("foo"), "[json.exception.type_error.304] cannot use at() with array"); + CHECK_THROWS_WITH(j_nonobject_const.at("foo"), "[json.exception.type_error.304] cannot use at() with array"); } SECTION("number (integer)") { json j_nonobject(json::value_t::number_integer); const json j_nonobject_const(j_nonobject); - CHECK_THROWS_AS(j_nonobject.at("foo"), std::domain_error); - CHECK_THROWS_AS(j_nonobject_const.at("foo"), std::domain_error); - CHECK_THROWS_WITH(j_nonobject.at("foo"), "cannot use at() with number"); - CHECK_THROWS_WITH(j_nonobject_const.at("foo"), "cannot use at() with number"); + CHECK_THROWS_AS(j_nonobject.at("foo"), json::type_error); + CHECK_THROWS_AS(j_nonobject_const.at("foo"), json::type_error); + CHECK_THROWS_WITH(j_nonobject.at("foo"), "[json.exception.type_error.304] cannot use at() with number"); + CHECK_THROWS_WITH(j_nonobject_const.at("foo"), "[json.exception.type_error.304] cannot use at() with number"); } SECTION("number (unsigned)") { json j_nonobject(json::value_t::number_unsigned); const json j_nonobject_const(j_nonobject); - CHECK_THROWS_AS(j_nonobject.at("foo"), std::domain_error); - CHECK_THROWS_AS(j_nonobject_const.at("foo"), std::domain_error); - CHECK_THROWS_WITH(j_nonobject.at("foo"), "cannot use at() with number"); - CHECK_THROWS_WITH(j_nonobject_const.at("foo"), "cannot use at() with number"); + CHECK_THROWS_AS(j_nonobject.at("foo"), json::type_error); + CHECK_THROWS_AS(j_nonobject_const.at("foo"), json::type_error); + CHECK_THROWS_WITH(j_nonobject.at("foo"), "[json.exception.type_error.304] cannot use at() with number"); + CHECK_THROWS_WITH(j_nonobject_const.at("foo"), "[json.exception.type_error.304] cannot use at() with number"); } SECTION("number (floating-point)") { json j_nonobject(json::value_t::number_float); const json j_nonobject_const(j_nonobject); - CHECK_THROWS_AS(j_nonobject.at("foo"), std::domain_error); - CHECK_THROWS_AS(j_nonobject_const.at("foo"), std::domain_error); - CHECK_THROWS_WITH(j_nonobject.at("foo"), "cannot use at() with number"); - CHECK_THROWS_WITH(j_nonobject_const.at("foo"), "cannot use at() with number"); + CHECK_THROWS_AS(j_nonobject.at("foo"), json::type_error); + CHECK_THROWS_AS(j_nonobject_const.at("foo"), json::type_error); + CHECK_THROWS_WITH(j_nonobject.at("foo"), "[json.exception.type_error.304] cannot use at() with number"); + CHECK_THROWS_WITH(j_nonobject_const.at("foo"), "[json.exception.type_error.304] cannot use at() with number"); } } } From aa842b4a27e9db9e624d4eb8a6da69a27589b6cc Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 5 Mar 2017 19:08:12 +0100 Subject: [PATCH 033/530] :hammer: added user-defined exceptions 305 --- src/json.hpp | 34 ++++----- src/json.hpp.re2c | 34 ++++----- test/src/unit-element_access1.cpp | 52 +++++++------- test/src/unit-element_access2.cpp | 116 ++++++++++++++++-------------- 4 files changed, 124 insertions(+), 112 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index f51c85ad..f4bca23d 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -3828,8 +3828,8 @@ class basic_json @return reference to the element at index @a idx - @throw std::domain_error if JSON is not an array or null; example: - `"cannot use operator[] with string"` + @throw type_error.305 if JSON is not an array or null; example: `"cannot + use operator[] with string" @complexity Constant if @a idx is in the range of the array. Otherwise linear in `idx - size()`. @@ -3864,7 +3864,7 @@ class basic_json return m_value.array->operator[](idx); } - JSON_THROW(std::domain_error("cannot use operator[] with " + type_name())); + JSON_THROW(type_error(305, "cannot use operator[] with " + type_name())); } /*! @@ -3876,7 +3876,7 @@ class basic_json @return const reference to the element at index @a idx - @throw std::domain_error if JSON is not an array; example: `"cannot use + @throw type_error.305 if JSON is not an array; example: `"cannot use operator[] with null"` @complexity Constant. @@ -3894,7 +3894,7 @@ class basic_json return m_value.array->operator[](idx); } - JSON_THROW(std::domain_error("cannot use operator[] with " + type_name())); + JSON_THROW(type_error(305, "cannot use operator[] with " + type_name())); } /*! @@ -3910,8 +3910,8 @@ class basic_json @return reference to the element at key @a key - @throw std::domain_error if JSON is not an object or null; example: - `"cannot use operator[] with string"` + @throw type_error.305 if JSON is not an object or null; example: `"cannot + use operator[] with string"` @complexity Logarithmic in the size of the container. @@ -3940,7 +3940,7 @@ class basic_json return m_value.object->operator[](key); } - JSON_THROW(std::domain_error("cannot use operator[] with " + type_name())); + JSON_THROW(type_error(305, "cannot use operator[] with " + type_name())); } /*! @@ -3959,7 +3959,7 @@ class basic_json @pre The element with key @a key must exist. **This precondition is enforced with an assertion.** - @throw std::domain_error if JSON is not an object; example: `"cannot use + @throw type_error.305 if JSON is not an object; example: `"cannot use operator[] with null"` @complexity Logarithmic in the size of the container. @@ -3982,7 +3982,7 @@ class basic_json return m_value.object->find(key)->second; } - JSON_THROW(std::domain_error("cannot use operator[] with " + type_name())); + JSON_THROW(type_error(305, "cannot use operator[] with " + type_name())); } /*! @@ -3998,8 +3998,8 @@ class basic_json @return reference to the element at key @a key - @throw std::domain_error if JSON is not an object or null; example: - `"cannot use operator[] with string"` + @throw type_error.305 if JSON is not an object or null; example: `"cannot + use operator[] with string"` @complexity Logarithmic in the size of the container. @@ -4066,8 +4066,8 @@ class basic_json @return reference to the element at key @a key - @throw std::domain_error if JSON is not an object or null; example: - `"cannot use operator[] with string"` + @throw type_error.305 if JSON is not an object or null; example: `"cannot + use operator[] with string"` @complexity Logarithmic in the size of the container. @@ -4097,7 +4097,7 @@ class basic_json return m_value.object->operator[](key); } - JSON_THROW(std::domain_error("cannot use operator[] with " + type_name())); + JSON_THROW(type_error(305, "cannot use operator[] with " + type_name())); } /*! @@ -4116,7 +4116,7 @@ class basic_json @pre The element with key @a key must exist. **This precondition is enforced with an assertion.** - @throw std::domain_error if JSON is not an object; example: `"cannot use + @throw type_error.305 if JSON is not an object; example: `"cannot use operator[] with null"` @complexity Logarithmic in the size of the container. @@ -4140,7 +4140,7 @@ class basic_json return m_value.object->find(key)->second; } - JSON_THROW(std::domain_error("cannot use operator[] with " + type_name())); + JSON_THROW(type_error(305, "cannot use operator[] with " + type_name())); } /*! diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 1e136f2c..4ba097af 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -3828,8 +3828,8 @@ class basic_json @return reference to the element at index @a idx - @throw std::domain_error if JSON is not an array or null; example: - `"cannot use operator[] with string"` + @throw type_error.305 if JSON is not an array or null; example: `"cannot + use operator[] with string" @complexity Constant if @a idx is in the range of the array. Otherwise linear in `idx - size()`. @@ -3864,7 +3864,7 @@ class basic_json return m_value.array->operator[](idx); } - JSON_THROW(std::domain_error("cannot use operator[] with " + type_name())); + JSON_THROW(type_error(305, "cannot use operator[] with " + type_name())); } /*! @@ -3876,7 +3876,7 @@ class basic_json @return const reference to the element at index @a idx - @throw std::domain_error if JSON is not an array; example: `"cannot use + @throw type_error.305 if JSON is not an array; example: `"cannot use operator[] with null"` @complexity Constant. @@ -3894,7 +3894,7 @@ class basic_json return m_value.array->operator[](idx); } - JSON_THROW(std::domain_error("cannot use operator[] with " + type_name())); + JSON_THROW(type_error(305, "cannot use operator[] with " + type_name())); } /*! @@ -3910,8 +3910,8 @@ class basic_json @return reference to the element at key @a key - @throw std::domain_error if JSON is not an object or null; example: - `"cannot use operator[] with string"` + @throw type_error.305 if JSON is not an object or null; example: `"cannot + use operator[] with string"` @complexity Logarithmic in the size of the container. @@ -3940,7 +3940,7 @@ class basic_json return m_value.object->operator[](key); } - JSON_THROW(std::domain_error("cannot use operator[] with " + type_name())); + JSON_THROW(type_error(305, "cannot use operator[] with " + type_name())); } /*! @@ -3959,7 +3959,7 @@ class basic_json @pre The element with key @a key must exist. **This precondition is enforced with an assertion.** - @throw std::domain_error if JSON is not an object; example: `"cannot use + @throw type_error.305 if JSON is not an object; example: `"cannot use operator[] with null"` @complexity Logarithmic in the size of the container. @@ -3982,7 +3982,7 @@ class basic_json return m_value.object->find(key)->second; } - JSON_THROW(std::domain_error("cannot use operator[] with " + type_name())); + JSON_THROW(type_error(305, "cannot use operator[] with " + type_name())); } /*! @@ -3998,8 +3998,8 @@ class basic_json @return reference to the element at key @a key - @throw std::domain_error if JSON is not an object or null; example: - `"cannot use operator[] with string"` + @throw type_error.305 if JSON is not an object or null; example: `"cannot + use operator[] with string"` @complexity Logarithmic in the size of the container. @@ -4066,8 +4066,8 @@ class basic_json @return reference to the element at key @a key - @throw std::domain_error if JSON is not an object or null; example: - `"cannot use operator[] with string"` + @throw type_error.305 if JSON is not an object or null; example: `"cannot + use operator[] with string"` @complexity Logarithmic in the size of the container. @@ -4097,7 +4097,7 @@ class basic_json return m_value.object->operator[](key); } - JSON_THROW(std::domain_error("cannot use operator[] with " + type_name())); + JSON_THROW(type_error(305, "cannot use operator[] with " + type_name())); } /*! @@ -4116,7 +4116,7 @@ class basic_json @pre The element with key @a key must exist. **This precondition is enforced with an assertion.** - @throw std::domain_error if JSON is not an object; example: `"cannot use + @throw type_error.305 if JSON is not an object; example: `"cannot use operator[] with null"` @complexity Logarithmic in the size of the container. @@ -4140,7 +4140,7 @@ class basic_json return m_value.object->find(key)->second; } - JSON_THROW(std::domain_error("cannot use operator[] with " + type_name())); + JSON_THROW(type_error(305, "cannot use operator[] with " + type_name())); } /*! diff --git a/test/src/unit-element_access1.cpp b/test/src/unit-element_access1.cpp index baea8f2a..37c138bb 100644 --- a/test/src/unit-element_access1.cpp +++ b/test/src/unit-element_access1.cpp @@ -191,8 +191,8 @@ TEST_CASE("element access 1") json j_nonarray(json::value_t::null); const json j_nonarray_const(j_nonarray); CHECK_NOTHROW(j_nonarray[0]); - CHECK_THROWS_AS(j_nonarray_const[0], std::domain_error); - CHECK_THROWS_WITH(j_nonarray_const[0], "cannot use operator[] with null"); + CHECK_THROWS_AS(j_nonarray_const[0], json::type_error); + CHECK_THROWS_WITH(j_nonarray_const[0], "[json.exception.type_error.305] cannot use operator[] with null"); } SECTION("implicit transformation to properly filled array") @@ -207,60 +207,60 @@ TEST_CASE("element access 1") { json j_nonarray(json::value_t::boolean); const json j_nonarray_const(j_nonarray); - CHECK_THROWS_AS(j_nonarray[0], std::domain_error); - CHECK_THROWS_AS(j_nonarray_const[0], std::domain_error); - CHECK_THROWS_WITH(j_nonarray[0], "cannot use operator[] with boolean"); - CHECK_THROWS_WITH(j_nonarray_const[0], "cannot use operator[] with boolean"); + CHECK_THROWS_AS(j_nonarray[0], json::type_error); + CHECK_THROWS_AS(j_nonarray_const[0], json::type_error); + CHECK_THROWS_WITH(j_nonarray[0], "[json.exception.type_error.305] cannot use operator[] with boolean"); + CHECK_THROWS_WITH(j_nonarray_const[0], "[json.exception.type_error.305] cannot use operator[] with boolean"); } SECTION("string") { json j_nonarray(json::value_t::string); const json j_nonarray_const(j_nonarray); - CHECK_THROWS_AS(j_nonarray[0], std::domain_error); - CHECK_THROWS_AS(j_nonarray_const[0], std::domain_error); - CHECK_THROWS_WITH(j_nonarray[0], "cannot use operator[] with string"); - CHECK_THROWS_WITH(j_nonarray_const[0], "cannot use operator[] with string"); + CHECK_THROWS_AS(j_nonarray[0], json::type_error); + CHECK_THROWS_AS(j_nonarray_const[0], json::type_error); + CHECK_THROWS_WITH(j_nonarray[0], "[json.exception.type_error.305] cannot use operator[] with string"); + CHECK_THROWS_WITH(j_nonarray_const[0], "[json.exception.type_error.305] cannot use operator[] with string"); } SECTION("object") { json j_nonarray(json::value_t::object); const json j_nonarray_const(j_nonarray); - CHECK_THROWS_AS(j_nonarray[0], std::domain_error); - CHECK_THROWS_AS(j_nonarray_const[0], std::domain_error); - CHECK_THROWS_WITH(j_nonarray[0], "cannot use operator[] with object"); - CHECK_THROWS_WITH(j_nonarray_const[0], "cannot use operator[] with object"); + CHECK_THROWS_AS(j_nonarray[0], json::type_error); + CHECK_THROWS_AS(j_nonarray_const[0], json::type_error); + CHECK_THROWS_WITH(j_nonarray[0], "[json.exception.type_error.305] cannot use operator[] with object"); + CHECK_THROWS_WITH(j_nonarray_const[0], "[json.exception.type_error.305] cannot use operator[] with object"); } SECTION("number (integer)") { json j_nonarray(json::value_t::number_integer); const json j_nonarray_const(j_nonarray); - CHECK_THROWS_AS(j_nonarray[0], std::domain_error); - CHECK_THROWS_AS(j_nonarray_const[0], std::domain_error); - CHECK_THROWS_WITH(j_nonarray[0], "cannot use operator[] with number"); - CHECK_THROWS_WITH(j_nonarray_const[0], "cannot use operator[] with number"); + CHECK_THROWS_AS(j_nonarray[0], json::type_error); + CHECK_THROWS_AS(j_nonarray_const[0], json::type_error); + CHECK_THROWS_WITH(j_nonarray[0], "[json.exception.type_error.305] cannot use operator[] with number"); + CHECK_THROWS_WITH(j_nonarray_const[0], "[json.exception.type_error.305] cannot use operator[] with number"); } SECTION("number (unsigned)") { json j_nonarray(json::value_t::number_unsigned); const json j_nonarray_const(j_nonarray); - CHECK_THROWS_AS(j_nonarray[0], std::domain_error); - CHECK_THROWS_AS(j_nonarray_const[0], std::domain_error); - CHECK_THROWS_WITH(j_nonarray[0], "cannot use operator[] with number"); - CHECK_THROWS_WITH(j_nonarray_const[0], "cannot use operator[] with number"); + CHECK_THROWS_AS(j_nonarray[0], json::type_error); + CHECK_THROWS_AS(j_nonarray_const[0], json::type_error); + CHECK_THROWS_WITH(j_nonarray[0], "[json.exception.type_error.305] cannot use operator[] with number"); + CHECK_THROWS_WITH(j_nonarray_const[0], "[json.exception.type_error.305] cannot use operator[] with number"); } SECTION("number (floating-point)") { json j_nonarray(json::value_t::number_float); const json j_nonarray_const(j_nonarray); - CHECK_THROWS_AS(j_nonarray[0], std::domain_error); - CHECK_THROWS_AS(j_nonarray_const[0], std::domain_error); - CHECK_THROWS_WITH(j_nonarray[0], "cannot use operator[] with number"); - CHECK_THROWS_WITH(j_nonarray_const[0], "cannot use operator[] with number"); + CHECK_THROWS_AS(j_nonarray[0], json::type_error); + CHECK_THROWS_AS(j_nonarray_const[0], json::type_error); + CHECK_THROWS_WITH(j_nonarray[0], "[json.exception.type_error.305] cannot use operator[] with number"); + CHECK_THROWS_WITH(j_nonarray_const[0], "[json.exception.type_error.305] cannot use operator[] with number"); } } } diff --git a/test/src/unit-element_access2.cpp b/test/src/unit-element_access2.cpp index c4de9d43..4e34ad92 100644 --- a/test/src/unit-element_access2.cpp +++ b/test/src/unit-element_access2.cpp @@ -447,106 +447,118 @@ TEST_CASE("element access 2") const json j_const_nonobject(j_nonobject); CHECK_NOTHROW(j_nonobject["foo"]); CHECK_NOTHROW(j_nonobject2[json::object_t::key_type("foo")]); - CHECK_THROWS_AS(j_const_nonobject["foo"], std::domain_error); - CHECK_THROWS_AS(j_const_nonobject[json::object_t::key_type("foo")], std::domain_error); - CHECK_THROWS_WITH(j_const_nonobject["foo"], "cannot use operator[] with null"); + CHECK_THROWS_AS(j_const_nonobject["foo"], json::type_error); + CHECK_THROWS_AS(j_const_nonobject[json::object_t::key_type("foo")], json::type_error); + CHECK_THROWS_WITH(j_const_nonobject["foo"], "[json.exception.type_error.305] cannot use operator[] with null"); CHECK_THROWS_WITH(j_const_nonobject[json::object_t::key_type("foo")], - "cannot use operator[] with null"); + "[json.exception.type_error.305] cannot use operator[] with null"); } SECTION("boolean") { json j_nonobject(json::value_t::boolean); const json j_const_nonobject(j_nonobject); - CHECK_THROWS_AS(j_nonobject["foo"], std::domain_error); - CHECK_THROWS_AS(j_nonobject[json::object_t::key_type("foo")], std::domain_error); - CHECK_THROWS_AS(j_const_nonobject["foo"], std::domain_error); - CHECK_THROWS_AS(j_const_nonobject[json::object_t::key_type("foo")], std::domain_error); - CHECK_THROWS_WITH(j_nonobject["foo"], "cannot use operator[] with boolean"); + CHECK_THROWS_AS(j_nonobject["foo"], json::type_error); + CHECK_THROWS_AS(j_nonobject[json::object_t::key_type("foo")], json::type_error); + CHECK_THROWS_AS(j_const_nonobject["foo"], json::type_error); + CHECK_THROWS_AS(j_const_nonobject[json::object_t::key_type("foo")], json::type_error); + CHECK_THROWS_WITH(j_nonobject["foo"], + "[json.exception.type_error.305] cannot use operator[] with boolean"); CHECK_THROWS_WITH(j_nonobject[json::object_t::key_type("foo")], - "cannot use operator[] with boolean"); - CHECK_THROWS_WITH(j_const_nonobject["foo"], "cannot use operator[] with boolean"); + "[json.exception.type_error.305] cannot use operator[] with boolean"); + CHECK_THROWS_WITH(j_const_nonobject["foo"], + "[json.exception.type_error.305] cannot use operator[] with boolean"); CHECK_THROWS_WITH(j_const_nonobject[json::object_t::key_type("foo")], - "cannot use operator[] with boolean"); + "[json.exception.type_error.305] cannot use operator[] with boolean"); } SECTION("string") { json j_nonobject(json::value_t::string); const json j_const_nonobject(j_nonobject); - CHECK_THROWS_AS(j_nonobject["foo"], std::domain_error); - CHECK_THROWS_AS(j_nonobject[json::object_t::key_type("foo")], std::domain_error); - CHECK_THROWS_AS(j_const_nonobject["foo"], std::domain_error); - CHECK_THROWS_AS(j_const_nonobject[json::object_t::key_type("foo")], std::domain_error); - CHECK_THROWS_WITH(j_nonobject["foo"], "cannot use operator[] with string"); + CHECK_THROWS_AS(j_nonobject["foo"], json::type_error); + CHECK_THROWS_AS(j_nonobject[json::object_t::key_type("foo")], json::type_error); + CHECK_THROWS_AS(j_const_nonobject["foo"], json::type_error); + CHECK_THROWS_AS(j_const_nonobject[json::object_t::key_type("foo")], json::type_error); + CHECK_THROWS_WITH(j_nonobject["foo"], + "[json.exception.type_error.305] cannot use operator[] with string"); CHECK_THROWS_WITH(j_nonobject[json::object_t::key_type("foo")], - "cannot use operator[] with string"); - CHECK_THROWS_WITH(j_const_nonobject["foo"], "cannot use operator[] with string"); + "[json.exception.type_error.305] cannot use operator[] with string"); + CHECK_THROWS_WITH(j_const_nonobject["foo"], + "[json.exception.type_error.305] cannot use operator[] with string"); CHECK_THROWS_WITH(j_const_nonobject[json::object_t::key_type("foo")], - "cannot use operator[] with string"); + "[json.exception.type_error.305] cannot use operator[] with string"); } SECTION("array") { json j_nonobject(json::value_t::array); const json j_const_nonobject(j_nonobject); - CHECK_THROWS_AS(j_nonobject["foo"], std::domain_error); - CHECK_THROWS_AS(j_nonobject[json::object_t::key_type("foo")], std::domain_error); - CHECK_THROWS_AS(j_const_nonobject["foo"], std::domain_error); - CHECK_THROWS_AS(j_const_nonobject[json::object_t::key_type("foo")], std::domain_error); - CHECK_THROWS_WITH(j_nonobject["foo"], "cannot use operator[] with array"); - CHECK_THROWS_WITH(j_nonobject[json::object_t::key_type("foo")], "cannot use operator[] with array"); - CHECK_THROWS_WITH(j_const_nonobject["foo"], "cannot use operator[] with array"); + CHECK_THROWS_AS(j_nonobject["foo"], json::type_error); + CHECK_THROWS_AS(j_nonobject[json::object_t::key_type("foo")], json::type_error); + CHECK_THROWS_AS(j_const_nonobject["foo"], json::type_error); + CHECK_THROWS_AS(j_const_nonobject[json::object_t::key_type("foo")], json::type_error); + CHECK_THROWS_WITH(j_nonobject["foo"], + "[json.exception.type_error.305] cannot use operator[] with array"); + CHECK_THROWS_WITH(j_nonobject[json::object_t::key_type("foo")], "[json.exception.type_error.305] cannot use operator[] with array"); + CHECK_THROWS_WITH(j_const_nonobject["foo"], + "[json.exception.type_error.305] cannot use operator[] with array"); CHECK_THROWS_WITH(j_const_nonobject[json::object_t::key_type("foo")], - "cannot use operator[] with array"); + "[json.exception.type_error.305] cannot use operator[] with array"); } SECTION("number (integer)") { json j_nonobject(json::value_t::number_integer); const json j_const_nonobject(j_nonobject); - CHECK_THROWS_AS(j_nonobject["foo"], std::domain_error); - CHECK_THROWS_AS(j_nonobject[json::object_t::key_type("foo")], std::domain_error); - CHECK_THROWS_AS(j_const_nonobject["foo"], std::domain_error); - CHECK_THROWS_AS(j_const_nonobject[json::object_t::key_type("foo")], std::domain_error); - CHECK_THROWS_WITH(j_nonobject["foo"], "cannot use operator[] with number"); + CHECK_THROWS_AS(j_nonobject["foo"], json::type_error); + CHECK_THROWS_AS(j_nonobject[json::object_t::key_type("foo")], json::type_error); + CHECK_THROWS_AS(j_const_nonobject["foo"], json::type_error); + CHECK_THROWS_AS(j_const_nonobject[json::object_t::key_type("foo")], json::type_error); + CHECK_THROWS_WITH(j_nonobject["foo"], + "[json.exception.type_error.305] cannot use operator[] with number"); CHECK_THROWS_WITH(j_nonobject[json::object_t::key_type("foo")], - "cannot use operator[] with number"); - CHECK_THROWS_WITH(j_const_nonobject["foo"], "cannot use operator[] with number"); + "[json.exception.type_error.305] cannot use operator[] with number"); + CHECK_THROWS_WITH(j_const_nonobject["foo"], + "[json.exception.type_error.305] cannot use operator[] with number"); CHECK_THROWS_WITH(j_const_nonobject[json::object_t::key_type("foo")], - "cannot use operator[] with number"); + "[json.exception.type_error.305] cannot use operator[] with number"); } SECTION("number (unsigned)") { json j_nonobject(json::value_t::number_unsigned); const json j_const_nonobject(j_nonobject); - CHECK_THROWS_AS(j_nonobject["foo"], std::domain_error); - CHECK_THROWS_AS(j_nonobject[json::object_t::key_type("foo")], std::domain_error); - CHECK_THROWS_AS(j_const_nonobject["foo"], std::domain_error); - CHECK_THROWS_AS(j_const_nonobject[json::object_t::key_type("foo")], std::domain_error); - CHECK_THROWS_WITH(j_nonobject["foo"], "cannot use operator[] with number"); + CHECK_THROWS_AS(j_nonobject["foo"], json::type_error); + CHECK_THROWS_AS(j_nonobject[json::object_t::key_type("foo")], json::type_error); + CHECK_THROWS_AS(j_const_nonobject["foo"], json::type_error); + CHECK_THROWS_AS(j_const_nonobject[json::object_t::key_type("foo")], json::type_error); + CHECK_THROWS_WITH(j_nonobject["foo"], + "[json.exception.type_error.305] cannot use operator[] with number"); CHECK_THROWS_WITH(j_nonobject[json::object_t::key_type("foo")], - "cannot use operator[] with number"); - CHECK_THROWS_WITH(j_const_nonobject["foo"], "cannot use operator[] with number"); + "[json.exception.type_error.305] cannot use operator[] with number"); + CHECK_THROWS_WITH(j_const_nonobject["foo"], + "[json.exception.type_error.305] cannot use operator[] with number"); CHECK_THROWS_WITH(j_const_nonobject[json::object_t::key_type("foo")], - "cannot use operator[] with number"); + "[json.exception.type_error.305] cannot use operator[] with number"); } SECTION("number (floating-point)") { json j_nonobject(json::value_t::number_float); const json j_const_nonobject(j_nonobject); - CHECK_THROWS_AS(j_nonobject["foo"], std::domain_error); - CHECK_THROWS_AS(j_nonobject[json::object_t::key_type("foo")], std::domain_error); - CHECK_THROWS_AS(j_const_nonobject["foo"], std::domain_error); - CHECK_THROWS_AS(j_const_nonobject[json::object_t::key_type("foo")], std::domain_error); - CHECK_THROWS_WITH(j_nonobject["foo"], "cannot use operator[] with number"); + CHECK_THROWS_AS(j_nonobject["foo"], json::type_error); + CHECK_THROWS_AS(j_nonobject[json::object_t::key_type("foo")], json::type_error); + CHECK_THROWS_AS(j_const_nonobject["foo"], json::type_error); + CHECK_THROWS_AS(j_const_nonobject[json::object_t::key_type("foo")], json::type_error); + CHECK_THROWS_WITH(j_nonobject["foo"], + "[json.exception.type_error.305] cannot use operator[] with number"); CHECK_THROWS_WITH(j_nonobject[json::object_t::key_type("foo")], - "cannot use operator[] with number"); - CHECK_THROWS_WITH(j_const_nonobject["foo"], "cannot use operator[] with number"); + "[json.exception.type_error.305] cannot use operator[] with number"); + CHECK_THROWS_WITH(j_const_nonobject["foo"], + "[json.exception.type_error.305] cannot use operator[] with number"); CHECK_THROWS_WITH(j_const_nonobject[json::object_t::key_type("foo")], - "cannot use operator[] with number"); + "[json.exception.type_error.305] cannot use operator[] with number"); } } } From bb740c43fba356cd68caee5e7c5a9611ec99de1c Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 5 Mar 2017 19:15:56 +0100 Subject: [PATCH 034/530] :hammer: added user-defined exceptions 306 --- src/json.hpp | 8 +- src/json.hpp.re2c | 8 +- test/src/unit-element_access2.cpp | 135 +++++++++++++++++------------- 3 files changed, 87 insertions(+), 64 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index f4bca23d..e9654577 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -4176,7 +4176,7 @@ class basic_json @return copy of the element at key @a key or @a default_value if @a key is not found - @throw std::domain_error if JSON is not an object; example: `"cannot use + @throw type_error.306 if JSON is not an object; example: `"cannot use value() with null"` @complexity Logarithmic in the size of the container. @@ -4209,7 +4209,7 @@ class basic_json } else { - JSON_THROW(std::domain_error("cannot use value() with " + type_name())); + JSON_THROW(type_error(306, "cannot use value() with " + type_name())); } } @@ -4251,7 +4251,7 @@ class basic_json @return copy of the element at key @a key or @a default_value if @a key is not found - @throw std::domain_error if JSON is not an object; example: `"cannot use + @throw type_error.306 if JSON is not an object; example: `"cannot use value() with null"` @complexity Logarithmic in the size of the container. @@ -4281,7 +4281,7 @@ class basic_json } } - JSON_THROW(std::domain_error("cannot use value() with " + type_name())); + JSON_THROW(type_error(306, "cannot use value() with " + type_name())); } /*! diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 4ba097af..d21dfdfa 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -4176,7 +4176,7 @@ class basic_json @return copy of the element at key @a key or @a default_value if @a key is not found - @throw std::domain_error if JSON is not an object; example: `"cannot use + @throw type_error.306 if JSON is not an object; example: `"cannot use value() with null"` @complexity Logarithmic in the size of the container. @@ -4209,7 +4209,7 @@ class basic_json } else { - JSON_THROW(std::domain_error("cannot use value() with " + type_name())); + JSON_THROW(type_error(306, "cannot use value() with " + type_name())); } } @@ -4251,7 +4251,7 @@ class basic_json @return copy of the element at key @a key or @a default_value if @a key is not found - @throw std::domain_error if JSON is not an object; example: `"cannot use + @throw type_error.306 if JSON is not an object; example: `"cannot use value() with null"` @complexity Logarithmic in the size of the container. @@ -4281,7 +4281,7 @@ class basic_json } } - JSON_THROW(std::domain_error("cannot use value() with " + type_name())); + JSON_THROW(type_error(306, "cannot use value() with " + type_name())); } /*! diff --git a/test/src/unit-element_access2.cpp b/test/src/unit-element_access2.cpp index 4e34ad92..ec70d8df 100644 --- a/test/src/unit-element_access2.cpp +++ b/test/src/unit-element_access2.cpp @@ -200,70 +200,84 @@ TEST_CASE("element access 2") { json j_nonobject(json::value_t::null); const json j_nonobject_const(j_nonobject); - CHECK_THROWS_AS(j_nonobject.value("foo", 1), std::domain_error); - CHECK_THROWS_AS(j_nonobject_const.value("foo", 1), std::domain_error); - CHECK_THROWS_WITH(j_nonobject.value("foo", 1), "cannot use value() with null"); - CHECK_THROWS_WITH(j_nonobject_const.value("foo", 1), "cannot use value() with null"); + CHECK_THROWS_AS(j_nonobject.value("foo", 1), json::type_error); + CHECK_THROWS_AS(j_nonobject_const.value("foo", 1), json::type_error); + CHECK_THROWS_WITH(j_nonobject.value("foo", 1), + "[json.exception.type_error.306] cannot use value() with null"); + CHECK_THROWS_WITH(j_nonobject_const.value("foo", 1), + "[json.exception.type_error.306] cannot use value() with null"); } SECTION("boolean") { json j_nonobject(json::value_t::boolean); const json j_nonobject_const(j_nonobject); - CHECK_THROWS_AS(j_nonobject.value("foo", 1), std::domain_error); - CHECK_THROWS_AS(j_nonobject_const.value("foo", 1), std::domain_error); - CHECK_THROWS_WITH(j_nonobject.value("foo", 1), "cannot use value() with boolean"); - CHECK_THROWS_WITH(j_nonobject_const.value("foo", 1), "cannot use value() with boolean"); + CHECK_THROWS_AS(j_nonobject.value("foo", 1), json::type_error); + CHECK_THROWS_AS(j_nonobject_const.value("foo", 1), json::type_error); + CHECK_THROWS_WITH(j_nonobject.value("foo", 1), + "[json.exception.type_error.306] cannot use value() with boolean"); + CHECK_THROWS_WITH(j_nonobject_const.value("foo", 1), + "[json.exception.type_error.306] cannot use value() with boolean"); } SECTION("string") { json j_nonobject(json::value_t::string); const json j_nonobject_const(j_nonobject); - CHECK_THROWS_AS(j_nonobject.value("foo", 1), std::domain_error); - CHECK_THROWS_AS(j_nonobject_const.value("foo", 1), std::domain_error); - CHECK_THROWS_WITH(j_nonobject.value("foo", 1), "cannot use value() with string"); - CHECK_THROWS_WITH(j_nonobject_const.value("foo", 1), "cannot use value() with string"); + CHECK_THROWS_AS(j_nonobject.value("foo", 1), json::type_error); + CHECK_THROWS_AS(j_nonobject_const.value("foo", 1), json::type_error); + CHECK_THROWS_WITH(j_nonobject.value("foo", 1), + "[json.exception.type_error.306] cannot use value() with string"); + CHECK_THROWS_WITH(j_nonobject_const.value("foo", 1), + "[json.exception.type_error.306] cannot use value() with string"); } SECTION("array") { json j_nonobject(json::value_t::array); const json j_nonobject_const(j_nonobject); - CHECK_THROWS_AS(j_nonobject.value("foo", 1), std::domain_error); - CHECK_THROWS_AS(j_nonobject_const.value("foo", 1), std::domain_error); - CHECK_THROWS_WITH(j_nonobject.value("foo", 1), "cannot use value() with array"); - CHECK_THROWS_WITH(j_nonobject_const.value("foo", 1), "cannot use value() with array"); + CHECK_THROWS_AS(j_nonobject.value("foo", 1), json::type_error); + CHECK_THROWS_AS(j_nonobject_const.value("foo", 1), json::type_error); + CHECK_THROWS_WITH(j_nonobject.value("foo", 1), + "[json.exception.type_error.306] cannot use value() with array"); + CHECK_THROWS_WITH(j_nonobject_const.value("foo", 1), + "[json.exception.type_error.306] cannot use value() with array"); } SECTION("number (integer)") { json j_nonobject(json::value_t::number_integer); const json j_nonobject_const(j_nonobject); - CHECK_THROWS_AS(j_nonobject.value("foo", 1), std::domain_error); - CHECK_THROWS_AS(j_nonobject_const.value("foo", 1), std::domain_error); - CHECK_THROWS_WITH(j_nonobject.value("foo", 1), "cannot use value() with number"); - CHECK_THROWS_WITH(j_nonobject_const.value("foo", 1), "cannot use value() with number"); + CHECK_THROWS_AS(j_nonobject.value("foo", 1), json::type_error); + CHECK_THROWS_AS(j_nonobject_const.value("foo", 1), json::type_error); + CHECK_THROWS_WITH(j_nonobject.value("foo", 1), + "[json.exception.type_error.306] cannot use value() with number"); + CHECK_THROWS_WITH(j_nonobject_const.value("foo", 1), + "[json.exception.type_error.306] cannot use value() with number"); } SECTION("number (unsigned)") { json j_nonobject(json::value_t::number_unsigned); const json j_nonobject_const(j_nonobject); - CHECK_THROWS_AS(j_nonobject.value("foo", 1), std::domain_error); - CHECK_THROWS_AS(j_nonobject_const.value("foo", 1), std::domain_error); - CHECK_THROWS_WITH(j_nonobject.value("foo", 1), "cannot use value() with number"); - CHECK_THROWS_WITH(j_nonobject_const.value("foo", 1), "cannot use value() with number"); + CHECK_THROWS_AS(j_nonobject.value("foo", 1), json::type_error); + CHECK_THROWS_AS(j_nonobject_const.value("foo", 1), json::type_error); + CHECK_THROWS_WITH(j_nonobject.value("foo", 1), + "[json.exception.type_error.306] cannot use value() with number"); + CHECK_THROWS_WITH(j_nonobject_const.value("foo", 1), + "[json.exception.type_error.306] cannot use value() with number"); } SECTION("number (floating-point)") { json j_nonobject(json::value_t::number_float); const json j_nonobject_const(j_nonobject); - CHECK_THROWS_AS(j_nonobject.value("foo", 1), std::domain_error); - CHECK_THROWS_AS(j_nonobject_const.value("foo", 1), std::domain_error); - CHECK_THROWS_WITH(j_nonobject.value("foo", 1), "cannot use value() with number"); - CHECK_THROWS_WITH(j_nonobject_const.value("foo", 1), "cannot use value() with number"); + CHECK_THROWS_AS(j_nonobject.value("foo", 1), json::type_error); + CHECK_THROWS_AS(j_nonobject_const.value("foo", 1), json::type_error); + CHECK_THROWS_WITH(j_nonobject.value("foo", 1), + "[json.exception.type_error.306] cannot use value() with number"); + CHECK_THROWS_WITH(j_nonobject_const.value("foo", 1), + "[json.exception.type_error.306] cannot use value() with number"); } } } @@ -304,75 +318,84 @@ TEST_CASE("element access 2") { json j_nonobject(json::value_t::null); const json j_nonobject_const(j_nonobject); - CHECK_THROWS_AS(j_nonobject.value("/foo"_json_pointer, 1), std::domain_error); - CHECK_THROWS_AS(j_nonobject_const.value("/foo"_json_pointer, 1), std::domain_error); - CHECK_THROWS_WITH(j_nonobject.value("/foo"_json_pointer, 1), "cannot use value() with null"); - CHECK_THROWS_WITH(j_nonobject_const.value("/foo"_json_pointer, 1), "cannot use value() with null"); + CHECK_THROWS_AS(j_nonobject.value("/foo"_json_pointer, 1), json::type_error); + CHECK_THROWS_AS(j_nonobject_const.value("/foo"_json_pointer, 1), json::type_error); + CHECK_THROWS_WITH(j_nonobject.value("/foo"_json_pointer, 1), + "[json.exception.type_error.306] cannot use value() with null"); + CHECK_THROWS_WITH(j_nonobject_const.value("/foo"_json_pointer, 1), + "[json.exception.type_error.306] cannot use value() with null"); } SECTION("boolean") { json j_nonobject(json::value_t::boolean); const json j_nonobject_const(j_nonobject); - CHECK_THROWS_AS(j_nonobject.value("/foo"_json_pointer, 1), std::domain_error); - CHECK_THROWS_AS(j_nonobject_const.value("/foo"_json_pointer, 1), std::domain_error); - CHECK_THROWS_WITH(j_nonobject.value("/foo"_json_pointer, 1), "cannot use value() with boolean"); + CHECK_THROWS_AS(j_nonobject.value("/foo"_json_pointer, 1), json::type_error); + CHECK_THROWS_AS(j_nonobject_const.value("/foo"_json_pointer, 1), json::type_error); + CHECK_THROWS_WITH(j_nonobject.value("/foo"_json_pointer, 1), + "[json.exception.type_error.306] cannot use value() with boolean"); CHECK_THROWS_WITH(j_nonobject_const.value("/foo"_json_pointer, 1), - "cannot use value() with boolean"); + "[json.exception.type_error.306] cannot use value() with boolean"); } SECTION("string") { json j_nonobject(json::value_t::string); const json j_nonobject_const(j_nonobject); - CHECK_THROWS_AS(j_nonobject.value("/foo"_json_pointer, 1), std::domain_error); - CHECK_THROWS_AS(j_nonobject_const.value("/foo"_json_pointer, 1), std::domain_error); - CHECK_THROWS_WITH(j_nonobject.value("/foo"_json_pointer, 1), "cannot use value() with string"); + CHECK_THROWS_AS(j_nonobject.value("/foo"_json_pointer, 1), json::type_error); + CHECK_THROWS_AS(j_nonobject_const.value("/foo"_json_pointer, 1), json::type_error); + CHECK_THROWS_WITH(j_nonobject.value("/foo"_json_pointer, 1), + "[json.exception.type_error.306] cannot use value() with string"); CHECK_THROWS_WITH(j_nonobject_const.value("/foo"_json_pointer, 1), - "cannot use value() with string"); + "[json.exception.type_error.306] cannot use value() with string"); } SECTION("array") { json j_nonobject(json::value_t::array); const json j_nonobject_const(j_nonobject); - CHECK_THROWS_AS(j_nonobject.value("/foo"_json_pointer, 1), std::domain_error); - CHECK_THROWS_AS(j_nonobject_const.value("/foo"_json_pointer, 1), std::domain_error); - CHECK_THROWS_WITH(j_nonobject.value("/foo"_json_pointer, 1), "cannot use value() with array"); - CHECK_THROWS_WITH(j_nonobject_const.value("/foo"_json_pointer, 1), "cannot use value() with array"); + CHECK_THROWS_AS(j_nonobject.value("/foo"_json_pointer, 1), json::type_error); + CHECK_THROWS_AS(j_nonobject_const.value("/foo"_json_pointer, 1), json::type_error); + CHECK_THROWS_WITH(j_nonobject.value("/foo"_json_pointer, 1), + "[json.exception.type_error.306] cannot use value() with array"); + CHECK_THROWS_WITH(j_nonobject_const.value("/foo"_json_pointer, 1), + "[json.exception.type_error.306] cannot use value() with array"); } SECTION("number (integer)") { json j_nonobject(json::value_t::number_integer); const json j_nonobject_const(j_nonobject); - CHECK_THROWS_AS(j_nonobject.value("/foo"_json_pointer, 1), std::domain_error); - CHECK_THROWS_AS(j_nonobject_const.value("/foo"_json_pointer, 1), std::domain_error); - CHECK_THROWS_WITH(j_nonobject.value("/foo"_json_pointer, 1), "cannot use value() with number"); + CHECK_THROWS_AS(j_nonobject.value("/foo"_json_pointer, 1), json::type_error); + CHECK_THROWS_AS(j_nonobject_const.value("/foo"_json_pointer, 1), json::type_error); + CHECK_THROWS_WITH(j_nonobject.value("/foo"_json_pointer, 1), + "[json.exception.type_error.306] cannot use value() with number"); CHECK_THROWS_WITH(j_nonobject_const.value("/foo"_json_pointer, 1), - "cannot use value() with number"); + "[json.exception.type_error.306] cannot use value() with number"); } SECTION("number (unsigned)") { json j_nonobject(json::value_t::number_unsigned); const json j_nonobject_const(j_nonobject); - CHECK_THROWS_AS(j_nonobject.value("/foo"_json_pointer, 1), std::domain_error); - CHECK_THROWS_AS(j_nonobject_const.value("/foo"_json_pointer, 1), std::domain_error); - CHECK_THROWS_WITH(j_nonobject.value("/foo"_json_pointer, 1), "cannot use value() with number"); + CHECK_THROWS_AS(j_nonobject.value("/foo"_json_pointer, 1), json::type_error); + CHECK_THROWS_AS(j_nonobject_const.value("/foo"_json_pointer, 1), json::type_error); + CHECK_THROWS_WITH(j_nonobject.value("/foo"_json_pointer, 1), + "[json.exception.type_error.306] cannot use value() with number"); CHECK_THROWS_WITH(j_nonobject_const.value("/foo"_json_pointer, 1), - "cannot use value() with number"); + "[json.exception.type_error.306] cannot use value() with number"); } SECTION("number (floating-point)") { json j_nonobject(json::value_t::number_float); const json j_nonobject_const(j_nonobject); - CHECK_THROWS_AS(j_nonobject.value("/foo"_json_pointer, 1), std::domain_error); - CHECK_THROWS_AS(j_nonobject_const.value("/foo"_json_pointer, 1), std::domain_error); - CHECK_THROWS_WITH(j_nonobject.value("/foo"_json_pointer, 1), "cannot use value() with number"); + CHECK_THROWS_AS(j_nonobject.value("/foo"_json_pointer, 1), json::type_error); + CHECK_THROWS_AS(j_nonobject_const.value("/foo"_json_pointer, 1), json::type_error); + CHECK_THROWS_WITH(j_nonobject.value("/foo"_json_pointer, 1), + "[json.exception.type_error.306] cannot use value() with number"); CHECK_THROWS_WITH(j_nonobject_const.value("/foo"_json_pointer, 1), - "cannot use value() with number"); + "[json.exception.type_error.306] cannot use value() with number"); } } } From 2a9393af0063c291f837057ea605d9813edd6a32 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 5 Mar 2017 19:26:44 +0100 Subject: [PATCH 035/530] :hammer: added user-defined exceptions 307 --- src/json.hpp | 20 ++++---- src/json.hpp.re2c | 20 ++++---- test/src/unit-element_access1.cpp | 85 +++++++++++++++++++------------ test/src/unit-element_access2.cpp | 30 ++++++----- 4 files changed, 91 insertions(+), 64 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index e9654577..48c826f1 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -4397,8 +4397,8 @@ class basic_json @post Invalidates iterators and references at or after the point of the erase, including the `end()` iterator. - @throw std::domain_error if called on a `null` value; example: `"cannot - use erase() with null"` + @throw type_error.307 if called on a `null` value; example: `"cannot use + erase() with null"` @throw invalid_iterator.202 if called on an iterator which does not belong to the current JSON value; example: `"iterator does not fit current value"` @@ -4478,7 +4478,7 @@ class basic_json default: { - JSON_THROW(std::domain_error("cannot use erase() with " + type_name())); + JSON_THROW(type_error(307, "cannot use erase() with " + type_name())); } } @@ -4505,8 +4505,8 @@ class basic_json @post Invalidates iterators and references at or after the point of the erase, including the `end()` iterator. - @throw std::domain_error if called on a `null` value; example: `"cannot - use erase() with null"` + @throw type_error.307 if called on a `null` value; example: `"cannot use + erase() with null"` @throw invalid_iterator.203 if called on iterators which does not belong to the current JSON value; example: `"iterators do not fit current value"` @throw invalid_iterator.204 if called on a primitive type with invalid @@ -4587,7 +4587,7 @@ class basic_json default: { - JSON_THROW(std::domain_error("cannot use erase() with " + type_name())); + JSON_THROW(type_error(307, "cannot use erase() with " + type_name())); } } @@ -4608,7 +4608,7 @@ class basic_json @post References and iterators to the erased elements are invalidated. Other references and iterators are not affected. - @throw std::domain_error when called on a type other than JSON object; + @throw type_error.307 when called on a type other than JSON object; example: `"cannot use erase() with null"` @complexity `log(size()) + count(key)` @@ -4631,7 +4631,7 @@ class basic_json return m_value.object->erase(key); } - JSON_THROW(std::domain_error("cannot use erase() with " + type_name())); + JSON_THROW(type_error(307, "cannot use erase() with " + type_name())); } /*! @@ -4641,7 +4641,7 @@ class basic_json @param[in] idx index of the element to remove - @throw std::domain_error when called on a type other than JSON array; + @throw type_error.307 when called on a type other than JSON object; example: `"cannot use erase() with null"` @throw std::out_of_range when `idx >= size()`; example: `"array index 17 is out of range"` @@ -4672,7 +4672,7 @@ class basic_json } else { - JSON_THROW(std::domain_error("cannot use erase() with " + type_name())); + JSON_THROW(type_error(307, "cannot use erase() with " + type_name())); } } diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index d21dfdfa..c26717f7 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -4397,8 +4397,8 @@ class basic_json @post Invalidates iterators and references at or after the point of the erase, including the `end()` iterator. - @throw std::domain_error if called on a `null` value; example: `"cannot - use erase() with null"` + @throw type_error.307 if called on a `null` value; example: `"cannot use + erase() with null"` @throw invalid_iterator.202 if called on an iterator which does not belong to the current JSON value; example: `"iterator does not fit current value"` @@ -4478,7 +4478,7 @@ class basic_json default: { - JSON_THROW(std::domain_error("cannot use erase() with " + type_name())); + JSON_THROW(type_error(307, "cannot use erase() with " + type_name())); } } @@ -4505,8 +4505,8 @@ class basic_json @post Invalidates iterators and references at or after the point of the erase, including the `end()` iterator. - @throw std::domain_error if called on a `null` value; example: `"cannot - use erase() with null"` + @throw type_error.307 if called on a `null` value; example: `"cannot use + erase() with null"` @throw invalid_iterator.203 if called on iterators which does not belong to the current JSON value; example: `"iterators do not fit current value"` @throw invalid_iterator.204 if called on a primitive type with invalid @@ -4587,7 +4587,7 @@ class basic_json default: { - JSON_THROW(std::domain_error("cannot use erase() with " + type_name())); + JSON_THROW(type_error(307, "cannot use erase() with " + type_name())); } } @@ -4608,7 +4608,7 @@ class basic_json @post References and iterators to the erased elements are invalidated. Other references and iterators are not affected. - @throw std::domain_error when called on a type other than JSON object; + @throw type_error.307 when called on a type other than JSON object; example: `"cannot use erase() with null"` @complexity `log(size()) + count(key)` @@ -4631,7 +4631,7 @@ class basic_json return m_value.object->erase(key); } - JSON_THROW(std::domain_error("cannot use erase() with " + type_name())); + JSON_THROW(type_error(307, "cannot use erase() with " + type_name())); } /*! @@ -4641,7 +4641,7 @@ class basic_json @param[in] idx index of the element to remove - @throw std::domain_error when called on a type other than JSON array; + @throw type_error.307 when called on a type other than JSON object; example: `"cannot use erase() with null"` @throw std::out_of_range when `idx >= size()`; example: `"array index 17 is out of range"` @@ -4672,7 +4672,7 @@ class basic_json } else { - JSON_THROW(std::domain_error("cannot use erase() with " + type_name())); + JSON_THROW(type_error(307, "cannot use erase() with " + type_name())); } } diff --git a/test/src/unit-element_access1.cpp b/test/src/unit-element_access1.cpp index 37c138bb..e0e3ee99 100644 --- a/test/src/unit-element_access1.cpp +++ b/test/src/unit-element_access1.cpp @@ -444,50 +444,57 @@ TEST_CASE("element access 1") SECTION("null") { json j_nonobject(json::value_t::null); - CHECK_THROWS_AS(j_nonobject.erase(0), std::domain_error); - CHECK_THROWS_WITH(j_nonobject.erase(0), "cannot use erase() with null"); + CHECK_THROWS_AS(j_nonobject.erase(0), json::type_error); + CHECK_THROWS_WITH(j_nonobject.erase(0), + "[json.exception.type_error.307] cannot use erase() with null"); } SECTION("boolean") { json j_nonobject(json::value_t::boolean); - CHECK_THROWS_AS(j_nonobject.erase(0), std::domain_error); - CHECK_THROWS_WITH(j_nonobject.erase(0), "cannot use erase() with boolean"); + CHECK_THROWS_AS(j_nonobject.erase(0), json::type_error); + CHECK_THROWS_WITH(j_nonobject.erase(0), + "[json.exception.type_error.307] cannot use erase() with boolean"); } SECTION("string") { json j_nonobject(json::value_t::string); - CHECK_THROWS_AS(j_nonobject.erase(0), std::domain_error); - CHECK_THROWS_WITH(j_nonobject.erase(0), "cannot use erase() with string"); + CHECK_THROWS_AS(j_nonobject.erase(0), json::type_error); + CHECK_THROWS_WITH(j_nonobject.erase(0), + "[json.exception.type_error.307] cannot use erase() with string"); } SECTION("object") { json j_nonobject(json::value_t::object); - CHECK_THROWS_AS(j_nonobject.erase(0), std::domain_error); - CHECK_THROWS_WITH(j_nonobject.erase(0), "cannot use erase() with object"); + CHECK_THROWS_AS(j_nonobject.erase(0), json::type_error); + CHECK_THROWS_WITH(j_nonobject.erase(0), + "[json.exception.type_error.307] cannot use erase() with object"); } SECTION("number (integer)") { json j_nonobject(json::value_t::number_integer); - CHECK_THROWS_AS(j_nonobject.erase(0), std::domain_error); - CHECK_THROWS_WITH(j_nonobject.erase(0), "cannot use erase() with number"); + CHECK_THROWS_AS(j_nonobject.erase(0), json::type_error); + CHECK_THROWS_WITH(j_nonobject.erase(0), + "[json.exception.type_error.307] cannot use erase() with number"); } SECTION("number (unsigned)") { json j_nonobject(json::value_t::number_unsigned); - CHECK_THROWS_AS(j_nonobject.erase(0), std::domain_error); - CHECK_THROWS_WITH(j_nonobject.erase(0), "cannot use erase() with number"); + CHECK_THROWS_AS(j_nonobject.erase(0), json::type_error); + CHECK_THROWS_WITH(j_nonobject.erase(0), + "[json.exception.type_error.307] cannot use erase() with number"); } SECTION("number (floating-point)") { json j_nonobject(json::value_t::number_float); - CHECK_THROWS_AS(j_nonobject.erase(0), std::domain_error); - CHECK_THROWS_WITH(j_nonobject.erase(0), "cannot use erase() with number"); + CHECK_THROWS_AS(j_nonobject.erase(0), json::type_error); + CHECK_THROWS_WITH(j_nonobject.erase(0), + "[json.exception.type_error.307] cannot use erase() with number"); } } } @@ -592,13 +599,15 @@ TEST_CASE("element access 1") { { json j; - CHECK_THROWS_AS(j.erase(j.begin()), std::domain_error); - CHECK_THROWS_WITH(j.erase(j.begin()), "cannot use erase() with null"); + CHECK_THROWS_AS(j.erase(j.begin()), json::type_error); + CHECK_THROWS_WITH(j.erase(j.begin()), + "[json.exception.type_error.307] cannot use erase() with null"); } { json j; - CHECK_THROWS_AS(j.erase(j.cbegin()), std::domain_error); - CHECK_THROWS_WITH(j.erase(j.begin()), "cannot use erase() with null"); + CHECK_THROWS_AS(j.erase(j.cbegin()), json::type_error); + CHECK_THROWS_WITH(j.erase(j.begin()), + "[json.exception.type_error.307] cannot use erase() with null"); } } @@ -690,12 +699,14 @@ TEST_CASE("element access 1") { json j = "foo"; CHECK_THROWS_AS(j.erase(j.end()), json::invalid_iterator); - CHECK_THROWS_WITH(j.erase(j.end()), "[json.exception.invalid_iterator.205] iterator out of range"); + CHECK_THROWS_WITH(j.erase(j.end()), + "[json.exception.invalid_iterator.205] iterator out of range"); } { json j = "bar"; CHECK_THROWS_AS(j.erase(j.cend()), json::invalid_iterator); - CHECK_THROWS_WITH(j.erase(j.cend()), "[json.exception.invalid_iterator.205] iterator out of range"); + CHECK_THROWS_WITH(j.erase(j.cend()), + "[json.exception.invalid_iterator.205] iterator out of range"); } } @@ -704,12 +715,14 @@ TEST_CASE("element access 1") { json j = false; CHECK_THROWS_AS(j.erase(j.end()), json::invalid_iterator); - CHECK_THROWS_WITH(j.erase(j.end()), "[json.exception.invalid_iterator.205] iterator out of range"); + CHECK_THROWS_WITH(j.erase(j.end()), + "[json.exception.invalid_iterator.205] iterator out of range"); } { json j = true; CHECK_THROWS_AS(j.erase(j.cend()), json::invalid_iterator); - CHECK_THROWS_WITH(j.erase(j.cend()), "[json.exception.invalid_iterator.205] iterator out of range"); + CHECK_THROWS_WITH(j.erase(j.cend()), + "[json.exception.invalid_iterator.205] iterator out of range"); } } @@ -718,12 +731,14 @@ TEST_CASE("element access 1") { json j = 17; CHECK_THROWS_AS(j.erase(j.end()), json::invalid_iterator); - CHECK_THROWS_WITH(j.erase(j.end()), "[json.exception.invalid_iterator.205] iterator out of range"); + CHECK_THROWS_WITH(j.erase(j.end()), + "[json.exception.invalid_iterator.205] iterator out of range"); } { json j = 17; CHECK_THROWS_AS(j.erase(j.cend()), json::invalid_iterator); - CHECK_THROWS_WITH(j.erase(j.cend()), "[json.exception.invalid_iterator.205] iterator out of range"); + CHECK_THROWS_WITH(j.erase(j.cend()), + "[json.exception.invalid_iterator.205] iterator out of range"); } } @@ -732,12 +747,14 @@ TEST_CASE("element access 1") { json j = 17u; CHECK_THROWS_AS(j.erase(j.end()), json::invalid_iterator); - CHECK_THROWS_WITH(j.erase(j.end()), "[json.exception.invalid_iterator.205] iterator out of range"); + CHECK_THROWS_WITH(j.erase(j.end()), + "[json.exception.invalid_iterator.205] iterator out of range"); } { json j = 17u; CHECK_THROWS_AS(j.erase(j.cend()), json::invalid_iterator); - CHECK_THROWS_WITH(j.erase(j.cend()), "[json.exception.invalid_iterator.205] iterator out of range"); + CHECK_THROWS_WITH(j.erase(j.cend()), + "[json.exception.invalid_iterator.205] iterator out of range"); } } @@ -746,12 +763,14 @@ TEST_CASE("element access 1") { json j = 23.42; CHECK_THROWS_AS(j.erase(j.end()), json::invalid_iterator); - CHECK_THROWS_WITH(j.erase(j.end()), "[json.exception.invalid_iterator.205] iterator out of range"); + CHECK_THROWS_WITH(j.erase(j.end()), + "[json.exception.invalid_iterator.205] iterator out of range"); } { json j = 23.42; CHECK_THROWS_AS(j.erase(j.cend()), json::invalid_iterator); - CHECK_THROWS_WITH(j.erase(j.cend()), "[json.exception.invalid_iterator.205] iterator out of range"); + CHECK_THROWS_WITH(j.erase(j.cend()), + "[json.exception.invalid_iterator.205] iterator out of range"); } } } @@ -762,13 +781,15 @@ TEST_CASE("element access 1") { { json j; - CHECK_THROWS_AS(j.erase(j.begin(), j.end()), std::domain_error); - CHECK_THROWS_WITH(j.erase(j.begin(), j.end()), "cannot use erase() with null"); + CHECK_THROWS_AS(j.erase(j.begin(), j.end()), json::type_error); + CHECK_THROWS_WITH(j.erase(j.begin(), j.end()), + "[json.exception.type_error.307] cannot use erase() with null"); } { json j; - CHECK_THROWS_AS(j.erase(j.cbegin(), j.cend()), std::domain_error); - CHECK_THROWS_WITH(j.erase(j.cbegin(), j.cend()), "cannot use erase() with null"); + CHECK_THROWS_AS(j.erase(j.cbegin(), j.cend()), json::type_error); + CHECK_THROWS_WITH(j.erase(j.cbegin(), j.cend()), + "[json.exception.type_error.307] cannot use erase() with null"); } } diff --git a/test/src/unit-element_access2.cpp b/test/src/unit-element_access2.cpp index ec70d8df..4a066e70 100644 --- a/test/src/unit-element_access2.cpp +++ b/test/src/unit-element_access2.cpp @@ -757,43 +757,49 @@ TEST_CASE("element access 2") SECTION("null") { json j_nonobject(json::value_t::null); - CHECK_THROWS_AS(j_nonobject.erase("foo"), std::domain_error); - CHECK_THROWS_WITH(j_nonobject.erase("foo"), "cannot use erase() with null"); + CHECK_THROWS_AS(j_nonobject.erase("foo"), json::type_error); + CHECK_THROWS_WITH(j_nonobject.erase("foo"), + "[json.exception.type_error.307] cannot use erase() with null"); } SECTION("boolean") { json j_nonobject(json::value_t::boolean); - CHECK_THROWS_AS(j_nonobject.erase("foo"), std::domain_error); - CHECK_THROWS_WITH(j_nonobject.erase("foo"), "cannot use erase() with boolean"); + CHECK_THROWS_AS(j_nonobject.erase("foo"), json::type_error); + CHECK_THROWS_WITH(j_nonobject.erase("foo"), + "[json.exception.type_error.307] cannot use erase() with boolean"); } SECTION("string") { json j_nonobject(json::value_t::string); - CHECK_THROWS_AS(j_nonobject.erase("foo"), std::domain_error); - CHECK_THROWS_WITH(j_nonobject.erase("foo"), "cannot use erase() with string"); + CHECK_THROWS_AS(j_nonobject.erase("foo"), json::type_error); + CHECK_THROWS_WITH(j_nonobject.erase("foo"), + "[json.exception.type_error.307] cannot use erase() with string"); } SECTION("array") { json j_nonobject(json::value_t::array); - CHECK_THROWS_AS(j_nonobject.erase("foo"), std::domain_error); - CHECK_THROWS_WITH(j_nonobject.erase("foo"), "cannot use erase() with array"); + CHECK_THROWS_AS(j_nonobject.erase("foo"), json::type_error); + CHECK_THROWS_WITH(j_nonobject.erase("foo"), + "[json.exception.type_error.307] cannot use erase() with array"); } SECTION("number (integer)") { json j_nonobject(json::value_t::number_integer); - CHECK_THROWS_AS(j_nonobject.erase("foo"), std::domain_error); - CHECK_THROWS_WITH(j_nonobject.erase("foo"), "cannot use erase() with number"); + CHECK_THROWS_AS(j_nonobject.erase("foo"), json::type_error); + CHECK_THROWS_WITH(j_nonobject.erase("foo"), + "[json.exception.type_error.307] cannot use erase() with number"); } SECTION("number (floating-point)") { json j_nonobject(json::value_t::number_float); - CHECK_THROWS_AS(j_nonobject.erase("foo"), std::domain_error); - CHECK_THROWS_WITH(j_nonobject.erase("foo"), "cannot use erase() with number"); + CHECK_THROWS_AS(j_nonobject.erase("foo"), json::type_error); + CHECK_THROWS_WITH(j_nonobject.erase("foo"), + "[json.exception.type_error.307] cannot use erase() with number"); } } } From 70b2c3f45e23e51fd658f2e5457159463be70d0e Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 5 Mar 2017 19:35:24 +0100 Subject: [PATCH 036/530] :hammer: added user-defined exceptions 308 --- src/json.hpp | 10 +++++----- src/json.hpp.re2c | 10 +++++----- test/src/unit-modifiers.cpp | 32 ++++++++++++++++---------------- 3 files changed, 26 insertions(+), 26 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 48c826f1..4a422209 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -5374,7 +5374,7 @@ class basic_json @param[in] val the value to add to the JSON array - @throw std::domain_error when called on a type other than JSON array or + @throw type_error.308 when called on a type other than JSON array or null; example: `"cannot use push_back() with number"` @complexity Amortized constant. @@ -5390,7 +5390,7 @@ class basic_json // push_back only works for null objects or arrays if (not(is_null() or is_array())) { - JSON_THROW(std::domain_error("cannot use push_back() with " + type_name())); + JSON_THROW(type_error(308, "cannot use push_back() with " + type_name())); } // transform null object into an array @@ -5426,7 +5426,7 @@ class basic_json // push_back only works for null objects or arrays if (not(is_null() or is_array())) { - JSON_THROW(std::domain_error("cannot use push_back() with " + type_name())); + JSON_THROW(type_error(308, "cannot use push_back() with " + type_name())); } // transform null object into an array @@ -5460,7 +5460,7 @@ class basic_json @param[in] val the value to add to the JSON object - @throw std::domain_error when called on a type other than JSON object or + @throw type_error.308 when called on a type other than JSON object or null; example: `"cannot use push_back() with number"` @complexity Logarithmic in the size of the container, O(log(`size()`)). @@ -5476,7 +5476,7 @@ class basic_json // push_back only works for null objects or objects if (not(is_null() or is_object())) { - JSON_THROW(std::domain_error("cannot use push_back() with " + type_name())); + JSON_THROW(type_error(308, "cannot use push_back() with " + type_name())); } // transform null object into an object diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index c26717f7..69fef976 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -5374,7 +5374,7 @@ class basic_json @param[in] val the value to add to the JSON array - @throw std::domain_error when called on a type other than JSON array or + @throw type_error.308 when called on a type other than JSON array or null; example: `"cannot use push_back() with number"` @complexity Amortized constant. @@ -5390,7 +5390,7 @@ class basic_json // push_back only works for null objects or arrays if (not(is_null() or is_array())) { - JSON_THROW(std::domain_error("cannot use push_back() with " + type_name())); + JSON_THROW(type_error(308, "cannot use push_back() with " + type_name())); } // transform null object into an array @@ -5426,7 +5426,7 @@ class basic_json // push_back only works for null objects or arrays if (not(is_null() or is_array())) { - JSON_THROW(std::domain_error("cannot use push_back() with " + type_name())); + JSON_THROW(type_error(308, "cannot use push_back() with " + type_name())); } // transform null object into an array @@ -5460,7 +5460,7 @@ class basic_json @param[in] val the value to add to the JSON object - @throw std::domain_error when called on a type other than JSON object or + @throw type_error.308 when called on a type other than JSON object or null; example: `"cannot use push_back() with number"` @complexity Logarithmic in the size of the container, O(log(`size()`)). @@ -5476,7 +5476,7 @@ class basic_json // push_back only works for null objects or objects if (not(is_null() or is_object())) { - JSON_THROW(std::domain_error("cannot use push_back() with " + type_name())); + JSON_THROW(type_error(308, "cannot use push_back() with " + type_name())); } // transform null object into an object diff --git a/test/src/unit-modifiers.cpp b/test/src/unit-modifiers.cpp index 2a3826c2..dc84a930 100644 --- a/test/src/unit-modifiers.cpp +++ b/test/src/unit-modifiers.cpp @@ -152,8 +152,8 @@ TEST_CASE("modifiers") SECTION("other type") { json j = 1; - CHECK_THROWS_AS(j.push_back("Hello"), std::domain_error); - CHECK_THROWS_WITH(j.push_back("Hello"), "cannot use push_back() with number"); + CHECK_THROWS_AS(j.push_back("Hello"), json::type_error); + CHECK_THROWS_WITH(j.push_back("Hello"), "[json.exception.type_error.308] cannot use push_back() with number"); } } @@ -182,8 +182,8 @@ TEST_CASE("modifiers") { json j = 1; json k("Hello"); - CHECK_THROWS_AS(j.push_back(k), std::domain_error); - CHECK_THROWS_WITH(j.push_back(k), "cannot use push_back() with number"); + CHECK_THROWS_AS(j.push_back(k), json::type_error); + CHECK_THROWS_WITH(j.push_back(k), "[json.exception.type_error.308] cannot use push_back() with number"); } } } @@ -215,9 +215,9 @@ TEST_CASE("modifiers") { json j = 1; json k("Hello"); - CHECK_THROWS_AS(j.push_back(json::object_t::value_type({"one", 1})), std::domain_error); + CHECK_THROWS_AS(j.push_back(json::object_t::value_type({"one", 1})), json::type_error); CHECK_THROWS_WITH(j.push_back(json::object_t::value_type({"one", 1})), - "cannot use push_back() with number"); + "[json.exception.type_error.308] cannot use push_back() with number"); } } @@ -252,8 +252,8 @@ TEST_CASE("modifiers") CHECK(j == json({{"key1", 1}, {"key2", "bar"}})); json k = {{"key1", 1}}; - CHECK_THROWS_AS(k.push_back({1, 2, 3, 4}), std::domain_error); - CHECK_THROWS_WITH(k.push_back({1, 2, 3, 4}), "cannot use push_back() with object"); + CHECK_THROWS_AS(k.push_back({1, 2, 3, 4}), json::type_error); + CHECK_THROWS_WITH(k.push_back({1, 2, 3, 4}), "[json.exception.type_error.308] cannot use push_back() with object"); } } } @@ -381,8 +381,8 @@ TEST_CASE("modifiers") SECTION("other type") { json j = 1; - CHECK_THROWS_AS(j += "Hello", std::domain_error); - CHECK_THROWS_WITH(j += "Hello", "cannot use push_back() with number"); + CHECK_THROWS_AS(j += "Hello", json::type_error); + CHECK_THROWS_WITH(j += "Hello", "[json.exception.type_error.308] cannot use push_back() with number"); } } @@ -411,8 +411,8 @@ TEST_CASE("modifiers") { json j = 1; json k("Hello"); - CHECK_THROWS_AS(j += k, std::domain_error); - CHECK_THROWS_WITH(j += k, "cannot use push_back() with number"); + CHECK_THROWS_AS(j += k, json::type_error); + CHECK_THROWS_WITH(j += k, "[json.exception.type_error.308] cannot use push_back() with number"); } } } @@ -444,9 +444,9 @@ TEST_CASE("modifiers") { json j = 1; json k("Hello"); - CHECK_THROWS_AS(j += json::object_t::value_type({"one", 1}), std::domain_error); + CHECK_THROWS_AS(j += json::object_t::value_type({"one", 1}), json::type_error); CHECK_THROWS_WITH(j += json::object_t::value_type({"one", 1}), - "cannot use push_back() with number"); + "[json.exception.type_error.308] cannot use push_back() with number"); } } @@ -481,8 +481,8 @@ TEST_CASE("modifiers") CHECK(j == json({{"key1", 1}, {"key2", "bar"}})); json k = {{"key1", 1}}; - CHECK_THROWS_AS((k += {1, 2, 3, 4}), std::domain_error); - CHECK_THROWS_WITH((k += {1, 2, 3, 4}), "cannot use push_back() with object"); + CHECK_THROWS_AS((k += {1, 2, 3, 4}), json::type_error); + CHECK_THROWS_WITH((k += {1, 2, 3, 4}), "[json.exception.type_error.308] cannot use push_back() with object"); } } } From 5cca44c1611f50078b08490a276e1aee48f8f38e Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 5 Mar 2017 19:42:05 +0100 Subject: [PATCH 037/530] :hammer: added user-defined exceptions 309 --- src/json.hpp | 22 +++++++++++----------- src/json.hpp.re2c | 22 +++++++++++----------- test/src/unit-modifiers.cpp | 20 ++++++++++---------- 3 files changed, 32 insertions(+), 32 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 4a422209..4753b032 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -5655,7 +5655,7 @@ class basic_json @param[in] val element to insert @return iterator pointing to the inserted @a val. - @throw std::domain_error if called on JSON values other than arrays; + @throw type_error.309 if called on JSON values other than arrays; example: `"cannot use insert() with string"` @throw invalid_iterator.202 if @a pos is not an iterator of *this; example: `"iterator does not fit current value"` @@ -5684,7 +5684,7 @@ class basic_json return result; } - JSON_THROW(std::domain_error("cannot use insert() with " + type_name())); + JSON_THROW(type_error(309, "cannot use insert() with " + type_name())); } /*! @@ -5708,8 +5708,8 @@ class basic_json @return iterator pointing to the first element inserted, or @a pos if `cnt==0` - @throw std::domain_error if called on JSON values other than arrays; - example: `"cannot use insert() with string"` + @throw type_error.309 if called on JSON values other than arrays; example: + `"cannot use insert() with string"` @throw invalid_iterator.202 if @a pos is not an iterator of *this; example: `"iterator does not fit current value"` @@ -5737,7 +5737,7 @@ class basic_json return result; } - JSON_THROW(std::domain_error("cannot use insert() with " + type_name())); + JSON_THROW(type_error(309, "cannot use insert() with " + type_name())); } /*! @@ -5750,8 +5750,8 @@ class basic_json @param[in] first begin of the range of elements to insert @param[in] last end of the range of elements to insert - @throw std::domain_error if called on JSON values other than arrays; - example: `"cannot use insert() with string"` + @throw type_error.309 if called on JSON values other than arrays; example: + `"cannot use insert() with string"` @throw invalid_iterator.202 if @a pos is not an iterator of *this; example: `"iterator does not fit current value"` @throw invalid_iterator.210 if @a first and @a last do not belong to the @@ -5775,7 +5775,7 @@ class basic_json // insert only works for arrays if (not is_array()) { - JSON_THROW(std::domain_error("cannot use insert() with " + type_name())); + JSON_THROW(type_error(309, "cannot use insert() with " + type_name())); } // check if iterator pos fits to this JSON value @@ -5813,8 +5813,8 @@ class basic_json the end() iterator @param[in] ilist initializer list to insert the values from - @throw std::domain_error if called on JSON values other than arrays; - example: `"cannot use insert() with string"` + @throw type_error.309 if called on JSON values other than arrays; example: + `"cannot use insert() with string"` @throw invalid_iterator.202 if @a pos is not an iterator of *this; example: `"iterator does not fit current value"` @@ -5833,7 +5833,7 @@ class basic_json // insert only works for arrays if (not is_array()) { - JSON_THROW(std::domain_error("cannot use insert() with " + type_name())); + JSON_THROW(type_error(309, "cannot use insert() with " + type_name())); } // check if iterator pos fits to this JSON value diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 69fef976..8f12b9e2 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -5655,7 +5655,7 @@ class basic_json @param[in] val element to insert @return iterator pointing to the inserted @a val. - @throw std::domain_error if called on JSON values other than arrays; + @throw type_error.309 if called on JSON values other than arrays; example: `"cannot use insert() with string"` @throw invalid_iterator.202 if @a pos is not an iterator of *this; example: `"iterator does not fit current value"` @@ -5684,7 +5684,7 @@ class basic_json return result; } - JSON_THROW(std::domain_error("cannot use insert() with " + type_name())); + JSON_THROW(type_error(309, "cannot use insert() with " + type_name())); } /*! @@ -5708,8 +5708,8 @@ class basic_json @return iterator pointing to the first element inserted, or @a pos if `cnt==0` - @throw std::domain_error if called on JSON values other than arrays; - example: `"cannot use insert() with string"` + @throw type_error.309 if called on JSON values other than arrays; example: + `"cannot use insert() with string"` @throw invalid_iterator.202 if @a pos is not an iterator of *this; example: `"iterator does not fit current value"` @@ -5737,7 +5737,7 @@ class basic_json return result; } - JSON_THROW(std::domain_error("cannot use insert() with " + type_name())); + JSON_THROW(type_error(309, "cannot use insert() with " + type_name())); } /*! @@ -5750,8 +5750,8 @@ class basic_json @param[in] first begin of the range of elements to insert @param[in] last end of the range of elements to insert - @throw std::domain_error if called on JSON values other than arrays; - example: `"cannot use insert() with string"` + @throw type_error.309 if called on JSON values other than arrays; example: + `"cannot use insert() with string"` @throw invalid_iterator.202 if @a pos is not an iterator of *this; example: `"iterator does not fit current value"` @throw invalid_iterator.210 if @a first and @a last do not belong to the @@ -5775,7 +5775,7 @@ class basic_json // insert only works for arrays if (not is_array()) { - JSON_THROW(std::domain_error("cannot use insert() with " + type_name())); + JSON_THROW(type_error(309, "cannot use insert() with " + type_name())); } // check if iterator pos fits to this JSON value @@ -5813,8 +5813,8 @@ class basic_json the end() iterator @param[in] ilist initializer list to insert the values from - @throw std::domain_error if called on JSON values other than arrays; - example: `"cannot use insert() with string"` + @throw type_error.309 if called on JSON values other than arrays; example: + `"cannot use insert() with string"` @throw invalid_iterator.202 if @a pos is not an iterator of *this; example: `"iterator does not fit current value"` @@ -5833,7 +5833,7 @@ class basic_json // insert only works for arrays if (not is_array()) { - JSON_THROW(std::domain_error("cannot use insert() with " + type_name())); + JSON_THROW(type_error(309, "cannot use insert() with " + type_name())); } // check if iterator pos fits to this JSON value diff --git a/test/src/unit-modifiers.cpp b/test/src/unit-modifiers.cpp index dc84a930..029dc996 100644 --- a/test/src/unit-modifiers.cpp +++ b/test/src/unit-modifiers.cpp @@ -686,20 +686,20 @@ TEST_CASE("modifiers") // call insert on a non-array type json j_nonarray = 3; json j_yet_another_array = {"first", "second"}; - CHECK_THROWS_AS(j_nonarray.insert(j_nonarray.end(), 10), std::domain_error); - CHECK_THROWS_AS(j_nonarray.insert(j_nonarray.end(), j_value), std::domain_error); - CHECK_THROWS_AS(j_nonarray.insert(j_nonarray.end(), 10, 11), std::domain_error); + CHECK_THROWS_AS(j_nonarray.insert(j_nonarray.end(), 10), json::type_error); + CHECK_THROWS_AS(j_nonarray.insert(j_nonarray.end(), j_value), json::type_error); + CHECK_THROWS_AS(j_nonarray.insert(j_nonarray.end(), 10, 11), json::type_error); CHECK_THROWS_AS(j_nonarray.insert(j_nonarray.end(), j_yet_another_array.begin(), - j_yet_another_array.end()), std::domain_error); - CHECK_THROWS_AS(j_nonarray.insert(j_nonarray.end(), {1, 2, 3, 4}), std::domain_error); + j_yet_another_array.end()), json::type_error); + CHECK_THROWS_AS(j_nonarray.insert(j_nonarray.end(), {1, 2, 3, 4}), json::type_error); - CHECK_THROWS_WITH(j_nonarray.insert(j_nonarray.end(), 10), "cannot use insert() with number"); - CHECK_THROWS_WITH(j_nonarray.insert(j_nonarray.end(), j_value), "cannot use insert() with number"); - CHECK_THROWS_WITH(j_nonarray.insert(j_nonarray.end(), 10, 11), "cannot use insert() with number"); + CHECK_THROWS_WITH(j_nonarray.insert(j_nonarray.end(), 10), "[json.exception.type_error.309] cannot use insert() with number"); + CHECK_THROWS_WITH(j_nonarray.insert(j_nonarray.end(), j_value), "[json.exception.type_error.309] cannot use insert() with number"); + CHECK_THROWS_WITH(j_nonarray.insert(j_nonarray.end(), 10, 11), "[json.exception.type_error.309] cannot use insert() with number"); CHECK_THROWS_WITH(j_nonarray.insert(j_nonarray.end(), j_yet_another_array.begin(), - j_yet_another_array.end()), "cannot use insert() with number"); + j_yet_another_array.end()), "[json.exception.type_error.309] cannot use insert() with number"); CHECK_THROWS_WITH(j_nonarray.insert(j_nonarray.end(), {1, 2, 3, 4}), - "cannot use insert() with number"); + "[json.exception.type_error.309] cannot use insert() with number"); } } From 144cf6a4c7eae74eabf4123429a1e985d0a5b656 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 5 Mar 2017 19:48:11 +0100 Subject: [PATCH 038/530] :hammer: added user-defined exceptions 310 --- src/json.hpp | 14 +++++++------- src/json.hpp.re2c | 14 +++++++------- test/src/unit-modifiers.cpp | 12 ++++++------ 3 files changed, 20 insertions(+), 20 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 4753b032..8d78430a 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -5887,8 +5887,8 @@ class basic_json @param[in,out] other array to exchange the contents with - @throw std::domain_error when JSON value is not an array; example: - `"cannot use swap() with string"` + @throw type_error.310 when JSON value is not an array; example: `"cannot + use swap() with string"` @complexity Constant. @@ -5906,7 +5906,7 @@ class basic_json } else { - JSON_THROW(std::domain_error("cannot use swap() with " + type_name())); + JSON_THROW(type_error(310, "cannot use swap() with " + type_name())); } } @@ -5920,7 +5920,7 @@ class basic_json @param[in,out] other object to exchange the contents with - @throw std::domain_error when JSON value is not an object; example: + @throw type_error.310 when JSON value is not an object; example: `"cannot use swap() with string"` @complexity Constant. @@ -5939,7 +5939,7 @@ class basic_json } else { - JSON_THROW(std::domain_error("cannot use swap() with " + type_name())); + JSON_THROW(type_error(310, "cannot use swap() with " + type_name())); } } @@ -5953,7 +5953,7 @@ class basic_json @param[in,out] other string to exchange the contents with - @throw std::domain_error when JSON value is not a string; example: `"cannot + @throw type_error.310 when JSON value is not a string; example: `"cannot use swap() with boolean"` @complexity Constant. @@ -5972,7 +5972,7 @@ class basic_json } else { - JSON_THROW(std::domain_error("cannot use swap() with " + type_name())); + JSON_THROW(type_error(310, "cannot use swap() with " + type_name())); } } diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 8f12b9e2..2f03506c 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -5887,8 +5887,8 @@ class basic_json @param[in,out] other array to exchange the contents with - @throw std::domain_error when JSON value is not an array; example: - `"cannot use swap() with string"` + @throw type_error.310 when JSON value is not an array; example: `"cannot + use swap() with string"` @complexity Constant. @@ -5906,7 +5906,7 @@ class basic_json } else { - JSON_THROW(std::domain_error("cannot use swap() with " + type_name())); + JSON_THROW(type_error(310, "cannot use swap() with " + type_name())); } } @@ -5920,7 +5920,7 @@ class basic_json @param[in,out] other object to exchange the contents with - @throw std::domain_error when JSON value is not an object; example: + @throw type_error.310 when JSON value is not an object; example: `"cannot use swap() with string"` @complexity Constant. @@ -5939,7 +5939,7 @@ class basic_json } else { - JSON_THROW(std::domain_error("cannot use swap() with " + type_name())); + JSON_THROW(type_error(310, "cannot use swap() with " + type_name())); } } @@ -5953,7 +5953,7 @@ class basic_json @param[in,out] other string to exchange the contents with - @throw std::domain_error when JSON value is not a string; example: `"cannot + @throw type_error.310 when JSON value is not a string; example: `"cannot use swap() with boolean"` @complexity Constant. @@ -5972,7 +5972,7 @@ class basic_json } else { - JSON_THROW(std::domain_error("cannot use swap() with " + type_name())); + JSON_THROW(type_error(310, "cannot use swap() with " + type_name())); } } diff --git a/test/src/unit-modifiers.cpp b/test/src/unit-modifiers.cpp index 029dc996..d3c03094 100644 --- a/test/src/unit-modifiers.cpp +++ b/test/src/unit-modifiers.cpp @@ -751,8 +751,8 @@ TEST_CASE("modifiers") json j = 17; json::array_t a = {"foo", "bar", "baz"}; - CHECK_THROWS_AS(j.swap(a), std::domain_error); - CHECK_THROWS_WITH(j.swap(a), "cannot use swap() with number"); + CHECK_THROWS_AS(j.swap(a), json::type_error); + CHECK_THROWS_WITH(j.swap(a), "[json.exception.type_error.310] cannot use swap() with number"); } } @@ -777,8 +777,8 @@ TEST_CASE("modifiers") json j = 17; json::object_t o = {{"cow", "Kuh"}, {"chicken", "Huhn"}}; - CHECK_THROWS_AS(j.swap(o), std::domain_error); - CHECK_THROWS_WITH(j.swap(o), "cannot use swap() with number"); + CHECK_THROWS_AS(j.swap(o), json::type_error); + CHECK_THROWS_WITH(j.swap(o), "[json.exception.type_error.310] cannot use swap() with number"); } } @@ -803,8 +803,8 @@ TEST_CASE("modifiers") json j = 17; json::string_t s = "Hallo Welt"; - CHECK_THROWS_AS(j.swap(s), std::domain_error); - CHECK_THROWS_WITH(j.swap(s), "cannot use swap() with number"); + CHECK_THROWS_AS(j.swap(s), json::type_error); + CHECK_THROWS_WITH(j.swap(s), "[json.exception.type_error.310] cannot use swap() with number"); } } } From 9e560ca40ce443b9d14e2889a5bb988cfbb900e3 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 5 Mar 2017 19:58:26 +0100 Subject: [PATCH 039/530] :hammer: added user-defined exceptions 313-315 --- src/json.hpp | 6 +++--- src/json.hpp.re2c | 6 +++--- test/src/unit-json_pointer.cpp | 15 +++++++++------ 3 files changed, 15 insertions(+), 12 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 8d78430a..54aaab56 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -12059,7 +12059,7 @@ basic_json_parser_74: */ default: { - JSON_THROW(std::domain_error("invalid value to unflatten")); + JSON_THROW(type_error(313, "invalid value to unflatten")); } } } @@ -12496,7 +12496,7 @@ basic_json_parser_74: { if (not value.is_object()) { - JSON_THROW(std::domain_error("only objects can be unflattened")); + JSON_THROW(type_error(314, "only objects can be unflattened")); } basic_json result; @@ -12506,7 +12506,7 @@ basic_json_parser_74: { if (not element.second.is_primitive()) { - JSON_THROW(std::domain_error("values in object must be primitive")); + JSON_THROW(type_error(315, "values in object must be primitive")); } // assign value to reference pointed to by JSON pointer; Note diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 2f03506c..af91aa36 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -11092,7 +11092,7 @@ class basic_json */ default: { - JSON_THROW(std::domain_error("invalid value to unflatten")); + JSON_THROW(type_error(313, "invalid value to unflatten")); } } } @@ -11529,7 +11529,7 @@ class basic_json { if (not value.is_object()) { - JSON_THROW(std::domain_error("only objects can be unflattened")); + JSON_THROW(type_error(314, "only objects can be unflattened")); } basic_json result; @@ -11539,7 +11539,7 @@ class basic_json { if (not element.second.is_primitive()) { - JSON_THROW(std::domain_error("values in object must be primitive")); + JSON_THROW(type_error(315, "values in object must be primitive")); } // assign value to reference pointed to by JSON pointer; Note diff --git a/test/src/unit-json_pointer.cpp b/test/src/unit-json_pointer.cpp index 3c05a278..99519675 100644 --- a/test/src/unit-json_pointer.cpp +++ b/test/src/unit-json_pointer.cpp @@ -358,17 +358,20 @@ TEST_CASE("JSON pointers") CHECK(j_flatten.unflatten() == j); // error for nonobjects - CHECK_THROWS_AS(json(1).unflatten(), std::domain_error); - CHECK_THROWS_WITH(json(1).unflatten(), "only objects can be unflattened"); + CHECK_THROWS_AS(json(1).unflatten(), json::type_error); + CHECK_THROWS_WITH(json(1).unflatten(), + "[json.exception.type_error.314] only objects can be unflattened"); // error for nonprimitve values - CHECK_THROWS_AS(json({{"/1", {1, 2, 3}}}).unflatten(), std::domain_error); - CHECK_THROWS_WITH(json({{"/1", {1, 2, 3}}}).unflatten(), "values in object must be primitive"); + CHECK_THROWS_AS(json({{"/1", {1, 2, 3}}}).unflatten(), json::type_error); + CHECK_THROWS_WITH(json({{"/1", {1, 2, 3}}}).unflatten(), + "[json.exception.type_error.315] values in object must be primitive"); // error for conflicting values json j_error = {{"", 42}, {"/foo", 17}}; - CHECK_THROWS_AS(j_error.unflatten(), std::domain_error); - CHECK_THROWS_WITH(j_error.unflatten(), "invalid value to unflatten"); + CHECK_THROWS_AS(j_error.unflatten(), json::type_error); + CHECK_THROWS_WITH(j_error.unflatten(), + "[json.exception.type_error.313] invalid value to unflatten"); // explicit roundtrip check CHECK(j.flatten().unflatten() == j); From 6751d650be02a84651f6d0b0eec9650e480069e8 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 5 Mar 2017 22:31:08 +0100 Subject: [PATCH 040/530] :hammer: added user-defined exception 311 --- src/json.hpp | 5 +++-- src/json.hpp.re2c | 5 +++-- test/src/unit-modifiers.cpp | 5 +++-- 3 files changed, 9 insertions(+), 6 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 54aaab56..b8ef5999 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -1137,6 +1137,7 @@ class basic_json json.exception.[type_error](@ref type_error).308 | "cannot use push_back() with string" | The @ref push_back() and @ref operator+= member functions can only be executed for certain JSON types. json.exception.[type_error](@ref type_error).309 | "cannot use insert() with" | The @ref insert() member functions can only be executed for certain JSON types. json.exception.[type_error](@ref type_error).310 | "cannot use swap() with number" | The @ref swap() member functions can only be executed for certain JSON types. + json.exception.[type_error](@ref type_error).311 | "cannot use emplace_back() with string" | The @ref emplace_back() member function can only be executed for certain JSON types. json.exception.[type_error](@ref type_error).313 | "invalid value to unflatten" | The @ref unflatten function converts an object whose keys are JSON Pointers back into an arbitrary nested JSON value. The JSON Pointers must not overlap, because then the resulting value would not be well defined. json.exception.[type_error](@ref type_error).314 | "only objects can be unflattened" | The @ref unflatten function only works for an object whose keys are JSON Pointers. json.exception.[type_error](@ref type_error).315 | "values in object must be primitive" | The @ref unflatten function only works for an object whose keys are JSON Pointers and whose values are primitive. @@ -5559,7 +5560,7 @@ class basic_json @param[in] args arguments to forward to a constructor of @ref basic_json @tparam Args compatible types to create a @ref basic_json object - @throw std::domain_error when called on a type other than JSON array or + @throw type_error.311 when called on a type other than JSON array or null; example: `"cannot use emplace_back() with number"` @complexity Amortized constant. @@ -5576,7 +5577,7 @@ class basic_json // emplace_back only works for null objects or arrays if (not(is_null() or is_array())) { - JSON_THROW(std::domain_error("cannot use emplace_back() with " + type_name())); + JSON_THROW(type_error(311, "cannot use emplace_back() with " + type_name())); } // transform null object into an array diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index af91aa36..1026ad79 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -1137,6 +1137,7 @@ class basic_json json.exception.[type_error](@ref type_error).308 | "cannot use push_back() with string" | The @ref push_back() and @ref operator+= member functions can only be executed for certain JSON types. json.exception.[type_error](@ref type_error).309 | "cannot use insert() with" | The @ref insert() member functions can only be executed for certain JSON types. json.exception.[type_error](@ref type_error).310 | "cannot use swap() with number" | The @ref swap() member functions can only be executed for certain JSON types. + json.exception.[type_error](@ref type_error).311 | "cannot use emplace_back() with string" | The @ref emplace_back() member function can only be executed for certain JSON types. json.exception.[type_error](@ref type_error).313 | "invalid value to unflatten" | The @ref unflatten function converts an object whose keys are JSON Pointers back into an arbitrary nested JSON value. The JSON Pointers must not overlap, because then the resulting value would not be well defined. json.exception.[type_error](@ref type_error).314 | "only objects can be unflattened" | The @ref unflatten function only works for an object whose keys are JSON Pointers. json.exception.[type_error](@ref type_error).315 | "values in object must be primitive" | The @ref unflatten function only works for an object whose keys are JSON Pointers and whose values are primitive. @@ -5559,7 +5560,7 @@ class basic_json @param[in] args arguments to forward to a constructor of @ref basic_json @tparam Args compatible types to create a @ref basic_json object - @throw std::domain_error when called on a type other than JSON array or + @throw type_error.311 when called on a type other than JSON array or null; example: `"cannot use emplace_back() with number"` @complexity Amortized constant. @@ -5576,7 +5577,7 @@ class basic_json // emplace_back only works for null objects or arrays if (not(is_null() or is_array())) { - JSON_THROW(std::domain_error("cannot use emplace_back() with " + type_name())); + JSON_THROW(type_error(311, "cannot use emplace_back() with " + type_name())); } // transform null object into an array diff --git a/test/src/unit-modifiers.cpp b/test/src/unit-modifiers.cpp index d3c03094..c1a76909 100644 --- a/test/src/unit-modifiers.cpp +++ b/test/src/unit-modifiers.cpp @@ -291,8 +291,9 @@ TEST_CASE("modifiers") SECTION("other type") { json j = 1; - CHECK_THROWS_AS(j.emplace_back("Hello"), std::domain_error); - CHECK_THROWS_WITH(j.emplace_back("Hello"), "cannot use emplace_back() with number"); + CHECK_THROWS_AS(j.emplace_back("Hello"), json::type_error); + CHECK_THROWS_WITH(j.emplace_back("Hello"), + "[json.exception.type_error.311] cannot use emplace_back() with number"); } } From 491c9780a7063e0a39c2adcb0aa9a8f3b2016a5e Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 5 Mar 2017 22:39:21 +0100 Subject: [PATCH 041/530] :hammer: added user-defined exception 311 --- src/json.hpp | 4 ++-- src/json.hpp.re2c | 4 ++-- test/src/unit-modifiers.cpp | 7 ++++--- 3 files changed, 8 insertions(+), 7 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index b8ef5999..00e3fe24 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -5607,7 +5607,7 @@ class basic_json already-existing element if no insertion happened, and a bool denoting whether the insertion took place. - @throw std::domain_error when called on a type other than JSON object or + @throw type_error.311 when called on a type other than JSON object or null; example: `"cannot use emplace() with number"` @complexity Logarithmic in the size of the container, O(log(`size()`)). @@ -5625,7 +5625,7 @@ class basic_json // emplace only works for null objects or arrays if (not(is_null() or is_object())) { - JSON_THROW(std::domain_error("cannot use emplace() with " + type_name())); + JSON_THROW(type_error(311, "cannot use emplace() with " + type_name())); } // transform null object into an object diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 1026ad79..8de7e16f 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -5607,7 +5607,7 @@ class basic_json already-existing element if no insertion happened, and a bool denoting whether the insertion took place. - @throw std::domain_error when called on a type other than JSON object or + @throw type_error.311 when called on a type other than JSON object or null; example: `"cannot use emplace() with number"` @complexity Logarithmic in the size of the container, O(log(`size()`)). @@ -5625,7 +5625,7 @@ class basic_json // emplace only works for null objects or arrays if (not(is_null() or is_object())) { - JSON_THROW(std::domain_error("cannot use emplace() with " + type_name())); + JSON_THROW(type_error(311, "cannot use emplace() with " + type_name())); } // transform null object into an object diff --git a/test/src/unit-modifiers.cpp b/test/src/unit-modifiers.cpp index c1a76909..ad055420 100644 --- a/test/src/unit-modifiers.cpp +++ b/test/src/unit-modifiers.cpp @@ -293,7 +293,7 @@ TEST_CASE("modifiers") json j = 1; CHECK_THROWS_AS(j.emplace_back("Hello"), json::type_error); CHECK_THROWS_WITH(j.emplace_back("Hello"), - "[json.exception.type_error.311] cannot use emplace_back() with number"); + "[json.exception.type_error.311] cannot use emplace_back() with number"); } } @@ -351,8 +351,9 @@ TEST_CASE("modifiers") SECTION("other type") { json j = 1; - CHECK_THROWS_AS(j.emplace("foo", "bar"), std::domain_error); - CHECK_THROWS_WITH(j.emplace("foo", "bar"), "cannot use emplace() with number"); + CHECK_THROWS_AS(j.emplace("foo", "bar"), json::type_error); + CHECK_THROWS_WITH(j.emplace("foo", "bar"), + "[json.exception.type_error.311] cannot use emplace() with number"); } } From 60da36aee214c1711edb87062a152bce3ca965a2 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 5 Mar 2017 22:56:39 +0100 Subject: [PATCH 042/530] :hammer: added user-defined exceptions 401-402 --- src/json.hpp | 30 +++++++++++++------------- src/json.hpp.re2c | 30 +++++++++++++------------- test/src/unit-element_access1.cpp | 15 +++++++------ test/src/unit-json_patch.cpp | 25 +++++++++++++--------- test/src/unit-json_pointer.cpp | 35 ++++++++++++++++++------------- 5 files changed, 75 insertions(+), 60 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 00e3fe24..3ee57f6e 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -3648,7 +3648,7 @@ class basic_json @throw type_error.304 if the JSON value is not an array; example: `"cannot use at() with string"` - @throw std::out_of_range if the index @a idx is out of range of the array; + @throw out_of_range.401 if the index @a idx is out of range of the array; that is, `idx >= size()`; example: `"array index 7 is out of range"` @complexity Constant. @@ -3670,7 +3670,7 @@ class basic_json JSON_CATCH (std::out_of_range&) { // create better exception explanation - JSON_THROW(std::out_of_range("array index " + std::to_string(idx) + " is out of range")); + JSON_THROW(out_of_range(401, "array index " + std::to_string(idx) + " is out of range")); } } else @@ -3691,7 +3691,7 @@ class basic_json @throw type_error.304 if the JSON value is not an array; example: `"cannot use at() with string"` - @throw std::out_of_range if the index @a idx is out of range of the array; + @throw out_of_range.401 if the index @a idx is out of range of the array; that is, `idx >= size()`; example: `"array index 7 is out of range"` @complexity Constant. @@ -3713,7 +3713,7 @@ class basic_json JSON_CATCH (std::out_of_range&) { // create better exception explanation - JSON_THROW(std::out_of_range("array index " + std::to_string(idx) + " is out of range")); + JSON_THROW(out_of_range(401, "array index " + std::to_string(idx) + " is out of range")); } } else @@ -4666,7 +4666,7 @@ class basic_json { if (idx >= size()) { - JSON_THROW(std::out_of_range("array index " + std::to_string(idx) + " is out of range")); + JSON_THROW(out_of_range(401, "array index " + std::to_string(idx) + " is out of range")); } m_value.array->erase(m_value.array->begin() + static_cast(idx)); @@ -12179,9 +12179,9 @@ basic_json_parser_74: if (reference_token == "-") { // "-" always fails the range check - JSON_THROW(std::out_of_range("array index '-' (" + - std::to_string(ptr->m_value.array->size()) + - ") is out of range")); + JSON_THROW(out_of_range(402, "array index '-' (" + + std::to_string(ptr->m_value.array->size()) + + ") is out of range")); } // error condition (cf. RFC 6901, Sect. 4) @@ -12238,9 +12238,9 @@ basic_json_parser_74: if (reference_token == "-") { // "-" cannot be used for const access - JSON_THROW(std::out_of_range("array index '-' (" + - std::to_string(ptr->m_value.array->size()) + - ") is out of range")); + JSON_THROW(out_of_range(402, "array index '-' (" + + std::to_string(ptr->m_value.array->size()) + + ") is out of range")); } // error condition (cf. RFC 6901, Sect. 4) @@ -12289,9 +12289,9 @@ basic_json_parser_74: if (reference_token == "-") { // "-" always fails the range check - JSON_THROW(std::out_of_range("array index '-' (" + - std::to_string(ptr->m_value.array->size()) + - ") is out of range")); + JSON_THROW(out_of_range(402, "array index '-' (" + + std::to_string(ptr->m_value.array->size()) + + ") is out of range")); } // error condition (cf. RFC 6901, Sect. 4) @@ -12850,7 +12850,7 @@ basic_json_parser_74: if (static_cast(idx) > parent.size()) { // avoid undefined behavior - JSON_THROW(std::out_of_range("array index " + std::to_string(idx) + " is out of range")); + JSON_THROW(out_of_range(401, "array index " + std::to_string(idx) + " is out of range")); } else { diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 8de7e16f..c02f6bf8 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -3648,7 +3648,7 @@ class basic_json @throw type_error.304 if the JSON value is not an array; example: `"cannot use at() with string"` - @throw std::out_of_range if the index @a idx is out of range of the array; + @throw out_of_range.401 if the index @a idx is out of range of the array; that is, `idx >= size()`; example: `"array index 7 is out of range"` @complexity Constant. @@ -3670,7 +3670,7 @@ class basic_json JSON_CATCH (std::out_of_range&) { // create better exception explanation - JSON_THROW(std::out_of_range("array index " + std::to_string(idx) + " is out of range")); + JSON_THROW(out_of_range(401, "array index " + std::to_string(idx) + " is out of range")); } } else @@ -3691,7 +3691,7 @@ class basic_json @throw type_error.304 if the JSON value is not an array; example: `"cannot use at() with string"` - @throw std::out_of_range if the index @a idx is out of range of the array; + @throw out_of_range.401 if the index @a idx is out of range of the array; that is, `idx >= size()`; example: `"array index 7 is out of range"` @complexity Constant. @@ -3713,7 +3713,7 @@ class basic_json JSON_CATCH (std::out_of_range&) { // create better exception explanation - JSON_THROW(std::out_of_range("array index " + std::to_string(idx) + " is out of range")); + JSON_THROW(out_of_range(401, "array index " + std::to_string(idx) + " is out of range")); } } else @@ -4666,7 +4666,7 @@ class basic_json { if (idx >= size()) { - JSON_THROW(std::out_of_range("array index " + std::to_string(idx) + " is out of range")); + JSON_THROW(out_of_range(401, "array index " + std::to_string(idx) + " is out of range")); } m_value.array->erase(m_value.array->begin() + static_cast(idx)); @@ -11212,9 +11212,9 @@ class basic_json if (reference_token == "-") { // "-" always fails the range check - JSON_THROW(std::out_of_range("array index '-' (" + - std::to_string(ptr->m_value.array->size()) + - ") is out of range")); + JSON_THROW(out_of_range(402, "array index '-' (" + + std::to_string(ptr->m_value.array->size()) + + ") is out of range")); } // error condition (cf. RFC 6901, Sect. 4) @@ -11271,9 +11271,9 @@ class basic_json if (reference_token == "-") { // "-" cannot be used for const access - JSON_THROW(std::out_of_range("array index '-' (" + - std::to_string(ptr->m_value.array->size()) + - ") is out of range")); + JSON_THROW(out_of_range(402, "array index '-' (" + + std::to_string(ptr->m_value.array->size()) + + ") is out of range")); } // error condition (cf. RFC 6901, Sect. 4) @@ -11322,9 +11322,9 @@ class basic_json if (reference_token == "-") { // "-" always fails the range check - JSON_THROW(std::out_of_range("array index '-' (" + - std::to_string(ptr->m_value.array->size()) + - ") is out of range")); + JSON_THROW(out_of_range(402, "array index '-' (" + + std::to_string(ptr->m_value.array->size()) + + ") is out of range")); } // error condition (cf. RFC 6901, Sect. 4) @@ -11883,7 +11883,7 @@ class basic_json if (static_cast(idx) > parent.size()) { // avoid undefined behavior - JSON_THROW(std::out_of_range("array index " + std::to_string(idx) + " is out of range")); + JSON_THROW(out_of_range(401, "array index " + std::to_string(idx) + " is out of range")); } else { diff --git a/test/src/unit-element_access1.cpp b/test/src/unit-element_access1.cpp index e0e3ee99..b8a57af4 100644 --- a/test/src/unit-element_access1.cpp +++ b/test/src/unit-element_access1.cpp @@ -63,11 +63,13 @@ TEST_CASE("element access 1") SECTION("access outside bounds") { - CHECK_THROWS_AS(j.at(8), std::out_of_range); - CHECK_THROWS_AS(j_const.at(8), std::out_of_range); + CHECK_THROWS_AS(j.at(8), json::out_of_range); + CHECK_THROWS_AS(j_const.at(8), json::out_of_range); - CHECK_THROWS_WITH(j.at(8), "array index 8 is out of range"); - CHECK_THROWS_WITH(j_const.at(8), "array index 8 is out of range"); + CHECK_THROWS_WITH(j.at(8), + "[json.exception.out_of_range.401] array index 8 is out of range"); + CHECK_THROWS_WITH(j_const.at(8), + "[json.exception.out_of_range.401] array index 8 is out of range"); } SECTION("access on non-array type") @@ -311,8 +313,9 @@ TEST_CASE("element access 1") } { json jarray = {1, 1u, true, nullptr, "string", 42.23, json::object(), {1, 2, 3}}; - CHECK_THROWS_AS(jarray.erase(8), std::out_of_range); - CHECK_THROWS_WITH(jarray.erase(8), "array index 8 is out of range"); + CHECK_THROWS_AS(jarray.erase(8), json::out_of_range); + CHECK_THROWS_WITH(jarray.erase(8), + "[json.exception.out_of_range.401] array index 8 is out of range"); } } diff --git a/test/src/unit-json_patch.cpp b/test/src/unit-json_patch.cpp index 1d905bb5..feb7a154 100644 --- a/test/src/unit-json_patch.cpp +++ b/test/src/unit-json_patch.cpp @@ -741,8 +741,9 @@ TEST_CASE("JSON patch") { json j = {1, 2}; json patch = {{{"op", "add"}, {"path", "/4"}, {"value", 4}}}; - CHECK_THROWS_AS(j.patch(patch), std::out_of_range); - CHECK_THROWS_WITH(j.patch(patch), "array index 4 is out of range"); + CHECK_THROWS_AS(j.patch(patch), json::out_of_range); + CHECK_THROWS_WITH(j.patch(patch), + "[json.exception.out_of_range.401] array index 4 is out of range"); } } @@ -770,8 +771,9 @@ TEST_CASE("JSON patch") { json j = {1, 2, 3}; json patch = {{{"op", "remove"}, {"path", "/17"}}}; - CHECK_THROWS_AS(j.patch(patch), std::out_of_range); - CHECK_THROWS_WITH(j.patch(patch), "array index 17 is out of range"); + CHECK_THROWS_AS(j.patch(patch), json::out_of_range); + CHECK_THROWS_WITH(j.patch(patch), + "[json.exception.out_of_range.401] array index 17 is out of range"); } SECTION("nonexisting target location (object)") @@ -824,8 +826,9 @@ TEST_CASE("JSON patch") { json j = {1, 2, 3}; json patch = {{{"op", "replace"}, {"path", "/17"}, {"value", 19}}}; - CHECK_THROWS_AS(j.patch(patch), std::out_of_range); - CHECK_THROWS_WITH(j.patch(patch), "array index 17 is out of range"); + CHECK_THROWS_AS(j.patch(patch), json::out_of_range); + CHECK_THROWS_WITH(j.patch(patch), + "[json.exception.out_of_range.401] array index 17 is out of range"); } SECTION("nonexisting target location (object)") @@ -879,8 +882,9 @@ TEST_CASE("JSON patch") { json j = {1, 2, 3}; json patch = {{{"op", "move"}, {"path", "/0"}, {"from", "/5"}}}; - CHECK_THROWS_AS(j.patch(patch), std::out_of_range); - CHECK_THROWS_WITH(j.patch(patch), "array index 5 is out of range"); + CHECK_THROWS_AS(j.patch(patch), json::out_of_range); + CHECK_THROWS_WITH(j.patch(patch), + "[json.exception.out_of_range.401] array index 5 is out of range"); } SECTION("nonexisting from location (object)") @@ -934,8 +938,9 @@ TEST_CASE("JSON patch") { json j = {1, 2, 3}; json patch = {{{"op", "copy"}, {"path", "/0"}, {"from", "/5"}}}; - CHECK_THROWS_AS(j.patch(patch), std::out_of_range); - CHECK_THROWS_WITH(j.patch(patch), "array index 5 is out of range"); + CHECK_THROWS_AS(j.patch(patch), json::out_of_range); + CHECK_THROWS_WITH(j.patch(patch), + "[json.exception.out_of_range.401] array index 5 is out of range"); } SECTION("nonexisting from location (object)") diff --git a/test/src/unit-json_pointer.cpp b/test/src/unit-json_pointer.cpp index 99519675..db45a966 100644 --- a/test/src/unit-json_pointer.cpp +++ b/test/src/unit-json_pointer.cpp @@ -271,14 +271,17 @@ TEST_CASE("JSON pointers") CHECK(j == json({1, 13, 3, 33, nullptr, 55, 99})); // error when using "-" in const object - CHECK_THROWS_AS(j_const["/-"_json_pointer], std::out_of_range); - CHECK_THROWS_WITH(j_const["/-"_json_pointer], "array index '-' (3) is out of range"); + CHECK_THROWS_AS(j_const["/-"_json_pointer], json::out_of_range); + CHECK_THROWS_WITH(j_const["/-"_json_pointer], + "[json.exception.out_of_range.402] array index '-' (3) is out of range"); // error when using "-" with at - CHECK_THROWS_AS(j.at("/-"_json_pointer), std::out_of_range); - CHECK_THROWS_WITH(j.at("/-"_json_pointer), "array index '-' (7) is out of range"); - CHECK_THROWS_AS(j_const.at("/-"_json_pointer), std::out_of_range); - CHECK_THROWS_WITH(j_const.at("/-"_json_pointer), "array index '-' (3) is out of range"); + CHECK_THROWS_AS(j.at("/-"_json_pointer), json::out_of_range); + CHECK_THROWS_WITH(j.at("/-"_json_pointer), + "[json.exception.out_of_range.402] array index '-' (7) is out of range"); + CHECK_THROWS_AS(j_const.at("/-"_json_pointer), json::out_of_range); + CHECK_THROWS_WITH(j_const.at("/-"_json_pointer), + "[json.exception.out_of_range.402] array index '-' (3) is out of range"); } SECTION("const access") @@ -291,18 +294,22 @@ TEST_CASE("JSON pointers") CHECK(j["/2"_json_pointer] == j[2]); // assign to nonexisting index - CHECK_THROWS_AS(j.at("/3"_json_pointer), std::out_of_range); - CHECK_THROWS_WITH(j.at("/3"_json_pointer), "array index 3 is out of range"); + CHECK_THROWS_AS(j.at("/3"_json_pointer), json::out_of_range); + CHECK_THROWS_WITH(j.at("/3"_json_pointer), + "[json.exception.out_of_range.401] array index 3 is out of range"); // assign to nonexisting index (with gap) - CHECK_THROWS_AS(j.at("/5"_json_pointer), std::out_of_range); - CHECK_THROWS_WITH(j.at("/5"_json_pointer), "array index 5 is out of range"); + CHECK_THROWS_AS(j.at("/5"_json_pointer), json::out_of_range); + CHECK_THROWS_WITH(j.at("/5"_json_pointer), + "[json.exception.out_of_range.401] array index 5 is out of range"); // assign to "-" - CHECK_THROWS_AS(j["/-"_json_pointer], std::out_of_range); - CHECK_THROWS_WITH(j["/-"_json_pointer], "array index '-' (3) is out of range"); - CHECK_THROWS_AS(j.at("/-"_json_pointer), std::out_of_range); - CHECK_THROWS_WITH(j.at("/-"_json_pointer), "array index '-' (3) is out of range"); + CHECK_THROWS_AS(j["/-"_json_pointer], json::out_of_range); + CHECK_THROWS_WITH(j["/-"_json_pointer], + "[json.exception.out_of_range.402] array index '-' (3) is out of range"); + CHECK_THROWS_AS(j.at("/-"_json_pointer), json::out_of_range); + CHECK_THROWS_WITH(j.at("/-"_json_pointer), + "[json.exception.out_of_range.402] array index '-' (3) is out of range"); } } From 30331fa21fc89c962756912a9f69d565dd3209fb Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 5 Mar 2017 23:16:17 +0100 Subject: [PATCH 043/530] :hammer: added user-defined exception 403 --- src/json.hpp | 24 ++++++++++++------------ src/json.hpp.re2c | 24 ++++++++++++------------ test/src/unit-element_access2.cpp | 10 ++++++---- test/src/unit-json_patch.cpp | 30 ++++++++++++++++++------------ test/src/unit-json_pointer.cpp | 5 +++-- 5 files changed, 51 insertions(+), 42 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 3ee57f6e..54528175 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -3734,7 +3734,7 @@ class basic_json @throw type_error.304 if the JSON value is not an object; example: `"cannot use at() with boolean"` - @throw std::out_of_range if the key @a key is is not stored in the object; + @throw out_of_range.403 if the key @a key is is not stored in the object; that is, `find(key) == end()`; example: `"key "the fast" not found"` @complexity Logarithmic in the size of the container. @@ -3760,7 +3760,7 @@ class basic_json JSON_CATCH (std::out_of_range&) { // create better exception explanation - JSON_THROW(std::out_of_range("key '" + key + "' not found")); + JSON_THROW(out_of_range(403, "key '" + key + "' not found")); } } else @@ -3781,7 +3781,7 @@ class basic_json @throw type_error.304 if the JSON value is not an object; example: `"cannot use at() with boolean"` - @throw std::out_of_range if the key @a key is is not stored in the object; + @throw out_of_range.403 if the key @a key is is not stored in the object; that is, `find(key) == end()`; example: `"key "the fast" not found"` @complexity Logarithmic in the size of the container. @@ -3807,7 +3807,7 @@ class basic_json JSON_CATCH (std::out_of_range&) { // create better exception explanation - JSON_THROW(std::out_of_range("key '" + key + "' not found")); + JSON_THROW(out_of_range(403, "key '" + key + "' not found")); } } else @@ -4154,7 +4154,7 @@ class basic_json @code {.cpp} try { return at(key); - } catch(std::out_of_range) { + } catch(out_of_range) { return default_value; } @endcode @@ -4233,7 +4233,7 @@ class basic_json @code {.cpp} try { return at(ptr); - } catch(std::out_of_range) { + } catch(out_of_range) { return default_value; } @endcode @@ -4276,7 +4276,7 @@ class basic_json { return ptr.get_checked(this); } - JSON_CATCH (std::out_of_range&) + JSON_CATCH (out_of_range&) { return default_value; } @@ -12746,9 +12746,9 @@ basic_json_parser_74: any case, the original value is not changed: the patch is applied to a copy of the value. - @throw std::out_of_range if a JSON pointer inside the patch could not - be resolved successfully in the current JSON value; example: `"key baz - not found"` + @throw out_of_range.403 if a JSON pointer inside the patch could not be + resolved successfully in the current JSON value; example: `"key baz not + found"` @throw invalid_argument if the JSON patch is malformed (e.g., mandatory attributes are missing); example: `"operation add must have member path"` @throw parse_error.104 if the JSON patch does not consist of an array of @@ -12888,7 +12888,7 @@ basic_json_parser_74: } else { - JSON_THROW(std::out_of_range("key '" + last_path + "' not found")); + JSON_THROW(out_of_range(403, "key '" + last_path + "' not found")); } } else if (parent.is_array()) @@ -13002,7 +13002,7 @@ basic_json_parser_74: // the "path" location must exist - use at() success = (result.at(ptr) == get_value("test", "value", false)); } - JSON_CATCH (std::out_of_range&) + JSON_CATCH (out_of_range&) { // ignore out of range errors: success remains false } diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index c02f6bf8..2bbff0fb 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -3734,7 +3734,7 @@ class basic_json @throw type_error.304 if the JSON value is not an object; example: `"cannot use at() with boolean"` - @throw std::out_of_range if the key @a key is is not stored in the object; + @throw out_of_range.403 if the key @a key is is not stored in the object; that is, `find(key) == end()`; example: `"key "the fast" not found"` @complexity Logarithmic in the size of the container. @@ -3760,7 +3760,7 @@ class basic_json JSON_CATCH (std::out_of_range&) { // create better exception explanation - JSON_THROW(std::out_of_range("key '" + key + "' not found")); + JSON_THROW(out_of_range(403, "key '" + key + "' not found")); } } else @@ -3781,7 +3781,7 @@ class basic_json @throw type_error.304 if the JSON value is not an object; example: `"cannot use at() with boolean"` - @throw std::out_of_range if the key @a key is is not stored in the object; + @throw out_of_range.403 if the key @a key is is not stored in the object; that is, `find(key) == end()`; example: `"key "the fast" not found"` @complexity Logarithmic in the size of the container. @@ -3807,7 +3807,7 @@ class basic_json JSON_CATCH (std::out_of_range&) { // create better exception explanation - JSON_THROW(std::out_of_range("key '" + key + "' not found")); + JSON_THROW(out_of_range(403, "key '" + key + "' not found")); } } else @@ -4154,7 +4154,7 @@ class basic_json @code {.cpp} try { return at(key); - } catch(std::out_of_range) { + } catch(out_of_range) { return default_value; } @endcode @@ -4233,7 +4233,7 @@ class basic_json @code {.cpp} try { return at(ptr); - } catch(std::out_of_range) { + } catch(out_of_range) { return default_value; } @endcode @@ -4276,7 +4276,7 @@ class basic_json { return ptr.get_checked(this); } - JSON_CATCH (std::out_of_range&) + JSON_CATCH (out_of_range&) { return default_value; } @@ -11779,9 +11779,9 @@ class basic_json any case, the original value is not changed: the patch is applied to a copy of the value. - @throw std::out_of_range if a JSON pointer inside the patch could not - be resolved successfully in the current JSON value; example: `"key baz - not found"` + @throw out_of_range.403 if a JSON pointer inside the patch could not be + resolved successfully in the current JSON value; example: `"key baz not + found"` @throw invalid_argument if the JSON patch is malformed (e.g., mandatory attributes are missing); example: `"operation add must have member path"` @throw parse_error.104 if the JSON patch does not consist of an array of @@ -11921,7 +11921,7 @@ class basic_json } else { - JSON_THROW(std::out_of_range("key '" + last_path + "' not found")); + JSON_THROW(out_of_range(403, "key '" + last_path + "' not found")); } } else if (parent.is_array()) @@ -12035,7 +12035,7 @@ class basic_json // the "path" location must exist - use at() success = (result.at(ptr) == get_value("test", "value", false)); } - JSON_CATCH (std::out_of_range&) + JSON_CATCH (out_of_range&) { // ignore out of range errors: success remains false } diff --git a/test/src/unit-element_access2.cpp b/test/src/unit-element_access2.cpp index 4a066e70..3533bdcd 100644 --- a/test/src/unit-element_access2.cpp +++ b/test/src/unit-element_access2.cpp @@ -63,10 +63,12 @@ TEST_CASE("element access 2") SECTION("access outside bounds") { - CHECK_THROWS_AS(j.at("foo"), std::out_of_range); - CHECK_THROWS_AS(j_const.at("foo"), std::out_of_range); - CHECK_THROWS_WITH(j.at("foo"), "key 'foo' not found"); - CHECK_THROWS_WITH(j_const.at("foo"), "key 'foo' not found"); + CHECK_THROWS_AS(j.at("foo"), json::out_of_range); + CHECK_THROWS_AS(j_const.at("foo"), json::out_of_range); + CHECK_THROWS_WITH(j.at("foo"), + "[json.exception.out_of_range.403] key 'foo' not found"); + CHECK_THROWS_WITH(j_const.at("foo"), + "[json.exception.out_of_range.403] key 'foo' not found"); } SECTION("access on non-object type") diff --git a/test/src/unit-json_patch.cpp b/test/src/unit-json_patch.cpp index feb7a154..9c07aae6 100644 --- a/test/src/unit-json_patch.cpp +++ b/test/src/unit-json_patch.cpp @@ -75,8 +75,9 @@ TEST_CASE("JSON patch") json doc2 = R"({ "q": { "bar": 2 } })"_json; // because "a" does not exist. - CHECK_THROWS_AS(doc2.patch(patch), std::out_of_range); - CHECK_THROWS_WITH(doc2.patch(patch), "key 'a' not found"); + CHECK_THROWS_AS(doc2.patch(patch), json::out_of_range); + CHECK_THROWS_WITH(doc2.patch(patch), + "[json.exception.out_of_range.403] key 'a' not found"); } SECTION("4.2 remove") @@ -420,8 +421,9 @@ TEST_CASE("JSON patch") // references neither the root of the document, nor a member of // an existing object, nor a member of an existing array. - CHECK_THROWS_AS(doc.patch(patch), std::out_of_range); - CHECK_THROWS_WITH(doc.patch(patch), "key 'baz' not found"); + CHECK_THROWS_AS(doc.patch(patch), json::out_of_range); + CHECK_THROWS_WITH(doc.patch(patch), + "[json.exception.out_of_range.403] key 'baz' not found"); } // A.13. Invalid JSON Patch Document @@ -780,8 +782,9 @@ TEST_CASE("JSON patch") { json j = {{"foo", 1}, {"bar", 2}}; json patch = {{{"op", "remove"}, {"path", "/baz"}}}; - CHECK_THROWS_AS(j.patch(patch), std::out_of_range); - CHECK_THROWS_WITH(j.patch(patch), "key 'baz' not found"); + CHECK_THROWS_AS(j.patch(patch), json::out_of_range); + CHECK_THROWS_WITH(j.patch(patch), + "[json.exception.out_of_range.403] key 'baz' not found"); } SECTION("root element as target location") @@ -835,8 +838,9 @@ TEST_CASE("JSON patch") { json j = {{"foo", 1}, {"bar", 2}}; json patch = {{{"op", "replace"}, {"path", "/baz"}, {"value", 3}}}; - CHECK_THROWS_AS(j.patch(patch), std::out_of_range); - CHECK_THROWS_WITH(j.patch(patch), "key 'baz' not found"); + CHECK_THROWS_AS(j.patch(patch), json::out_of_range); + CHECK_THROWS_WITH(j.patch(patch), + "[json.exception.out_of_range.403] key 'baz' not found"); } } @@ -891,8 +895,9 @@ TEST_CASE("JSON patch") { json j = {{"foo", 1}, {"bar", 2}}; json patch = {{{"op", "move"}, {"path", "/baz"}, {"from", "/baz"}}}; - CHECK_THROWS_AS(j.patch(patch), std::out_of_range); - CHECK_THROWS_WITH(j.patch(patch), "key 'baz' not found"); + CHECK_THROWS_AS(j.patch(patch), json::out_of_range); + CHECK_THROWS_WITH(j.patch(patch), + "[json.exception.out_of_range.403] key 'baz' not found"); } } @@ -947,8 +952,9 @@ TEST_CASE("JSON patch") { json j = {{"foo", 1}, {"bar", 2}}; json patch = {{{"op", "copy"}, {"path", "/fob"}, {"from", "/baz"}}}; - CHECK_THROWS_AS(j.patch(patch), std::out_of_range); - CHECK_THROWS_WITH(j.patch(patch), "key 'baz' not found"); + CHECK_THROWS_AS(j.patch(patch), json::out_of_range); + CHECK_THROWS_WITH(j.patch(patch), + "[json.exception.out_of_range.403] key 'baz' not found"); } } diff --git a/test/src/unit-json_pointer.cpp b/test/src/unit-json_pointer.cpp index db45a966..c12a56ad 100644 --- a/test/src/unit-json_pointer.cpp +++ b/test/src/unit-json_pointer.cpp @@ -185,8 +185,9 @@ TEST_CASE("JSON pointers") CHECK(j[json::json_pointer("/m~0n")] == j["m~n"]); // unescaped access - CHECK_THROWS_AS(j.at(json::json_pointer("/a/b")), std::out_of_range); - CHECK_THROWS_WITH(j.at(json::json_pointer("/a/b")), "key 'a' not found"); + CHECK_THROWS_AS(j.at(json::json_pointer("/a/b")), json::out_of_range); + CHECK_THROWS_WITH(j.at(json::json_pointer("/a/b")), + "[json.exception.out_of_range.403] key 'a' not found"); // unresolved access const json j_primitive = 1; From 38c2e20ce8ceee95077f3adba1bbb402b1192022 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 5 Mar 2017 23:25:22 +0100 Subject: [PATCH 044/530] added user-defined exceptions 404-405 --- src/json.hpp | 14 +++++++------- src/json.hpp.re2c | 14 +++++++------- test/src/unit-json_patch.cpp | 5 +++-- test/src/unit-json_pointer.cpp | 30 ++++++++++++++++++------------ 4 files changed, 35 insertions(+), 28 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 54528175..3de94bce 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -11974,7 +11974,7 @@ basic_json_parser_74: { if (is_root()) { - JSON_THROW(std::domain_error("JSON pointer has no parent")); + JSON_THROW(out_of_range(405, "JSON pointer has no parent")); } auto last = reference_tokens.back(); @@ -11992,7 +11992,7 @@ basic_json_parser_74: { if (is_root()) { - JSON_THROW(std::domain_error("JSON pointer has no parent")); + JSON_THROW(out_of_range(405, "JSON pointer has no parent")); } json_pointer result = *this; @@ -12083,7 +12083,7 @@ basic_json_parser_74: @complexity Linear in the length of the JSON pointer. - @throw std::out_of_range if the JSON pointer can not be resolved + @throw out_of_range.404 if the JSON pointer can not be resolved @throw parse_error.106 if an array index begins with '0' @throw std::invalid_argument if an array index was not a number */ @@ -12153,7 +12153,7 @@ basic_json_parser_74: default: { - JSON_THROW(std::out_of_range("unresolved reference token '" + reference_token + "'")); + JSON_THROW(out_of_range(404, "unresolved reference token '" + reference_token + "'")); } } } @@ -12204,7 +12204,7 @@ basic_json_parser_74: default: { - JSON_THROW(std::out_of_range("unresolved reference token '" + reference_token + "'")); + JSON_THROW(out_of_range(404, "unresolved reference token '" + reference_token + "'")); } } } @@ -12263,7 +12263,7 @@ basic_json_parser_74: default: { - JSON_THROW(std::out_of_range("unresolved reference token '" + reference_token + "'")); + JSON_THROW(out_of_range(404, "unresolved reference token '" + reference_token + "'")); } } } @@ -12314,7 +12314,7 @@ basic_json_parser_74: default: { - JSON_THROW(std::out_of_range("unresolved reference token '" + reference_token + "'")); + JSON_THROW(out_of_range(404, "unresolved reference token '" + reference_token + "'")); } } } diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 2bbff0fb..b47a48cf 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -11007,7 +11007,7 @@ class basic_json { if (is_root()) { - JSON_THROW(std::domain_error("JSON pointer has no parent")); + JSON_THROW(out_of_range(405, "JSON pointer has no parent")); } auto last = reference_tokens.back(); @@ -11025,7 +11025,7 @@ class basic_json { if (is_root()) { - JSON_THROW(std::domain_error("JSON pointer has no parent")); + JSON_THROW(out_of_range(405, "JSON pointer has no parent")); } json_pointer result = *this; @@ -11116,7 +11116,7 @@ class basic_json @complexity Linear in the length of the JSON pointer. - @throw std::out_of_range if the JSON pointer can not be resolved + @throw out_of_range.404 if the JSON pointer can not be resolved @throw parse_error.106 if an array index begins with '0' @throw std::invalid_argument if an array index was not a number */ @@ -11186,7 +11186,7 @@ class basic_json default: { - JSON_THROW(std::out_of_range("unresolved reference token '" + reference_token + "'")); + JSON_THROW(out_of_range(404, "unresolved reference token '" + reference_token + "'")); } } } @@ -11237,7 +11237,7 @@ class basic_json default: { - JSON_THROW(std::out_of_range("unresolved reference token '" + reference_token + "'")); + JSON_THROW(out_of_range(404, "unresolved reference token '" + reference_token + "'")); } } } @@ -11296,7 +11296,7 @@ class basic_json default: { - JSON_THROW(std::out_of_range("unresolved reference token '" + reference_token + "'")); + JSON_THROW(out_of_range(404, "unresolved reference token '" + reference_token + "'")); } } } @@ -11347,7 +11347,7 @@ class basic_json default: { - JSON_THROW(std::out_of_range("unresolved reference token '" + reference_token + "'")); + JSON_THROW(out_of_range(404, "unresolved reference token '" + reference_token + "'")); } } } diff --git a/test/src/unit-json_patch.cpp b/test/src/unit-json_patch.cpp index 9c07aae6..c021450b 100644 --- a/test/src/unit-json_patch.cpp +++ b/test/src/unit-json_patch.cpp @@ -791,8 +791,9 @@ TEST_CASE("JSON patch") { json j = "string"; json patch = {{{"op", "remove"}, {"path", ""}}}; - CHECK_THROWS_AS(j.patch(patch), std::domain_error); - CHECK_THROWS_WITH(j.patch(patch), "JSON pointer has no parent"); + CHECK_THROWS_AS(j.patch(patch), json::out_of_range); + CHECK_THROWS_WITH(j.patch(patch), + "[json.exception.out_of_range.405] JSON pointer has no parent"); } } diff --git a/test/src/unit-json_pointer.cpp b/test/src/unit-json_pointer.cpp index c12a56ad..a6f811eb 100644 --- a/test/src/unit-json_pointer.cpp +++ b/test/src/unit-json_pointer.cpp @@ -49,10 +49,12 @@ TEST_CASE("JSON pointers") "[json.exception.parse_error.108] parse error: escape character '~' must be followed with '0' or '1'"); json::json_pointer p; - CHECK_THROWS_AS(p.top(), std::domain_error); - CHECK_THROWS_WITH(p.top(), "JSON pointer has no parent"); - CHECK_THROWS_AS(p.pop_back(), std::domain_error); - CHECK_THROWS_WITH(p.pop_back(), "JSON pointer has no parent"); + CHECK_THROWS_AS(p.top(), json::out_of_range); + CHECK_THROWS_WITH(p.top(), + "[json.exception.out_of_range.405] JSON pointer has no parent"); + CHECK_THROWS_AS(p.pop_back(), json::out_of_range); + CHECK_THROWS_WITH(p.pop_back(), + "[json.exception.out_of_range.405] JSON pointer has no parent"); } SECTION("examples from RFC 6901") @@ -124,10 +126,12 @@ TEST_CASE("JSON pointers") // unresolved access json j_primitive = 1; - CHECK_THROWS_AS(j_primitive["/foo"_json_pointer], std::out_of_range); - CHECK_THROWS_WITH(j_primitive["/foo"_json_pointer], "unresolved reference token 'foo'"); - CHECK_THROWS_AS(j_primitive.at("/foo"_json_pointer), std::out_of_range); - CHECK_THROWS_WITH(j_primitive.at("/foo"_json_pointer), "unresolved reference token 'foo'"); + CHECK_THROWS_AS(j_primitive["/foo"_json_pointer], json::out_of_range); + CHECK_THROWS_WITH(j_primitive["/foo"_json_pointer], + "[json.exception.out_of_range.404] unresolved reference token 'foo'"); + CHECK_THROWS_AS(j_primitive.at("/foo"_json_pointer), json::out_of_range); + CHECK_THROWS_WITH(j_primitive.at("/foo"_json_pointer), + "[json.exception.out_of_range.404] unresolved reference token 'foo'"); } SECTION("const access") @@ -191,10 +195,12 @@ TEST_CASE("JSON pointers") // unresolved access const json j_primitive = 1; - CHECK_THROWS_AS(j_primitive["/foo"_json_pointer], std::out_of_range); - CHECK_THROWS_WITH(j_primitive["/foo"_json_pointer], "unresolved reference token 'foo'"); - CHECK_THROWS_AS(j_primitive.at("/foo"_json_pointer), std::out_of_range); - CHECK_THROWS_WITH(j_primitive.at("/foo"_json_pointer), "unresolved reference token 'foo'"); + CHECK_THROWS_AS(j_primitive["/foo"_json_pointer], json::out_of_range); + CHECK_THROWS_WITH(j_primitive["/foo"_json_pointer], + "[json.exception.out_of_range.404] unresolved reference token 'foo'"); + CHECK_THROWS_AS(j_primitive.at("/foo"_json_pointer), json::out_of_range); + CHECK_THROWS_WITH(j_primitive.at("/foo"_json_pointer), + "[json.exception.out_of_range.404] unresolved reference token 'foo'"); } SECTION("user-defined string literal") From a186106bde35032d813b82897a73efbcfbda230c Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Mon, 6 Mar 2017 19:34:44 +0100 Subject: [PATCH 045/530] :hammer: user-defined exception 302 MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Also moved all exception classes into the detail namespace and introduced them via “using” into basic_json. --- src/json.hpp | 404 ++++++++++++++++++---------------- src/json.hpp.re2c | 404 ++++++++++++++++++---------------- test/src/unit-conversions.cpp | 184 ++++++++-------- 3 files changed, 530 insertions(+), 462 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 3de94bce..d7108ed6 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -110,6 +110,187 @@ This namespace collects some functions that could not be defined inside the */ namespace detail { +//////////////// +// exceptions // +//////////////// + +/*! +@brief general exception of the @ref basic_json class + +Extension of std::exception objects with a member @a id for exception ids. + +@since version 3.0.0 +*/ +class exception : public std::exception +{ + public: + /// create exception with id an explanatory string + exception(int id_, const std::string& ename, const std::string& what_arg_) + : id(id_), + what_arg("[json.exception." + ename + "." + std::to_string(id_) + "] " + what_arg_) + {} + + /// returns the explanatory string + virtual const char* what() const noexcept + { + return what_arg.c_str(); + } + + /// the id of the exception + const int id; + + private: + /// the explanatory string + const std::string what_arg; +}; + +/*! +@brief exception indicating a parse error + +This excpetion is thrown by the library when a parse error occurs. Parse +errors can occur during the deserialization of JSON text as well as when +using JSON Patch. + +Member @a byte holds the byte index of the last read character in the input +file. + +@note For an input with n bytes, 1 is the index of the first character + and n+1 is the index of the terminating null byte or the end of + file. + +Exceptions have ids 1xx. + +name / id | example massage | description +------------------------------ | --------------- | ------------------------- +json.exception.[parse_error](@ref parse_error).101 | "parse error at 2: unexpected end of input; expected string literal" | This error indicates a syntax error while deserializing a JSON text. The error message describes that an unexpected token (character) was encountered, and the member @ref parse_error::byte indicates the error position. +json.exception.[parse_error](@ref parse_error).102 | "parse error at 14: missing or wrong low surrogate" | JSON uses the `\uxxxx` format to describe Unicode characters. Code points above above 0xFFFF are split into two `\uxxxx` entries ("surrogate pairs"). This error indicates that the surrogate pair is incomplete or contains an invalid code point. +json.exception.[parse_error](@ref parse_error).103 | "parse error: code points above 0x10FFFF are invalid" | Unicode supports code points up to 0x10FFFF. Code points above 0x10FFFF are invalid. +json.exception.[parse_error](@ref parse_error).104 | "parse error: JSON patch must be an array of objects" | [RFC 6902](https://tools.ietf.org/html/rfc6902) requires a JSON Patch document to be a JSON document that represents an array of objects. +json.exception.[parse_error](@ref parse_error).105 | "parse error: operation must have string member 'op'" | An operation of a JSON Patch document must contain exactly one "op" member, whose value indicates the operation to perform. Its value must be one of "add", "remove", "replace", "move", "copy", or "test"; other values are errors. +json.exception.[parse_error](@ref parse_error).106 | "parse error: array index '01' must not begin with '0'" | An array index in a JSON Pointer ([RFC 6901](https://tools.ietf.org/html/rfc6901)) may be `0` or any number wihtout a leading `0`. +json.exception.[parse_error](@ref parse_error).107 | "parse error: JSON pointer must be empty or begin with '/' - was: 'foo'" | A JSON Pointer must be a Unicode string containing a sequence of zero or more reference tokens, each prefixed by a `/` character. +json.exception.[parse_error](@ref parse_error).108 | "parse error: escape character '~' must be followed with '0' or '1'" | In a JSON Pointer, only `~0` and `~1` are valid escape sequences. +json.exception.[parse_error](@ref parse_error).109 | "parse error: array index 'one' is not a number" | A JSON Pointer array index must be a number. + +@since version 3.0.0 +*/ +class parse_error : public exception +{ + public: + /*! + @brief create a parse error exception + @param[in] id_ the id of the exception + @param[in] byte_ the byte index where the error occured (or 0 if + the position cannot be determined) + @param[in] what_arg_ the explanatory string + */ + parse_error(int id_, size_t byte_, const std::string& what_arg_) + : exception(id_, "parse_error", "parse error" + + (byte_ != 0 ? (" at " + std::to_string(byte_)) : "") + + ": " + what_arg_), + byte(byte_) + {} + + /*! + @brief byte index of the parse error + + The byte index of the last read character in the input file. + + @note For an input with n bytes, 1 is the index of the first character + and n+1 is the index of the terminating null byte or the end of + file. + */ + const size_t byte; +}; + +/*! +@brief exception indicating errors with iterators + +Exceptions have ids 2xx. + +name / id | example massage | description +------------------------------ | --------------- | ------------------------- +json.exception.[invalid_iterator](@ref invalid_iterator).201 | "iterators are not compatible" | The iterators passed to constructor @ref basic_json(InputIT first, InputIT last) are not compatible, meaning they do not belong to the same container. Therefore, the range (@a first, @a last) is invalid. +json.exception.[invalid_iterator](@ref invalid_iterator).202 | "iterator does not fit current value" | In an erase or insert function, the passed iterator @a pos does not belong to the JSON value for which the function was called. It hence does not define a valid position for the deletion/insertion. +json.exception.[invalid_iterator](@ref invalid_iterator).203 | "iterators do not fit current value" | Either iterator passed to function @ref erase(IteratorType first, IteratorType last) does not belong to the JSON value from which values shall be erased. It hence does not define a valid range to delete values from. +json.exception.[invalid_iterator](@ref invalid_iterator).204 | "iterators out of range" | When an iterator range for a primitive type (number, boolean, or string) is passed to a constructor or an erase function, this range has to be exactly (@ref begin(), @ref end()), because this is the only way the single stored value is expressed. All other ranges are invalid. +json.exception.[invalid_iterator](@ref invalid_iterator).205 | "iterator out of range" | When an iterator for a primitive type (number, boolean, or string) is passed to an erase function, the iterator has to be the @ref begin() iterator, because it is the only way to address the stored value. All other iterators are invalid. +json.exception.[invalid_iterator](@ref invalid_iterator).206 | "cannot construct with iterators from null" | The iterators passed to constructor @ref basic_json(InputIT first, InputIT last) belong to a JSON null value and hence to not define a valid range. +json.exception.[invalid_iterator](@ref invalid_iterator).207 | "cannot use key() for non-object iterators" | The key() member function can only be used on iterators belonging to a JSON object, because other types do not have a concept of a key. +json.exception.[invalid_iterator](@ref invalid_iterator).208 | "cannot use operator[] for object iterators" | The operator[] to specify a concrete offset cannot be used on iterators belonging to a JSON object, because JSON objects are unordered. +json.exception.[invalid_iterator](@ref invalid_iterator).209 | "cannot use offsets with object iterators" | The offset operators (+, -, +=, -=) cannot be used on iterators belonging to a JSON object, because JSON objects are unordered. +json.exception.[invalid_iterator](@ref invalid_iterator).210 | "iterators do not fit" | The iterator range passed to the insert function are not compatible, meaning they do not belong to the same container. Therefore, the range (@a first, @a last) is invalid. +json.exception.[invalid_iterator](@ref invalid_iterator).211 | "passed iterators may not belong to container" | The iterator range passed to the insert function must not be a subrange of the container to insert to. +json.exception.[invalid_iterator](@ref invalid_iterator).212 | "cannot compare iterators of different containers" | When two iterators are compared, they must belong to the same container. +json.exception.[invalid_iterator](@ref invalid_iterator).213 | "cannot compare order of object iterators" | The order of object iterators cannot be compated, because JSON objects are unordered. +json.exception.[invalid_iterator](@ref invalid_iterator).214 | "cannot get value" | Cannot get value for iterator: Either the iterator belongs to a null value or it is an iterator to a primitive type (number, boolean, or string), but the iterator is different to @ref begin(). + +@since version 3.0.0 +*/ +class invalid_iterator : public exception +{ + public: + invalid_iterator(int id_, const std::string& what_arg_) + : exception(id_, "invalid_iterator", what_arg_) + {} +}; + +/*! +@brief exception indicating executing a member function with a wrong type + +Exceptions have ids 3xx. + +name / id | example massage | description +------------------------------ | --------------- | ------------------------- +json.exception.[type_error](@ref type_error).301 | "cannot create object from initializer list" | To create an object from an initializer list, the initializer list must consist only of a list of pairs whose first element is a string. When this constraint is violated, an array is created instead. +json.exception.[type_error](@ref type_error).302 | "type must be object, but is array" | During implicit or explicit value conversion, the JSON type must be compatible to the target type. For instance, a JSON string can only be converted into string types, but not into numbers or boolean types. +json.exception.[type_error](@ref type_error).303 | "incompatible ReferenceType for get_ref, actual type is object" | To retrieve a reference to a value stored in a @ref basic_json object with @ref get_ref, the type of the reference must match the value type. For instance, for a JSON array, the @a ReferenceType must be @ref array_t&. +json.exception.[type_error](@ref type_error).304 | "cannot use at() with string" | The @ref at() member functions can only be executed for certain JSON types. +json.exception.[type_error](@ref type_error).305 | "cannot use operator[] with string" | The @ref operator[] member functions can only be executed for certain JSON types. +json.exception.[type_error](@ref type_error).306 | "cannot use value() with string" | The @ref value() member functions can only be executed for certain JSON types. +json.exception.[type_error](@ref type_error).307 | "cannot use erase() with string" | The @ref erase() member functions can only be executed for certain JSON types. +json.exception.[type_error](@ref type_error).308 | "cannot use push_back() with string" | The @ref push_back() and @ref operator+= member functions can only be executed for certain JSON types. +json.exception.[type_error](@ref type_error).309 | "cannot use insert() with" | The @ref insert() member functions can only be executed for certain JSON types. +json.exception.[type_error](@ref type_error).310 | "cannot use swap() with number" | The @ref swap() member functions can only be executed for certain JSON types. +json.exception.[type_error](@ref type_error).311 | "cannot use emplace_back() with string" | The @ref emplace_back() member function can only be executed for certain JSON types. +json.exception.[type_error](@ref type_error).313 | "invalid value to unflatten" | The @ref unflatten function converts an object whose keys are JSON Pointers back into an arbitrary nested JSON value. The JSON Pointers must not overlap, because then the resulting value would not be well defined. +json.exception.[type_error](@ref type_error).314 | "only objects can be unflattened" | The @ref unflatten function only works for an object whose keys are JSON Pointers. +json.exception.[type_error](@ref type_error).315 | "values in object must be primitive" | The @ref unflatten function only works for an object whose keys are JSON Pointers and whose values are primitive. + +@since version 3.0.0 +*/ +class type_error : public exception +{ + public: + type_error(int id_, const std::string& what_arg_) + : exception(id_, "type_error", what_arg_) + {} +}; + +/*! +@brief exception indicating access out of the defined range + +Exceptions have ids 4xx. + +name / id | example massage | description +------------------------------ | --------------- | ------------------------- +json.exception.[out_of_range](@ref out_of_range).401 | "array index 3 is out of range" | The provided array index @a i is larger than @a size-1. +json.exception.[out_of_range](@ref out_of_range).402 | "array index '-' (3) is out of range" | The special array index `-` in a JSON Pointer never describes a valid element of the array, but the index past the end. +json.exception.[out_of_range](@ref out_of_range).403 | "key 'foo' not found" | The provided key was not found in the JSON object. +json.exception.[out_of_range](@ref out_of_range).404 | "unresolved reference token 'foo'" | A reference token in a JSON Pointer could not be resolved. +json.exception.[out_of_range](@ref out_of_range).405 | "JSON pointer has no parent" | The JSON Patch operations 'remove' and 'add' can not be applied to the root element of the JSON value. + +@since version 3.0.0 +*/ +class out_of_range : public exception +{ + public: + out_of_range(int id_, const std::string& what_arg_) + : exception(id_, "out_of_range", what_arg_) + {} +}; + + /////////////////////////// // JSON type enumeration // /////////////////////////// @@ -617,8 +798,7 @@ void get_arithmetic_value(const BasicJsonType& j, ArithmeticType& val) } default: { - JSON_THROW( - std::domain_error("type must be number, but is " + j.type_name())); + JSON_THROW(type_error(302, "type must be number, but is " + j.type_name())); } } } @@ -628,7 +808,7 @@ void from_json(const BasicJsonType& j, typename BasicJsonType::boolean_t& b) { if (not j.is_boolean()) { - JSON_THROW(std::domain_error("type must be boolean, but is " + j.type_name())); + JSON_THROW(type_error(302, "type must be boolean, but is " + j.type_name())); } b = *j.template get_ptr(); } @@ -638,7 +818,7 @@ void from_json(const BasicJsonType& j, typename BasicJsonType::string_t& s) { if (not j.is_string()) { - JSON_THROW(std::domain_error("type must be string, but is " + j.type_name())); + JSON_THROW(type_error(302, "type must be string, but is " + j.type_name())); } s = *j.template get_ptr(); } @@ -675,7 +855,7 @@ void from_json(const BasicJsonType& j, typename BasicJsonType::array_t& arr) { if (not j.is_array()) { - JSON_THROW(std::domain_error("type must be array, but is " + j.type_name())); + JSON_THROW(type_error(302, "type must be array, but is " + j.type_name())); } arr = *j.template get_ptr(); } @@ -688,13 +868,13 @@ void from_json(const BasicJsonType& j, std::forward_list& l) // (except when it's null.. ?) if (j.is_null()) { - JSON_THROW(std::domain_error("type must be array, but is " + j.type_name())); + JSON_THROW(type_error(302, "type must be array, but is " + j.type_name())); } if (not std::is_same::value) { if (not j.is_array()) { - JSON_THROW(std::domain_error("type must be array, but is " + j.type_name())); + JSON_THROW(type_error(302, "type must be array, but is " + j.type_name())); } } for (auto it = j.rbegin(), end = j.rend(); it != end; ++it) @@ -744,7 +924,7 @@ void from_json(const BasicJsonType& j, CompatibleArrayType& arr) { if (j.is_null()) { - JSON_THROW(std::domain_error("type must be array, but is " + j.type_name())); + JSON_THROW(type_error(302, "type must be array, but is " + j.type_name())); } // when T == BasicJsonType, do not check if value_t is correct @@ -752,7 +932,7 @@ void from_json(const BasicJsonType& j, CompatibleArrayType& arr) { if (not j.is_array()) { - JSON_THROW(std::domain_error("type must be array, but is " + j.type_name())); + JSON_THROW(type_error(302, "type must be array, but is " + j.type_name())); } } from_json_array_impl(j, arr, priority_tag<1> {}); @@ -764,7 +944,7 @@ void from_json(const BasicJsonType& j, CompatibleObjectType& obj) { if (not j.is_object()) { - JSON_THROW(std::domain_error("type must be object, but is " + j.type_name())); + JSON_THROW(type_error(302, "type must be object, but is " + j.type_name())); } auto inner_object = j.template get_ptr(); @@ -814,7 +994,7 @@ void from_json(const BasicJsonType& j, ArithmeticType& val) } default: { - JSON_THROW(std::domain_error("type must be number, but is " + j.type_name())); + JSON_THROW(type_error(302, "type must be number, but is " + j.type_name())); } } } @@ -1046,6 +1226,33 @@ class basic_json template using json_serializer = JSONSerializer; + + //////////////// + // exceptions // + //////////////// + + /// @name exceptions + /// Classes to implement user-defined exceptions. + /// @{ + + /* + name / id | example massage | description + ------------------------------ | --------------- | ------------------------- + json.exception.other.500 | "unsuccessful" | A JSON Patch operation 'test' failed. + */ + + /// @copydoc detail::parse_error + using parse_error = detail::parse_error; + /// @copydoc detail::invalid_iterator + using invalid_iterator = detail::invalid_iterator; + /// @copydoc detail::type_error + using type_error = detail::type_error; + /// @copydoc detail::out_of_range + using out_of_range = detail::out_of_range; + + /// @} + + ///////////////////// // container types // ///////////////////// @@ -1088,179 +1295,6 @@ class basic_json /// @} - public: - //////////////// - // exceptions // - //////////////// - - /// @name exceptions - /// Classes to implement user-defined exceptions. - /// @{ - - /*! - @brief general exception of the @ref basic_json class - - Extension of std::exception objects with a member @a id for exception ids. - - name / id | example massage | description - ------------------------------ | --------------- | ------------------------- - json.exception.[parse_error](@ref parse_error).101 | "parse error at 2: unexpected end of input; expected string literal" | This error indicates a syntax error while deserializing a JSON text. The error message describes that an unexpected token (character) was encountered, and the member @ref parse_error::byte indicates the error position. - json.exception.[parse_error](@ref parse_error).102 | "parse error at 14: missing or wrong low surrogate" | JSON uses the `\uxxxx` format to describe Unicode characters. Code points above above 0xFFFF are split into two `\uxxxx` entries ("surrogate pairs"). This error indicates that the surrogate pair is incomplete or contains an invalid code point. - json.exception.[parse_error](@ref parse_error).103 | "parse error: code points above 0x10FFFF are invalid" | Unicode supports code points up to 0x10FFFF. Code points above 0x10FFFF are invalid. - json.exception.[parse_error](@ref parse_error).104 | "parse error: JSON patch must be an array of objects" | [RFC 6902](https://tools.ietf.org/html/rfc6902) requires a JSON Patch document to be a JSON document that represents an array of objects. - json.exception.[parse_error](@ref parse_error).105 | "parse error: operation must have string member 'op'" | An operation of a JSON Patch document must contain exactly one "op" member, whose value indicates the operation to perform. Its value must be one of "add", "remove", "replace", "move", "copy", or "test"; other values are errors. - json.exception.[parse_error](@ref parse_error).106 | "parse error: array index '01' must not begin with '0'" | An array index in a JSON Pointer ([RFC 6901](https://tools.ietf.org/html/rfc6901)) may be `0` or any number wihtout a leading `0`. - json.exception.[parse_error](@ref parse_error).107 | "parse error: JSON pointer must be empty or begin with '/' - was: 'foo'" | A JSON Pointer must be a Unicode string containing a sequence of zero or more reference tokens, each prefixed by a `/` character. - json.exception.[parse_error](@ref parse_error).108 | "parse error: escape character '~' must be followed with '0' or '1'" | In a JSON Pointer, only `~0` and `~1` are valid escape sequences. - json.exception.[parse_error](@ref parse_error).109 | "parse error: array index 'one' is not a number" | A JSON Pointer array index must be a number. - json.exception.[invalid_iterator](@ref invalid_iterator).201 | "iterators are not compatible" | The iterators passed to constructor @ref basic_json(InputIT first, InputIT last) are not compatible, meaning they do not belong to the same container. Therefore, the range (@a first, @a last) is invalid. - json.exception.[invalid_iterator](@ref invalid_iterator).202 | "iterator does not fit current value" | In an erase or insert function, the passed iterator @a pos does not belong to the JSON value for which the function was called. It hence does not define a valid position for the deletion/insertion. - json.exception.[invalid_iterator](@ref invalid_iterator).203 | "iterators do not fit current value" | Either iterator passed to function @ref erase(IteratorType first, IteratorType last) does not belong to the JSON value from which values shall be erased. It hence does not define a valid range to delete values from. - json.exception.[invalid_iterator](@ref invalid_iterator).204 | "iterators out of range" | When an iterator range for a primitive type (number, boolean, or string) is passed to a constructor or an erase function, this range has to be exactly (@ref begin(), @ref end()), because this is the only way the single stored value is expressed. All other ranges are invalid. - json.exception.[invalid_iterator](@ref invalid_iterator).205 | "iterator out of range" | When an iterator for a primitive type (number, boolean, or string) is passed to an erase function, the iterator has to be the @ref begin() iterator, because it is the only way to address the stored value. All other iterators are invalid. - json.exception.[invalid_iterator](@ref invalid_iterator).206 | "cannot construct with iterators from null" | The iterators passed to constructor @ref basic_json(InputIT first, InputIT last) belong to a JSON null value and hence to not define a valid range. - json.exception.[invalid_iterator](@ref invalid_iterator).207 | "cannot use key() for non-object iterators" | The key() member function can only be used on iterators belonging to a JSON object, because other types do not have a concept of a key. - json.exception.[invalid_iterator](@ref invalid_iterator).208 | "cannot use operator[] for object iterators" | The operator[] to specify a concrete offset cannot be used on iterators belonging to a JSON object, because JSON objects are unordered. - json.exception.[invalid_iterator](@ref invalid_iterator).209 | "cannot use offsets with object iterators" | The offset operators (+, -, +=, -=) cannot be used on iterators belonging to a JSON object, because JSON objects are unordered. - json.exception.[invalid_iterator](@ref invalid_iterator).210 | "iterators do not fit" | The iterator range passed to the insert function are not compatible, meaning they do not belong to the same container. Therefore, the range (@a first, @a last) is invalid. - json.exception.[invalid_iterator](@ref invalid_iterator).211 | "passed iterators may not belong to container" | The iterator range passed to the insert function must not be a subrange of the container to insert to. - json.exception.[invalid_iterator](@ref invalid_iterator).212 | "cannot compare iterators of different containers" | When two iterators are compared, they must belong to the same container. - json.exception.[invalid_iterator](@ref invalid_iterator).213 | "cannot compare order of object iterators" | The order of object iterators cannot be compated, because JSON objects are unordered. - json.exception.[invalid_iterator](@ref invalid_iterator).214 | "cannot get value" | Cannot get value for iterator: Either the iterator belongs to a null value or it is an iterator to a primitive type (number, boolean, or string), but the iterator is different to @ref begin(). - json.exception.[type_error](@ref type_error).301 | "cannot create object from initializer list" | To create an object from an initializer list, the initializer list must consist only of a list of pairs whose first element is a string. When this constraint is violated, an array is created instead. - json.exception.[type_error](@ref type_error).302 | "type must be object, but is array" | During implicit or explicit value conversion, the JSON type must be compatible to the target type. For instance, a JSON string can only be converted into string types, but not into numbers or boolean types. - json.exception.[type_error](@ref type_error).303 | "incompatible ReferenceType for get_ref, actual type is object" | To retrieve a reference to a value stored in a @ref basic_json object with @ref get_ref, the type of the reference must match the value type. For instance, for a JSON array, the @a ReferenceType must be @ref array_t&. - json.exception.[type_error](@ref type_error).304 | "cannot use at() with string" | The @ref at() member functions can only be executed for certain JSON types. - json.exception.[type_error](@ref type_error).305 | "cannot use operator[] with string" | The @ref operator[] member functions can only be executed for certain JSON types. - json.exception.[type_error](@ref type_error).306 | "cannot use value() with string" | The @ref value() member functions can only be executed for certain JSON types. - json.exception.[type_error](@ref type_error).307 | "cannot use erase() with string" | The @ref erase() member functions can only be executed for certain JSON types. - json.exception.[type_error](@ref type_error).308 | "cannot use push_back() with string" | The @ref push_back() and @ref operator+= member functions can only be executed for certain JSON types. - json.exception.[type_error](@ref type_error).309 | "cannot use insert() with" | The @ref insert() member functions can only be executed for certain JSON types. - json.exception.[type_error](@ref type_error).310 | "cannot use swap() with number" | The @ref swap() member functions can only be executed for certain JSON types. - json.exception.[type_error](@ref type_error).311 | "cannot use emplace_back() with string" | The @ref emplace_back() member function can only be executed for certain JSON types. - json.exception.[type_error](@ref type_error).313 | "invalid value to unflatten" | The @ref unflatten function converts an object whose keys are JSON Pointers back into an arbitrary nested JSON value. The JSON Pointers must not overlap, because then the resulting value would not be well defined. - json.exception.[type_error](@ref type_error).314 | "only objects can be unflattened" | The @ref unflatten function only works for an object whose keys are JSON Pointers. - json.exception.[type_error](@ref type_error).315 | "values in object must be primitive" | The @ref unflatten function only works for an object whose keys are JSON Pointers and whose values are primitive. - json.exception.[out_of_range](@ref out_of_range).401 | "array index 3 is out of range" | The provided array index @a i is larger than @a size-1. - json.exception.[out_of_range](@ref out_of_range).402 | "array index '-' (3) is out of range" | The special array index `-` in a JSON Pointer never describes a valid element of the array, but the index past the end. - json.exception.[out_of_range](@ref out_of_range).403 | "key 'foo' not found" | The provided key was not found in the JSON object. - json.exception.[out_of_range](@ref out_of_range).404 | "unresolved reference token 'foo'" | A reference token in a JSON Pointer could not be resolved. - json.exception.[out_of_range](@ref out_of_range).405 | "JSON pointer has no parent" | The JSON Patch operations 'remove' and 'add' can not be applied to the root element of the JSON value. - json.exception.other.500 | "unsuccessful" | A JSON Patch operation 'test' failed. - - @since version 3.0.0 - */ - class exception : public std::exception - { - public: - /// create exception with id an explanatory string - exception(int id_, const std::string& ename, const std::string& what_arg_) - : id(id_), - what_arg("[json.exception." + ename + "." + std::to_string(id_) + "] " + what_arg_) - {} - - /// returns the explanatory string - virtual const char* what() const noexcept - { - return what_arg.c_str(); - } - - /// the id of the exception - const int id; - - private: - /// the explanatory string - const std::string what_arg; - }; - - /*! - @brief exception indicating a parse error - - This excpetion is thrown by the library when a parse error occurs. Parse - errors can occur during the deserialization of JSON text as well as when - using JSON Patch. - - Exceptions have ids 1xx. - - @since version 3.0.0 - */ - class parse_error : public exception - { - public: - /*! - @brief create a parse error exception - @param[in] id_ the id of the exception - @param[in] byte_ the byte index where the error occured (or 0 if - the position cannot be determined) - @param[in] what_arg_ the explanatory string - */ - parse_error(int id_, size_t byte_, const std::string& what_arg_) - : exception(id_, "parse_error", "parse error" + - (byte_ != 0 ? (" at " + std::to_string(byte_)) : "") + - ": " + what_arg_), - byte(byte_) - {} - - /*! - @brief byte index of the parse error - - The byte index of the last read character in the input file. - - @note For an input with n bytes, 1 is the index of the first character - and n+1 is the index of the terminating null byte or the end of - file. - */ - const size_t byte; - }; - - /*! - @brief exception indicating errors with iterators - - Exceptions have ids 2xx. - - @since version 3.0.0 - */ - class invalid_iterator : public exception - { - public: - invalid_iterator(int id_, const std::string& what_arg_) - : exception(id_, "invalid_iterator", what_arg_) - {} - }; - - /*! - @brief exception indicating executing a member function with a wrong type - - Exceptions have ids 3xx. - - @since version 3.0.0 - */ - class type_error : public exception - { - public: - type_error(int id_, const std::string& what_arg_) - : exception(id_, "type_error", what_arg_) - {} - }; - - /*! - @brief exception indicating access out of the defined range - - Exceptions have ids 4xx. - - @since version 3.0.0 - */ - class out_of_range : public exception - { - public: - out_of_range(int id_, const std::string& what_arg_) - : exception(id_, "out_of_range", what_arg_) - {} - }; - - /// @} - - /*! @brief returns the allocator associated with the container */ @@ -3161,7 +3195,7 @@ class basic_json return m_value.boolean; } - JSON_THROW(std::domain_error("type must be boolean, but is " + type_name())); + JSON_THROW(type_error(302, "type must be boolean, but is " + type_name())); } /// get a pointer to the value (object) diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index b47a48cf..4466a509 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -110,6 +110,187 @@ This namespace collects some functions that could not be defined inside the */ namespace detail { +//////////////// +// exceptions // +//////////////// + +/*! +@brief general exception of the @ref basic_json class + +Extension of std::exception objects with a member @a id for exception ids. + +@since version 3.0.0 +*/ +class exception : public std::exception +{ + public: + /// create exception with id an explanatory string + exception(int id_, const std::string& ename, const std::string& what_arg_) + : id(id_), + what_arg("[json.exception." + ename + "." + std::to_string(id_) + "] " + what_arg_) + {} + + /// returns the explanatory string + virtual const char* what() const noexcept + { + return what_arg.c_str(); + } + + /// the id of the exception + const int id; + + private: + /// the explanatory string + const std::string what_arg; +}; + +/*! +@brief exception indicating a parse error + +This excpetion is thrown by the library when a parse error occurs. Parse +errors can occur during the deserialization of JSON text as well as when +using JSON Patch. + +Member @a byte holds the byte index of the last read character in the input +file. + +@note For an input with n bytes, 1 is the index of the first character + and n+1 is the index of the terminating null byte or the end of + file. + +Exceptions have ids 1xx. + +name / id | example massage | description +------------------------------ | --------------- | ------------------------- +json.exception.[parse_error](@ref parse_error).101 | "parse error at 2: unexpected end of input; expected string literal" | This error indicates a syntax error while deserializing a JSON text. The error message describes that an unexpected token (character) was encountered, and the member @ref parse_error::byte indicates the error position. +json.exception.[parse_error](@ref parse_error).102 | "parse error at 14: missing or wrong low surrogate" | JSON uses the `\uxxxx` format to describe Unicode characters. Code points above above 0xFFFF are split into two `\uxxxx` entries ("surrogate pairs"). This error indicates that the surrogate pair is incomplete or contains an invalid code point. +json.exception.[parse_error](@ref parse_error).103 | "parse error: code points above 0x10FFFF are invalid" | Unicode supports code points up to 0x10FFFF. Code points above 0x10FFFF are invalid. +json.exception.[parse_error](@ref parse_error).104 | "parse error: JSON patch must be an array of objects" | [RFC 6902](https://tools.ietf.org/html/rfc6902) requires a JSON Patch document to be a JSON document that represents an array of objects. +json.exception.[parse_error](@ref parse_error).105 | "parse error: operation must have string member 'op'" | An operation of a JSON Patch document must contain exactly one "op" member, whose value indicates the operation to perform. Its value must be one of "add", "remove", "replace", "move", "copy", or "test"; other values are errors. +json.exception.[parse_error](@ref parse_error).106 | "parse error: array index '01' must not begin with '0'" | An array index in a JSON Pointer ([RFC 6901](https://tools.ietf.org/html/rfc6901)) may be `0` or any number wihtout a leading `0`. +json.exception.[parse_error](@ref parse_error).107 | "parse error: JSON pointer must be empty or begin with '/' - was: 'foo'" | A JSON Pointer must be a Unicode string containing a sequence of zero or more reference tokens, each prefixed by a `/` character. +json.exception.[parse_error](@ref parse_error).108 | "parse error: escape character '~' must be followed with '0' or '1'" | In a JSON Pointer, only `~0` and `~1` are valid escape sequences. +json.exception.[parse_error](@ref parse_error).109 | "parse error: array index 'one' is not a number" | A JSON Pointer array index must be a number. + +@since version 3.0.0 +*/ +class parse_error : public exception +{ + public: + /*! + @brief create a parse error exception + @param[in] id_ the id of the exception + @param[in] byte_ the byte index where the error occured (or 0 if + the position cannot be determined) + @param[in] what_arg_ the explanatory string + */ + parse_error(int id_, size_t byte_, const std::string& what_arg_) + : exception(id_, "parse_error", "parse error" + + (byte_ != 0 ? (" at " + std::to_string(byte_)) : "") + + ": " + what_arg_), + byte(byte_) + {} + + /*! + @brief byte index of the parse error + + The byte index of the last read character in the input file. + + @note For an input with n bytes, 1 is the index of the first character + and n+1 is the index of the terminating null byte or the end of + file. + */ + const size_t byte; +}; + +/*! +@brief exception indicating errors with iterators + +Exceptions have ids 2xx. + +name / id | example massage | description +------------------------------ | --------------- | ------------------------- +json.exception.[invalid_iterator](@ref invalid_iterator).201 | "iterators are not compatible" | The iterators passed to constructor @ref basic_json(InputIT first, InputIT last) are not compatible, meaning they do not belong to the same container. Therefore, the range (@a first, @a last) is invalid. +json.exception.[invalid_iterator](@ref invalid_iterator).202 | "iterator does not fit current value" | In an erase or insert function, the passed iterator @a pos does not belong to the JSON value for which the function was called. It hence does not define a valid position for the deletion/insertion. +json.exception.[invalid_iterator](@ref invalid_iterator).203 | "iterators do not fit current value" | Either iterator passed to function @ref erase(IteratorType first, IteratorType last) does not belong to the JSON value from which values shall be erased. It hence does not define a valid range to delete values from. +json.exception.[invalid_iterator](@ref invalid_iterator).204 | "iterators out of range" | When an iterator range for a primitive type (number, boolean, or string) is passed to a constructor or an erase function, this range has to be exactly (@ref begin(), @ref end()), because this is the only way the single stored value is expressed. All other ranges are invalid. +json.exception.[invalid_iterator](@ref invalid_iterator).205 | "iterator out of range" | When an iterator for a primitive type (number, boolean, or string) is passed to an erase function, the iterator has to be the @ref begin() iterator, because it is the only way to address the stored value. All other iterators are invalid. +json.exception.[invalid_iterator](@ref invalid_iterator).206 | "cannot construct with iterators from null" | The iterators passed to constructor @ref basic_json(InputIT first, InputIT last) belong to a JSON null value and hence to not define a valid range. +json.exception.[invalid_iterator](@ref invalid_iterator).207 | "cannot use key() for non-object iterators" | The key() member function can only be used on iterators belonging to a JSON object, because other types do not have a concept of a key. +json.exception.[invalid_iterator](@ref invalid_iterator).208 | "cannot use operator[] for object iterators" | The operator[] to specify a concrete offset cannot be used on iterators belonging to a JSON object, because JSON objects are unordered. +json.exception.[invalid_iterator](@ref invalid_iterator).209 | "cannot use offsets with object iterators" | The offset operators (+, -, +=, -=) cannot be used on iterators belonging to a JSON object, because JSON objects are unordered. +json.exception.[invalid_iterator](@ref invalid_iterator).210 | "iterators do not fit" | The iterator range passed to the insert function are not compatible, meaning they do not belong to the same container. Therefore, the range (@a first, @a last) is invalid. +json.exception.[invalid_iterator](@ref invalid_iterator).211 | "passed iterators may not belong to container" | The iterator range passed to the insert function must not be a subrange of the container to insert to. +json.exception.[invalid_iterator](@ref invalid_iterator).212 | "cannot compare iterators of different containers" | When two iterators are compared, they must belong to the same container. +json.exception.[invalid_iterator](@ref invalid_iterator).213 | "cannot compare order of object iterators" | The order of object iterators cannot be compated, because JSON objects are unordered. +json.exception.[invalid_iterator](@ref invalid_iterator).214 | "cannot get value" | Cannot get value for iterator: Either the iterator belongs to a null value or it is an iterator to a primitive type (number, boolean, or string), but the iterator is different to @ref begin(). + +@since version 3.0.0 +*/ +class invalid_iterator : public exception +{ + public: + invalid_iterator(int id_, const std::string& what_arg_) + : exception(id_, "invalid_iterator", what_arg_) + {} +}; + +/*! +@brief exception indicating executing a member function with a wrong type + +Exceptions have ids 3xx. + +name / id | example massage | description +------------------------------ | --------------- | ------------------------- +json.exception.[type_error](@ref type_error).301 | "cannot create object from initializer list" | To create an object from an initializer list, the initializer list must consist only of a list of pairs whose first element is a string. When this constraint is violated, an array is created instead. +json.exception.[type_error](@ref type_error).302 | "type must be object, but is array" | During implicit or explicit value conversion, the JSON type must be compatible to the target type. For instance, a JSON string can only be converted into string types, but not into numbers or boolean types. +json.exception.[type_error](@ref type_error).303 | "incompatible ReferenceType for get_ref, actual type is object" | To retrieve a reference to a value stored in a @ref basic_json object with @ref get_ref, the type of the reference must match the value type. For instance, for a JSON array, the @a ReferenceType must be @ref array_t&. +json.exception.[type_error](@ref type_error).304 | "cannot use at() with string" | The @ref at() member functions can only be executed for certain JSON types. +json.exception.[type_error](@ref type_error).305 | "cannot use operator[] with string" | The @ref operator[] member functions can only be executed for certain JSON types. +json.exception.[type_error](@ref type_error).306 | "cannot use value() with string" | The @ref value() member functions can only be executed for certain JSON types. +json.exception.[type_error](@ref type_error).307 | "cannot use erase() with string" | The @ref erase() member functions can only be executed for certain JSON types. +json.exception.[type_error](@ref type_error).308 | "cannot use push_back() with string" | The @ref push_back() and @ref operator+= member functions can only be executed for certain JSON types. +json.exception.[type_error](@ref type_error).309 | "cannot use insert() with" | The @ref insert() member functions can only be executed for certain JSON types. +json.exception.[type_error](@ref type_error).310 | "cannot use swap() with number" | The @ref swap() member functions can only be executed for certain JSON types. +json.exception.[type_error](@ref type_error).311 | "cannot use emplace_back() with string" | The @ref emplace_back() member function can only be executed for certain JSON types. +json.exception.[type_error](@ref type_error).313 | "invalid value to unflatten" | The @ref unflatten function converts an object whose keys are JSON Pointers back into an arbitrary nested JSON value. The JSON Pointers must not overlap, because then the resulting value would not be well defined. +json.exception.[type_error](@ref type_error).314 | "only objects can be unflattened" | The @ref unflatten function only works for an object whose keys are JSON Pointers. +json.exception.[type_error](@ref type_error).315 | "values in object must be primitive" | The @ref unflatten function only works for an object whose keys are JSON Pointers and whose values are primitive. + +@since version 3.0.0 +*/ +class type_error : public exception +{ + public: + type_error(int id_, const std::string& what_arg_) + : exception(id_, "type_error", what_arg_) + {} +}; + +/*! +@brief exception indicating access out of the defined range + +Exceptions have ids 4xx. + +name / id | example massage | description +------------------------------ | --------------- | ------------------------- +json.exception.[out_of_range](@ref out_of_range).401 | "array index 3 is out of range" | The provided array index @a i is larger than @a size-1. +json.exception.[out_of_range](@ref out_of_range).402 | "array index '-' (3) is out of range" | The special array index `-` in a JSON Pointer never describes a valid element of the array, but the index past the end. +json.exception.[out_of_range](@ref out_of_range).403 | "key 'foo' not found" | The provided key was not found in the JSON object. +json.exception.[out_of_range](@ref out_of_range).404 | "unresolved reference token 'foo'" | A reference token in a JSON Pointer could not be resolved. +json.exception.[out_of_range](@ref out_of_range).405 | "JSON pointer has no parent" | The JSON Patch operations 'remove' and 'add' can not be applied to the root element of the JSON value. + +@since version 3.0.0 +*/ +class out_of_range : public exception +{ + public: + out_of_range(int id_, const std::string& what_arg_) + : exception(id_, "out_of_range", what_arg_) + {} +}; + + /////////////////////////// // JSON type enumeration // /////////////////////////// @@ -617,8 +798,7 @@ void get_arithmetic_value(const BasicJsonType& j, ArithmeticType& val) } default: { - JSON_THROW( - std::domain_error("type must be number, but is " + j.type_name())); + JSON_THROW(type_error(302, "type must be number, but is " + j.type_name())); } } } @@ -628,7 +808,7 @@ void from_json(const BasicJsonType& j, typename BasicJsonType::boolean_t& b) { if (not j.is_boolean()) { - JSON_THROW(std::domain_error("type must be boolean, but is " + j.type_name())); + JSON_THROW(type_error(302, "type must be boolean, but is " + j.type_name())); } b = *j.template get_ptr(); } @@ -638,7 +818,7 @@ void from_json(const BasicJsonType& j, typename BasicJsonType::string_t& s) { if (not j.is_string()) { - JSON_THROW(std::domain_error("type must be string, but is " + j.type_name())); + JSON_THROW(type_error(302, "type must be string, but is " + j.type_name())); } s = *j.template get_ptr(); } @@ -675,7 +855,7 @@ void from_json(const BasicJsonType& j, typename BasicJsonType::array_t& arr) { if (not j.is_array()) { - JSON_THROW(std::domain_error("type must be array, but is " + j.type_name())); + JSON_THROW(type_error(302, "type must be array, but is " + j.type_name())); } arr = *j.template get_ptr(); } @@ -688,13 +868,13 @@ void from_json(const BasicJsonType& j, std::forward_list& l) // (except when it's null.. ?) if (j.is_null()) { - JSON_THROW(std::domain_error("type must be array, but is " + j.type_name())); + JSON_THROW(type_error(302, "type must be array, but is " + j.type_name())); } if (not std::is_same::value) { if (not j.is_array()) { - JSON_THROW(std::domain_error("type must be array, but is " + j.type_name())); + JSON_THROW(type_error(302, "type must be array, but is " + j.type_name())); } } for (auto it = j.rbegin(), end = j.rend(); it != end; ++it) @@ -744,7 +924,7 @@ void from_json(const BasicJsonType& j, CompatibleArrayType& arr) { if (j.is_null()) { - JSON_THROW(std::domain_error("type must be array, but is " + j.type_name())); + JSON_THROW(type_error(302, "type must be array, but is " + j.type_name())); } // when T == BasicJsonType, do not check if value_t is correct @@ -752,7 +932,7 @@ void from_json(const BasicJsonType& j, CompatibleArrayType& arr) { if (not j.is_array()) { - JSON_THROW(std::domain_error("type must be array, but is " + j.type_name())); + JSON_THROW(type_error(302, "type must be array, but is " + j.type_name())); } } from_json_array_impl(j, arr, priority_tag<1> {}); @@ -764,7 +944,7 @@ void from_json(const BasicJsonType& j, CompatibleObjectType& obj) { if (not j.is_object()) { - JSON_THROW(std::domain_error("type must be object, but is " + j.type_name())); + JSON_THROW(type_error(302, "type must be object, but is " + j.type_name())); } auto inner_object = j.template get_ptr(); @@ -814,7 +994,7 @@ void from_json(const BasicJsonType& j, ArithmeticType& val) } default: { - JSON_THROW(std::domain_error("type must be number, but is " + j.type_name())); + JSON_THROW(type_error(302, "type must be number, but is " + j.type_name())); } } } @@ -1046,6 +1226,33 @@ class basic_json template using json_serializer = JSONSerializer; + + //////////////// + // exceptions // + //////////////// + + /// @name exceptions + /// Classes to implement user-defined exceptions. + /// @{ + + /* + name / id | example massage | description + ------------------------------ | --------------- | ------------------------- + json.exception.other.500 | "unsuccessful" | A JSON Patch operation 'test' failed. + */ + + /// @copydoc detail::parse_error + using parse_error = detail::parse_error; + /// @copydoc detail::invalid_iterator + using invalid_iterator = detail::invalid_iterator; + /// @copydoc detail::type_error + using type_error = detail::type_error; + /// @copydoc detail::out_of_range + using out_of_range = detail::out_of_range; + + /// @} + + ///////////////////// // container types // ///////////////////// @@ -1088,179 +1295,6 @@ class basic_json /// @} - public: - //////////////// - // exceptions // - //////////////// - - /// @name exceptions - /// Classes to implement user-defined exceptions. - /// @{ - - /*! - @brief general exception of the @ref basic_json class - - Extension of std::exception objects with a member @a id for exception ids. - - name / id | example massage | description - ------------------------------ | --------------- | ------------------------- - json.exception.[parse_error](@ref parse_error).101 | "parse error at 2: unexpected end of input; expected string literal" | This error indicates a syntax error while deserializing a JSON text. The error message describes that an unexpected token (character) was encountered, and the member @ref parse_error::byte indicates the error position. - json.exception.[parse_error](@ref parse_error).102 | "parse error at 14: missing or wrong low surrogate" | JSON uses the `\uxxxx` format to describe Unicode characters. Code points above above 0xFFFF are split into two `\uxxxx` entries ("surrogate pairs"). This error indicates that the surrogate pair is incomplete or contains an invalid code point. - json.exception.[parse_error](@ref parse_error).103 | "parse error: code points above 0x10FFFF are invalid" | Unicode supports code points up to 0x10FFFF. Code points above 0x10FFFF are invalid. - json.exception.[parse_error](@ref parse_error).104 | "parse error: JSON patch must be an array of objects" | [RFC 6902](https://tools.ietf.org/html/rfc6902) requires a JSON Patch document to be a JSON document that represents an array of objects. - json.exception.[parse_error](@ref parse_error).105 | "parse error: operation must have string member 'op'" | An operation of a JSON Patch document must contain exactly one "op" member, whose value indicates the operation to perform. Its value must be one of "add", "remove", "replace", "move", "copy", or "test"; other values are errors. - json.exception.[parse_error](@ref parse_error).106 | "parse error: array index '01' must not begin with '0'" | An array index in a JSON Pointer ([RFC 6901](https://tools.ietf.org/html/rfc6901)) may be `0` or any number wihtout a leading `0`. - json.exception.[parse_error](@ref parse_error).107 | "parse error: JSON pointer must be empty or begin with '/' - was: 'foo'" | A JSON Pointer must be a Unicode string containing a sequence of zero or more reference tokens, each prefixed by a `/` character. - json.exception.[parse_error](@ref parse_error).108 | "parse error: escape character '~' must be followed with '0' or '1'" | In a JSON Pointer, only `~0` and `~1` are valid escape sequences. - json.exception.[parse_error](@ref parse_error).109 | "parse error: array index 'one' is not a number" | A JSON Pointer array index must be a number. - json.exception.[invalid_iterator](@ref invalid_iterator).201 | "iterators are not compatible" | The iterators passed to constructor @ref basic_json(InputIT first, InputIT last) are not compatible, meaning they do not belong to the same container. Therefore, the range (@a first, @a last) is invalid. - json.exception.[invalid_iterator](@ref invalid_iterator).202 | "iterator does not fit current value" | In an erase or insert function, the passed iterator @a pos does not belong to the JSON value for which the function was called. It hence does not define a valid position for the deletion/insertion. - json.exception.[invalid_iterator](@ref invalid_iterator).203 | "iterators do not fit current value" | Either iterator passed to function @ref erase(IteratorType first, IteratorType last) does not belong to the JSON value from which values shall be erased. It hence does not define a valid range to delete values from. - json.exception.[invalid_iterator](@ref invalid_iterator).204 | "iterators out of range" | When an iterator range for a primitive type (number, boolean, or string) is passed to a constructor or an erase function, this range has to be exactly (@ref begin(), @ref end()), because this is the only way the single stored value is expressed. All other ranges are invalid. - json.exception.[invalid_iterator](@ref invalid_iterator).205 | "iterator out of range" | When an iterator for a primitive type (number, boolean, or string) is passed to an erase function, the iterator has to be the @ref begin() iterator, because it is the only way to address the stored value. All other iterators are invalid. - json.exception.[invalid_iterator](@ref invalid_iterator).206 | "cannot construct with iterators from null" | The iterators passed to constructor @ref basic_json(InputIT first, InputIT last) belong to a JSON null value and hence to not define a valid range. - json.exception.[invalid_iterator](@ref invalid_iterator).207 | "cannot use key() for non-object iterators" | The key() member function can only be used on iterators belonging to a JSON object, because other types do not have a concept of a key. - json.exception.[invalid_iterator](@ref invalid_iterator).208 | "cannot use operator[] for object iterators" | The operator[] to specify a concrete offset cannot be used on iterators belonging to a JSON object, because JSON objects are unordered. - json.exception.[invalid_iterator](@ref invalid_iterator).209 | "cannot use offsets with object iterators" | The offset operators (+, -, +=, -=) cannot be used on iterators belonging to a JSON object, because JSON objects are unordered. - json.exception.[invalid_iterator](@ref invalid_iterator).210 | "iterators do not fit" | The iterator range passed to the insert function are not compatible, meaning they do not belong to the same container. Therefore, the range (@a first, @a last) is invalid. - json.exception.[invalid_iterator](@ref invalid_iterator).211 | "passed iterators may not belong to container" | The iterator range passed to the insert function must not be a subrange of the container to insert to. - json.exception.[invalid_iterator](@ref invalid_iterator).212 | "cannot compare iterators of different containers" | When two iterators are compared, they must belong to the same container. - json.exception.[invalid_iterator](@ref invalid_iterator).213 | "cannot compare order of object iterators" | The order of object iterators cannot be compated, because JSON objects are unordered. - json.exception.[invalid_iterator](@ref invalid_iterator).214 | "cannot get value" | Cannot get value for iterator: Either the iterator belongs to a null value or it is an iterator to a primitive type (number, boolean, or string), but the iterator is different to @ref begin(). - json.exception.[type_error](@ref type_error).301 | "cannot create object from initializer list" | To create an object from an initializer list, the initializer list must consist only of a list of pairs whose first element is a string. When this constraint is violated, an array is created instead. - json.exception.[type_error](@ref type_error).302 | "type must be object, but is array" | During implicit or explicit value conversion, the JSON type must be compatible to the target type. For instance, a JSON string can only be converted into string types, but not into numbers or boolean types. - json.exception.[type_error](@ref type_error).303 | "incompatible ReferenceType for get_ref, actual type is object" | To retrieve a reference to a value stored in a @ref basic_json object with @ref get_ref, the type of the reference must match the value type. For instance, for a JSON array, the @a ReferenceType must be @ref array_t&. - json.exception.[type_error](@ref type_error).304 | "cannot use at() with string" | The @ref at() member functions can only be executed for certain JSON types. - json.exception.[type_error](@ref type_error).305 | "cannot use operator[] with string" | The @ref operator[] member functions can only be executed for certain JSON types. - json.exception.[type_error](@ref type_error).306 | "cannot use value() with string" | The @ref value() member functions can only be executed for certain JSON types. - json.exception.[type_error](@ref type_error).307 | "cannot use erase() with string" | The @ref erase() member functions can only be executed for certain JSON types. - json.exception.[type_error](@ref type_error).308 | "cannot use push_back() with string" | The @ref push_back() and @ref operator+= member functions can only be executed for certain JSON types. - json.exception.[type_error](@ref type_error).309 | "cannot use insert() with" | The @ref insert() member functions can only be executed for certain JSON types. - json.exception.[type_error](@ref type_error).310 | "cannot use swap() with number" | The @ref swap() member functions can only be executed for certain JSON types. - json.exception.[type_error](@ref type_error).311 | "cannot use emplace_back() with string" | The @ref emplace_back() member function can only be executed for certain JSON types. - json.exception.[type_error](@ref type_error).313 | "invalid value to unflatten" | The @ref unflatten function converts an object whose keys are JSON Pointers back into an arbitrary nested JSON value. The JSON Pointers must not overlap, because then the resulting value would not be well defined. - json.exception.[type_error](@ref type_error).314 | "only objects can be unflattened" | The @ref unflatten function only works for an object whose keys are JSON Pointers. - json.exception.[type_error](@ref type_error).315 | "values in object must be primitive" | The @ref unflatten function only works for an object whose keys are JSON Pointers and whose values are primitive. - json.exception.[out_of_range](@ref out_of_range).401 | "array index 3 is out of range" | The provided array index @a i is larger than @a size-1. - json.exception.[out_of_range](@ref out_of_range).402 | "array index '-' (3) is out of range" | The special array index `-` in a JSON Pointer never describes a valid element of the array, but the index past the end. - json.exception.[out_of_range](@ref out_of_range).403 | "key 'foo' not found" | The provided key was not found in the JSON object. - json.exception.[out_of_range](@ref out_of_range).404 | "unresolved reference token 'foo'" | A reference token in a JSON Pointer could not be resolved. - json.exception.[out_of_range](@ref out_of_range).405 | "JSON pointer has no parent" | The JSON Patch operations 'remove' and 'add' can not be applied to the root element of the JSON value. - json.exception.other.500 | "unsuccessful" | A JSON Patch operation 'test' failed. - - @since version 3.0.0 - */ - class exception : public std::exception - { - public: - /// create exception with id an explanatory string - exception(int id_, const std::string& ename, const std::string& what_arg_) - : id(id_), - what_arg("[json.exception." + ename + "." + std::to_string(id_) + "] " + what_arg_) - {} - - /// returns the explanatory string - virtual const char* what() const noexcept - { - return what_arg.c_str(); - } - - /// the id of the exception - const int id; - - private: - /// the explanatory string - const std::string what_arg; - }; - - /*! - @brief exception indicating a parse error - - This excpetion is thrown by the library when a parse error occurs. Parse - errors can occur during the deserialization of JSON text as well as when - using JSON Patch. - - Exceptions have ids 1xx. - - @since version 3.0.0 - */ - class parse_error : public exception - { - public: - /*! - @brief create a parse error exception - @param[in] id_ the id of the exception - @param[in] byte_ the byte index where the error occured (or 0 if - the position cannot be determined) - @param[in] what_arg_ the explanatory string - */ - parse_error(int id_, size_t byte_, const std::string& what_arg_) - : exception(id_, "parse_error", "parse error" + - (byte_ != 0 ? (" at " + std::to_string(byte_)) : "") + - ": " + what_arg_), - byte(byte_) - {} - - /*! - @brief byte index of the parse error - - The byte index of the last read character in the input file. - - @note For an input with n bytes, 1 is the index of the first character - and n+1 is the index of the terminating null byte or the end of - file. - */ - const size_t byte; - }; - - /*! - @brief exception indicating errors with iterators - - Exceptions have ids 2xx. - - @since version 3.0.0 - */ - class invalid_iterator : public exception - { - public: - invalid_iterator(int id_, const std::string& what_arg_) - : exception(id_, "invalid_iterator", what_arg_) - {} - }; - - /*! - @brief exception indicating executing a member function with a wrong type - - Exceptions have ids 3xx. - - @since version 3.0.0 - */ - class type_error : public exception - { - public: - type_error(int id_, const std::string& what_arg_) - : exception(id_, "type_error", what_arg_) - {} - }; - - /*! - @brief exception indicating access out of the defined range - - Exceptions have ids 4xx. - - @since version 3.0.0 - */ - class out_of_range : public exception - { - public: - out_of_range(int id_, const std::string& what_arg_) - : exception(id_, "out_of_range", what_arg_) - {} - }; - - /// @} - - /*! @brief returns the allocator associated with the container */ @@ -3161,7 +3195,7 @@ class basic_json return m_value.boolean; } - JSON_THROW(std::domain_error("type must be boolean, but is " + type_name())); + JSON_THROW(type_error(302, "type must be boolean, but is " + type_name())); } /// get a pointer to the value (object) diff --git a/test/src/unit-conversions.cpp b/test/src/unit-conversions.cpp index e4545fbe..3a83aef3 100644 --- a/test/src/unit-conversions.cpp +++ b/test/src/unit-conversions.cpp @@ -78,28 +78,28 @@ TEST_CASE("value conversion") SECTION("exception in case of a non-object type") { - CHECK_THROWS_AS(json(json::value_t::null).get(), std::logic_error); - CHECK_THROWS_AS(json(json::value_t::array).get(), std::logic_error); - CHECK_THROWS_AS(json(json::value_t::string).get(), std::logic_error); - CHECK_THROWS_AS(json(json::value_t::boolean).get(), std::logic_error); - CHECK_THROWS_AS(json(json::value_t::number_integer).get(), std::logic_error); - CHECK_THROWS_AS(json(json::value_t::number_unsigned).get(), std::logic_error); - CHECK_THROWS_AS(json(json::value_t::number_float).get(), std::logic_error); + CHECK_THROWS_AS(json(json::value_t::null).get(), json::type_error); + CHECK_THROWS_AS(json(json::value_t::array).get(), json::type_error); + CHECK_THROWS_AS(json(json::value_t::string).get(), json::type_error); + CHECK_THROWS_AS(json(json::value_t::boolean).get(), json::type_error); + CHECK_THROWS_AS(json(json::value_t::number_integer).get(), json::type_error); + CHECK_THROWS_AS(json(json::value_t::number_unsigned).get(), json::type_error); + CHECK_THROWS_AS(json(json::value_t::number_float).get(), json::type_error); CHECK_THROWS_WITH(json(json::value_t::null).get(), - "type must be object, but is null"); + "[json.exception.type_error.302] type must be object, but is null"); CHECK_THROWS_WITH(json(json::value_t::array).get(), - "type must be object, but is array"); + "[json.exception.type_error.302] type must be object, but is array"); CHECK_THROWS_WITH(json(json::value_t::string).get(), - "type must be object, but is string"); + "[json.exception.type_error.302] type must be object, but is string"); CHECK_THROWS_WITH(json(json::value_t::boolean).get(), - "type must be object, but is boolean"); + "[json.exception.type_error.302] type must be object, but is boolean"); CHECK_THROWS_WITH(json(json::value_t::number_integer).get(), - "type must be object, but is number"); + "[json.exception.type_error.302] type must be object, but is number"); CHECK_THROWS_WITH(json(json::value_t::number_unsigned).get(), - "type must be object, but is number"); + "[json.exception.type_error.302] type must be object, but is number"); CHECK_THROWS_WITH(json(json::value_t::number_float).get(), - "type must be object, but is number"); + "[json.exception.type_error.302] type must be object, but is number"); } } @@ -161,9 +161,9 @@ TEST_CASE("value conversion") std::forward_list a = j.get>(); CHECK(json(a) == j); - CHECK_THROWS_AS(json(json::value_t::null).get>(), std::logic_error); + CHECK_THROWS_AS(json(json::value_t::null).get>(), json::type_error); CHECK_THROWS_WITH(json(json::value_t::null).get>(), - "type must be array, but is null"); + "[json.exception.type_error.302] type must be array, but is null"); } SECTION("std::vector") @@ -171,16 +171,16 @@ TEST_CASE("value conversion") std::vector a = j.get>(); CHECK(json(a) == j); - CHECK_THROWS_AS(json(json::value_t::null).get>(), std::logic_error); + CHECK_THROWS_AS(json(json::value_t::null).get>(), json::type_error); CHECK_THROWS_WITH(json(json::value_t::null).get>(), - "type must be array, but is null"); + "[json.exception.type_error.302] type must be array, but is null"); #if not defined(JSON_NOEXCEPTION) SECTION("reserve is called on containers that supports it") { // making the call to from_json throw in order to check capacity std::vector v; - CHECK_THROWS_AS(nlohmann::from_json(j, v), std::logic_error); + CHECK_THROWS_AS(nlohmann::from_json(j, v), json::type_error); CHECK(v.capacity() == j.size()); // make sure all values are properly copied @@ -198,30 +198,30 @@ TEST_CASE("value conversion") SECTION("exception in case of a non-array type") { - CHECK_THROWS_AS(json(json::value_t::null).get(), std::logic_error); - CHECK_THROWS_AS(json(json::value_t::object).get(), std::logic_error); - CHECK_THROWS_AS(json(json::value_t::string).get(), std::logic_error); - CHECK_THROWS_AS(json(json::value_t::boolean).get(), std::logic_error); - CHECK_THROWS_AS(json(json::value_t::number_integer).get(), std::logic_error); - CHECK_THROWS_AS(json(json::value_t::number_unsigned).get(), std::logic_error); - CHECK_THROWS_AS(json(json::value_t::number_float).get(), std::logic_error); + CHECK_THROWS_AS(json(json::value_t::null).get(), json::type_error); + CHECK_THROWS_AS(json(json::value_t::object).get(), json::type_error); + CHECK_THROWS_AS(json(json::value_t::string).get(), json::type_error); + CHECK_THROWS_AS(json(json::value_t::boolean).get(), json::type_error); + CHECK_THROWS_AS(json(json::value_t::number_integer).get(), json::type_error); + CHECK_THROWS_AS(json(json::value_t::number_unsigned).get(), json::type_error); + CHECK_THROWS_AS(json(json::value_t::number_float).get(), json::type_error); CHECK_THROWS_WITH(json(json::value_t::object).get>(), - "type must be array, but is object"); + "[json.exception.type_error.302] type must be array, but is object"); CHECK_THROWS_WITH(json(json::value_t::null).get(), - "type must be array, but is null"); + "[json.exception.type_error.302] type must be array, but is null"); CHECK_THROWS_WITH(json(json::value_t::object).get(), - "type must be array, but is object"); + "[json.exception.type_error.302] type must be array, but is object"); CHECK_THROWS_WITH(json(json::value_t::string).get(), - "type must be array, but is string"); + "[json.exception.type_error.302] type must be array, but is string"); CHECK_THROWS_WITH(json(json::value_t::boolean).get(), - "type must be array, but is boolean"); + "[json.exception.type_error.302] type must be array, but is boolean"); CHECK_THROWS_WITH(json(json::value_t::number_integer).get(), - "type must be array, but is number"); + "[json.exception.type_error.302] type must be array, but is number"); CHECK_THROWS_WITH(json(json::value_t::number_unsigned).get(), - "type must be array, but is number"); + "[json.exception.type_error.302] type must be array, but is number"); CHECK_THROWS_WITH(json(json::value_t::number_float).get(), - "type must be array, but is number"); + "[json.exception.type_error.302] type must be array, but is number"); } } @@ -280,28 +280,28 @@ TEST_CASE("value conversion") SECTION("exception in case of a non-string type") { - CHECK_THROWS_AS(json(json::value_t::null).get(), std::logic_error); - CHECK_THROWS_AS(json(json::value_t::object).get(), std::logic_error); - CHECK_THROWS_AS(json(json::value_t::array).get(), std::logic_error); - CHECK_THROWS_AS(json(json::value_t::boolean).get(), std::logic_error); - CHECK_THROWS_AS(json(json::value_t::number_integer).get(), std::logic_error); - CHECK_THROWS_AS(json(json::value_t::number_unsigned).get(), std::logic_error); - CHECK_THROWS_AS(json(json::value_t::number_float).get(), std::logic_error); + CHECK_THROWS_AS(json(json::value_t::null).get(), json::type_error); + CHECK_THROWS_AS(json(json::value_t::object).get(), json::type_error); + CHECK_THROWS_AS(json(json::value_t::array).get(), json::type_error); + CHECK_THROWS_AS(json(json::value_t::boolean).get(), json::type_error); + CHECK_THROWS_AS(json(json::value_t::number_integer).get(), json::type_error); + CHECK_THROWS_AS(json(json::value_t::number_unsigned).get(), json::type_error); + CHECK_THROWS_AS(json(json::value_t::number_float).get(), json::type_error); CHECK_THROWS_WITH(json(json::value_t::null).get(), - "type must be string, but is null"); + "[json.exception.type_error.302] type must be string, but is null"); CHECK_THROWS_WITH(json(json::value_t::object).get(), - "type must be string, but is object"); + "[json.exception.type_error.302] type must be string, but is object"); CHECK_THROWS_WITH(json(json::value_t::array).get(), - "type must be string, but is array"); + "[json.exception.type_error.302] type must be string, but is array"); CHECK_THROWS_WITH(json(json::value_t::boolean).get(), - "type must be string, but is boolean"); + "[json.exception.type_error.302] type must be string, but is boolean"); CHECK_THROWS_WITH(json(json::value_t::number_integer).get(), - "type must be string, but is number"); + "[json.exception.type_error.302] type must be string, but is number"); CHECK_THROWS_WITH(json(json::value_t::number_unsigned).get(), - "type must be string, but is number"); + "[json.exception.type_error.302] type must be string, but is number"); CHECK_THROWS_WITH(json(json::value_t::number_float).get(), - "type must be string, but is number"); + "[json.exception.type_error.302] type must be string, but is number"); } } @@ -342,28 +342,28 @@ TEST_CASE("value conversion") SECTION("exception in case of a non-string type") { - CHECK_THROWS_AS(json(json::value_t::null).get(), std::logic_error); - CHECK_THROWS_AS(json(json::value_t::object).get(), std::logic_error); - CHECK_THROWS_AS(json(json::value_t::array).get(), std::logic_error); - CHECK_THROWS_AS(json(json::value_t::string).get(), std::logic_error); - CHECK_THROWS_AS(json(json::value_t::number_integer).get(), std::logic_error); - CHECK_THROWS_AS(json(json::value_t::number_unsigned).get(), std::logic_error); - CHECK_THROWS_AS(json(json::value_t::number_float).get(), std::logic_error); + CHECK_THROWS_AS(json(json::value_t::null).get(), json::type_error); + CHECK_THROWS_AS(json(json::value_t::object).get(), json::type_error); + CHECK_THROWS_AS(json(json::value_t::array).get(), json::type_error); + CHECK_THROWS_AS(json(json::value_t::string).get(), json::type_error); + CHECK_THROWS_AS(json(json::value_t::number_integer).get(), json::type_error); + CHECK_THROWS_AS(json(json::value_t::number_unsigned).get(), json::type_error); + CHECK_THROWS_AS(json(json::value_t::number_float).get(), json::type_error); CHECK_THROWS_WITH(json(json::value_t::null).get(), - "type must be boolean, but is null"); + "[json.exception.type_error.302] type must be boolean, but is null"); CHECK_THROWS_WITH(json(json::value_t::object).get(), - "type must be boolean, but is object"); + "[json.exception.type_error.302] type must be boolean, but is object"); CHECK_THROWS_WITH(json(json::value_t::array).get(), - "type must be boolean, but is array"); + "[json.exception.type_error.302] type must be boolean, but is array"); CHECK_THROWS_WITH(json(json::value_t::string).get(), - "type must be boolean, but is string"); + "[json.exception.type_error.302] type must be boolean, but is string"); CHECK_THROWS_WITH(json(json::value_t::number_integer).get(), - "type must be boolean, but is number"); + "[json.exception.type_error.302] type must be boolean, but is number"); CHECK_THROWS_WITH(json(json::value_t::number_unsigned).get(), - "type must be boolean, but is number"); + "[json.exception.type_error.302] type must be boolean, but is number"); CHECK_THROWS_WITH(json(json::value_t::number_float).get(), - "type must be boolean, but is number"); + "[json.exception.type_error.302] type must be boolean, but is number"); } } @@ -598,22 +598,22 @@ TEST_CASE("value conversion") SECTION("exception in case of a non-number type") { - CHECK_THROWS_AS(json(json::value_t::null).get(), std::logic_error); - CHECK_THROWS_AS(json(json::value_t::object).get(), std::logic_error); - CHECK_THROWS_AS(json(json::value_t::array).get(), std::logic_error); - CHECK_THROWS_AS(json(json::value_t::string).get(), std::logic_error); - CHECK_THROWS_AS(json(json::value_t::boolean).get(), std::logic_error); + CHECK_THROWS_AS(json(json::value_t::null).get(), json::type_error); + CHECK_THROWS_AS(json(json::value_t::object).get(), json::type_error); + CHECK_THROWS_AS(json(json::value_t::array).get(), json::type_error); + CHECK_THROWS_AS(json(json::value_t::string).get(), json::type_error); + CHECK_THROWS_AS(json(json::value_t::boolean).get(), json::type_error); CHECK_THROWS_WITH(json(json::value_t::null).get(), - "type must be number, but is null"); + "[json.exception.type_error.302] type must be number, but is null"); CHECK_THROWS_WITH(json(json::value_t::object).get(), - "type must be number, but is object"); + "[json.exception.type_error.302] type must be number, but is object"); CHECK_THROWS_WITH(json(json::value_t::array).get(), - "type must be number, but is array"); + "[json.exception.type_error.302] type must be number, but is array"); CHECK_THROWS_WITH(json(json::value_t::string).get(), - "type must be number, but is string"); + "[json.exception.type_error.302] type must be number, but is string"); CHECK_THROWS_WITH(json(json::value_t::boolean).get(), - "type must be number, but is boolean"); + "[json.exception.type_error.302] type must be number, but is boolean"); CHECK_NOTHROW(json(json::value_t::number_float).get()); CHECK_NOTHROW(json(json::value_t::number_float).get()); @@ -857,22 +857,22 @@ TEST_CASE("value conversion") SECTION("exception in case of a non-string type") { - CHECK_THROWS_AS(json(json::value_t::null).get(), std::logic_error); - CHECK_THROWS_AS(json(json::value_t::object).get(), std::logic_error); - CHECK_THROWS_AS(json(json::value_t::array).get(), std::logic_error); - CHECK_THROWS_AS(json(json::value_t::string).get(), std::logic_error); - CHECK_THROWS_AS(json(json::value_t::boolean).get(), std::logic_error); + CHECK_THROWS_AS(json(json::value_t::null).get(), json::type_error); + CHECK_THROWS_AS(json(json::value_t::object).get(), json::type_error); + CHECK_THROWS_AS(json(json::value_t::array).get(), json::type_error); + CHECK_THROWS_AS(json(json::value_t::string).get(), json::type_error); + CHECK_THROWS_AS(json(json::value_t::boolean).get(), json::type_error); CHECK_THROWS_WITH(json(json::value_t::null).get(), - "type must be number, but is null"); + "[json.exception.type_error.302] type must be number, but is null"); CHECK_THROWS_WITH(json(json::value_t::object).get(), - "type must be number, but is object"); + "[json.exception.type_error.302] type must be number, but is object"); CHECK_THROWS_WITH(json(json::value_t::array).get(), - "type must be number, but is array"); + "[json.exception.type_error.302] type must be number, but is array"); CHECK_THROWS_WITH(json(json::value_t::string).get(), - "type must be number, but is string"); + "[json.exception.type_error.302] type must be number, but is string"); CHECK_THROWS_WITH(json(json::value_t::boolean).get(), - "type must be number, but is boolean"); + "[json.exception.type_error.302] type must be number, but is boolean"); CHECK_NOTHROW(json(json::value_t::number_integer).get()); CHECK_NOTHROW(json(json::value_t::number_unsigned).get()); @@ -954,8 +954,8 @@ TEST_CASE("value conversion") SECTION("exception in case of a non-object type") { - CHECK_THROWS_AS((json().get>()), std::logic_error); - CHECK_THROWS_WITH((json().get>()), "type must be object, but is null"); + CHECK_THROWS_AS((json().get>()), json::type_error); + CHECK_THROWS_WITH((json().get>()), "[json.exception.type_error.302] type must be object, but is null"); } } @@ -1023,17 +1023,17 @@ TEST_CASE("value conversion") SECTION("exception in case of a non-object type") { - CHECK_THROWS_AS((json().get>()), std::logic_error); - CHECK_THROWS_AS((json().get>()), std::logic_error); - CHECK_THROWS_AS((json().get>()), std::logic_error); - CHECK_THROWS_AS((json().get>()), std::logic_error); + CHECK_THROWS_AS((json().get>()), json::type_error); + CHECK_THROWS_AS((json().get>()), json::type_error); + CHECK_THROWS_AS((json().get>()), json::type_error); + CHECK_THROWS_AS((json().get>()), json::type_error); // does type really must be an array? or it rather must not be null? // that's what I thought when other test like this one broke - CHECK_THROWS_WITH((json().get>()), "type must be array, but is null"); - CHECK_THROWS_WITH((json().get>()), "type must be array, but is null"); - CHECK_THROWS_WITH((json().get>()), "type must be array, but is null"); - CHECK_THROWS_WITH((json().get>()), "type must be array, but is null"); + CHECK_THROWS_WITH((json().get>()), "[json.exception.type_error.302] type must be array, but is null"); + CHECK_THROWS_WITH((json().get>()), "[json.exception.type_error.302] type must be array, but is null"); + CHECK_THROWS_WITH((json().get>()), "[json.exception.type_error.302] type must be array, but is null"); + CHECK_THROWS_WITH((json().get>()), "[json.exception.type_error.302] type must be array, but is null"); } } } From 21ec0e78069154d09c65a9b4c883abf735ba3be8 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Mon, 6 Mar 2017 21:00:13 +0100 Subject: [PATCH 046/530] :hammer: added user-defined exception 110 --- Makefile | 2 ++ src/json.hpp | 12 ++++++--- src/json.hpp.re2c | 12 ++++++--- test/src/unit-cbor.cpp | 61 +++++++++++++++++++++++++++++++----------- 4 files changed, 64 insertions(+), 23 deletions(-) diff --git a/Makefile b/Makefile index 1b7cf8c1..f52511c6 100644 --- a/Makefile +++ b/Makefile @@ -49,6 +49,7 @@ doctest: # -Wno-documentation-unknown-command: code uses user-defined commands like @complexity # -Wno-exit-time-destructors: warning in Catch code # -Wno-keyword-macro: unit-tests use "#define private public" +# -Wno-weak-vtables: exception class is defined inline, but has virtual method # -Wno-range-loop-analysis: iterator_wrapper tests tests "for(const auto i...)" pedantic: $(MAKE) json_unit CXXFLAGS="\ @@ -58,6 +59,7 @@ pedantic: -Wno-documentation-unknown-command \ -Wno-exit-time-destructors \ -Wno-keyword-macro \ + -Wno-weak-vtables \ -Wno-range-loop-analysis" diff --git a/src/json.hpp b/src/json.hpp index d7108ed6..6a36b85d 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -156,7 +156,8 @@ file. @note For an input with n bytes, 1 is the index of the first character and n+1 is the index of the terminating null byte or the end of - file. + file. This also holds true when reading a byte vector (CBOR or + MessagePack). Exceptions have ids 1xx. @@ -171,6 +172,8 @@ json.exception.[parse_error](@ref parse_error).106 | "parse error: array index ' json.exception.[parse_error](@ref parse_error).107 | "parse error: JSON pointer must be empty or begin with '/' - was: 'foo'" | A JSON Pointer must be a Unicode string containing a sequence of zero or more reference tokens, each prefixed by a `/` character. json.exception.[parse_error](@ref parse_error).108 | "parse error: escape character '~' must be followed with '0' or '1'" | In a JSON Pointer, only `~0` and `~1` are valid escape sequences. json.exception.[parse_error](@ref parse_error).109 | "parse error: array index 'one' is not a number" | A JSON Pointer array index must be a number. +json.exception.[parse_error](@ref parse_error).110 | "parse error at 1: cannot read 2 bytes from vector" | When parsing CBOR or MessagePack, the byte vector ends before the complete value has been read. + @since version 3.0.0 */ @@ -198,7 +201,8 @@ class parse_error : public exception @note For an input with n bytes, 1 is the index of the first character and n+1 is the index of the terminating null byte or the end of - file. + file. This also holds true when reading a byte vector (CBOR or + MessagePack). */ const size_t byte; }; @@ -7344,7 +7348,7 @@ class basic_json @tparam T the integral return type - @throw std::out_of_range if there are less than sizeof(T)+1 bytes in the + @throw parse_error.110 if there are less than sizeof(T)+1 bytes in the vector @a vec to read In the for loop, the bytes from the vector are copied in reverse order into @@ -7371,7 +7375,7 @@ class basic_json { if (current_index + sizeof(T) + 1 > vec.size()) { - JSON_THROW(std::out_of_range("cannot read " + std::to_string(sizeof(T)) + " bytes from vector")); + JSON_THROW(parse_error(110, current_index + 1, "cannot read " + std::to_string(sizeof(T)) + " bytes from vector")); } T result; diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 4466a509..4ee6cced 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -156,7 +156,8 @@ file. @note For an input with n bytes, 1 is the index of the first character and n+1 is the index of the terminating null byte or the end of - file. + file. This also holds true when reading a byte vector (CBOR or + MessagePack). Exceptions have ids 1xx. @@ -171,6 +172,8 @@ json.exception.[parse_error](@ref parse_error).106 | "parse error: array index ' json.exception.[parse_error](@ref parse_error).107 | "parse error: JSON pointer must be empty or begin with '/' - was: 'foo'" | A JSON Pointer must be a Unicode string containing a sequence of zero or more reference tokens, each prefixed by a `/` character. json.exception.[parse_error](@ref parse_error).108 | "parse error: escape character '~' must be followed with '0' or '1'" | In a JSON Pointer, only `~0` and `~1` are valid escape sequences. json.exception.[parse_error](@ref parse_error).109 | "parse error: array index 'one' is not a number" | A JSON Pointer array index must be a number. +json.exception.[parse_error](@ref parse_error).110 | "parse error at 1: cannot read 2 bytes from vector" | When parsing CBOR or MessagePack, the byte vector ends before the complete value has been read. + @since version 3.0.0 */ @@ -198,7 +201,8 @@ class parse_error : public exception @note For an input with n bytes, 1 is the index of the first character and n+1 is the index of the terminating null byte or the end of - file. + file. This also holds true when reading a byte vector (CBOR or + MessagePack). */ const size_t byte; }; @@ -7344,7 +7348,7 @@ class basic_json @tparam T the integral return type - @throw std::out_of_range if there are less than sizeof(T)+1 bytes in the + @throw parse_error.110 if there are less than sizeof(T)+1 bytes in the vector @a vec to read In the for loop, the bytes from the vector are copied in reverse order into @@ -7371,7 +7375,7 @@ class basic_json { if (current_index + sizeof(T) + 1 > vec.size()) { - JSON_THROW(std::out_of_range("cannot read " + std::to_string(sizeof(T)) + " bytes from vector")); + JSON_THROW(parse_error(110, current_index + 1, "cannot read " + std::to_string(sizeof(T)) + " bytes from vector")); } T result; diff --git a/test/src/unit-cbor.cpp b/test/src/unit-cbor.cpp index 84b280bc..9525c722 100644 --- a/test/src/unit-cbor.cpp +++ b/test/src/unit-cbor.cpp @@ -1145,21 +1145,52 @@ TEST_CASE("CBOR") { SECTION("too short byte vector") { - CHECK_THROWS_AS(json::from_cbor(std::vector({0x18})), std::out_of_range); - CHECK_THROWS_AS(json::from_cbor(std::vector({0x19})), std::out_of_range); - CHECK_THROWS_AS(json::from_cbor(std::vector({0x19, 0x00})), std::out_of_range); - CHECK_THROWS_AS(json::from_cbor(std::vector({0x1a})), std::out_of_range); - CHECK_THROWS_AS(json::from_cbor(std::vector({0x1a, 0x00})), std::out_of_range); - CHECK_THROWS_AS(json::from_cbor(std::vector({0x1a, 0x00, 0x00})), std::out_of_range); - CHECK_THROWS_AS(json::from_cbor(std::vector({0x1a, 0x00, 0x00, 0x00})), std::out_of_range); - CHECK_THROWS_AS(json::from_cbor(std::vector({0x1b})), std::out_of_range); - CHECK_THROWS_AS(json::from_cbor(std::vector({0x1b, 0x00})), std::out_of_range); - CHECK_THROWS_AS(json::from_cbor(std::vector({0x1b, 0x00, 0x00})), std::out_of_range); - CHECK_THROWS_AS(json::from_cbor(std::vector({0x1b, 0x00, 0x00, 0x00})), std::out_of_range); - CHECK_THROWS_AS(json::from_cbor(std::vector({0x1b, 0x00, 0x00, 0x00, 0x00})), std::out_of_range); - CHECK_THROWS_AS(json::from_cbor(std::vector({0x1b, 0x00, 0x00, 0x00, 0x00, 0x00})), std::out_of_range); - CHECK_THROWS_AS(json::from_cbor(std::vector({0x1b, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00})), std::out_of_range); - CHECK_THROWS_AS(json::from_cbor(std::vector({0x1b, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00})), std::out_of_range); + CHECK_THROWS_AS(json::from_cbor(std::vector({0x18})), json::parse_error); + CHECK_THROWS_AS(json::from_cbor(std::vector({0x19})), json::parse_error); + CHECK_THROWS_AS(json::from_cbor(std::vector({0x19, 0x00})), json::parse_error); + CHECK_THROWS_AS(json::from_cbor(std::vector({0x1a})), json::parse_error); + CHECK_THROWS_AS(json::from_cbor(std::vector({0x1a, 0x00})), json::parse_error); + CHECK_THROWS_AS(json::from_cbor(std::vector({0x1a, 0x00, 0x00})), json::parse_error); + CHECK_THROWS_AS(json::from_cbor(std::vector({0x1a, 0x00, 0x00, 0x00})), json::parse_error); + CHECK_THROWS_AS(json::from_cbor(std::vector({0x1b})), json::parse_error); + CHECK_THROWS_AS(json::from_cbor(std::vector({0x1b, 0x00})), json::parse_error); + CHECK_THROWS_AS(json::from_cbor(std::vector({0x1b, 0x00, 0x00})), json::parse_error); + CHECK_THROWS_AS(json::from_cbor(std::vector({0x1b, 0x00, 0x00, 0x00})), json::parse_error); + CHECK_THROWS_AS(json::from_cbor(std::vector({0x1b, 0x00, 0x00, 0x00, 0x00})), json::parse_error); + CHECK_THROWS_AS(json::from_cbor(std::vector({0x1b, 0x00, 0x00, 0x00, 0x00, 0x00})), json::parse_error); + CHECK_THROWS_AS(json::from_cbor(std::vector({0x1b, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00})), json::parse_error); + CHECK_THROWS_AS(json::from_cbor(std::vector({0x1b, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00})), json::parse_error); + + CHECK_THROWS_WITH(json::from_cbor(std::vector({0x18})), + "[json.exception.parse_error.110] parse error at 1: cannot read 1 bytes from vector"); + CHECK_THROWS_WITH(json::from_cbor(std::vector({0x19})), + "[json.exception.parse_error.110] parse error at 1: cannot read 2 bytes from vector"); + CHECK_THROWS_WITH(json::from_cbor(std::vector({0x19, 0x00})), + "[json.exception.parse_error.110] parse error at 1: cannot read 2 bytes from vector"); + CHECK_THROWS_WITH(json::from_cbor(std::vector({0x1a})), + "[json.exception.parse_error.110] parse error at 1: cannot read 4 bytes from vector"); + CHECK_THROWS_WITH(json::from_cbor(std::vector({0x1a, 0x00})), + "[json.exception.parse_error.110] parse error at 1: cannot read 4 bytes from vector"); + CHECK_THROWS_WITH(json::from_cbor(std::vector({0x1a, 0x00, 0x00})), + "[json.exception.parse_error.110] parse error at 1: cannot read 4 bytes from vector"); + CHECK_THROWS_WITH(json::from_cbor(std::vector({0x1a, 0x00, 0x00, 0x00})), + "[json.exception.parse_error.110] parse error at 1: cannot read 4 bytes from vector"); + CHECK_THROWS_WITH(json::from_cbor(std::vector({0x1b})), + "[json.exception.parse_error.110] parse error at 1: cannot read 8 bytes from vector"); + CHECK_THROWS_WITH(json::from_cbor(std::vector({0x1b, 0x00})), + "[json.exception.parse_error.110] parse error at 1: cannot read 8 bytes from vector"); + CHECK_THROWS_WITH(json::from_cbor(std::vector({0x1b, 0x00, 0x00})), + "[json.exception.parse_error.110] parse error at 1: cannot read 8 bytes from vector"); + CHECK_THROWS_WITH(json::from_cbor(std::vector({0x1b, 0x00, 0x00, 0x00})), + "[json.exception.parse_error.110] parse error at 1: cannot read 8 bytes from vector"); + CHECK_THROWS_WITH(json::from_cbor(std::vector({0x1b, 0x00, 0x00, 0x00, 0x00})), + "[json.exception.parse_error.110] parse error at 1: cannot read 8 bytes from vector"); + CHECK_THROWS_WITH(json::from_cbor(std::vector({0x1b, 0x00, 0x00, 0x00, 0x00, 0x00})), + "[json.exception.parse_error.110] parse error at 1: cannot read 8 bytes from vector"); + CHECK_THROWS_WITH(json::from_cbor(std::vector({0x1b, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00})), + "[json.exception.parse_error.110] parse error at 1: cannot read 8 bytes from vector"); + CHECK_THROWS_WITH(json::from_cbor(std::vector({0x1b, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00})), + "[json.exception.parse_error.110] parse error at 1: cannot read 8 bytes from vector"); } } } From 54073332248e90de6b08ff7c5a3f1288205fc8f4 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Mon, 6 Mar 2017 22:37:46 +0100 Subject: [PATCH 047/530] :hammer: added user-defined exception 111 --- src/json.hpp | 4 ++-- src/json.hpp.re2c | 4 ++-- test/src/unit-regression.cpp | 3 ++- 3 files changed, 6 insertions(+), 5 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 6a36b85d..5fc66dc5 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -173,7 +173,7 @@ json.exception.[parse_error](@ref parse_error).107 | "parse error: JSON pointer json.exception.[parse_error](@ref parse_error).108 | "parse error: escape character '~' must be followed with '0' or '1'" | In a JSON Pointer, only `~0` and `~1` are valid escape sequences. json.exception.[parse_error](@ref parse_error).109 | "parse error: array index 'one' is not a number" | A JSON Pointer array index must be a number. json.exception.[parse_error](@ref parse_error).110 | "parse error at 1: cannot read 2 bytes from vector" | When parsing CBOR or MessagePack, the byte vector ends before the complete value has been read. - +json.exception.[parse_error](@ref parse_error).111 | "parse error: bad input stream" | Parsing CBOR or MessagePack from an input stream where the `badbit` or `failbit` is set. @since version 3.0.0 */ @@ -9889,7 +9889,7 @@ class basic_json // immediately abort if stream is erroneous if (s.fail()) { - JSON_THROW(std::invalid_argument("stream error")); + JSON_THROW(parse_error(111, 0, "bad input stream")); } // fill buffer diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 4ee6cced..d022375b 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -173,7 +173,7 @@ json.exception.[parse_error](@ref parse_error).107 | "parse error: JSON pointer json.exception.[parse_error](@ref parse_error).108 | "parse error: escape character '~' must be followed with '0' or '1'" | In a JSON Pointer, only `~0` and `~1` are valid escape sequences. json.exception.[parse_error](@ref parse_error).109 | "parse error: array index 'one' is not a number" | A JSON Pointer array index must be a number. json.exception.[parse_error](@ref parse_error).110 | "parse error at 1: cannot read 2 bytes from vector" | When parsing CBOR or MessagePack, the byte vector ends before the complete value has been read. - +json.exception.[parse_error](@ref parse_error).111 | "parse error: bad input stream" | Parsing CBOR or MessagePack from an input stream where the `badbit` or `failbit` is set. @since version 3.0.0 */ @@ -9889,7 +9889,7 @@ class basic_json // immediately abort if stream is erroneous if (s.fail()) { - JSON_THROW(std::invalid_argument("stream error")); + JSON_THROW(parse_error(111, 0, "bad input stream")); } // fill buffer diff --git a/test/src/unit-regression.cpp b/test/src/unit-regression.cpp index 9dea8083..2d799067 100644 --- a/test/src/unit-regression.cpp +++ b/test/src/unit-regression.cpp @@ -565,7 +565,8 @@ TEST_CASE("regression tests") SECTION("issue #366 - json::parse on failed stream gets stuck") { std::ifstream f("file_not_found.json"); - CHECK_THROWS_AS(json::parse(f), std::invalid_argument); + CHECK_THROWS_AS(json::parse(f), json::parse_error); + CHECK_THROWS_WITH(json::parse(f), "[json.exception.parse_error.111] parse error: bad input stream"); } SECTION("issue #367 - calling stream at EOF") From 757d2c6c7aa4af0d2df46f3f5bb8562592892226 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Mon, 6 Mar 2017 23:45:08 +0100 Subject: [PATCH 048/530] :hammer: replaced at() calls in msgpack/cbor --- src/json.hpp | 32 ++++++++++++++++++++------------ src/json.hpp.re2c | 32 ++++++++++++++++++++------------ 2 files changed, 40 insertions(+), 24 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 5fc66dc5..9a2c2c01 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -7957,12 +7957,12 @@ class basic_json */ static basic_json from_msgpack_internal(const std::vector& v, size_t& idx) { - // make sure reading 1 byte is safe - check_length(v.size(), 1, idx); - // store and increment index const size_t current_idx = idx++; + // make sure reading 1 byte is safe + check_length(v.size(), 1, current_idx); + if (v[current_idx] <= 0xbf) { if (v[current_idx] <= 0x7f) // positive fixint @@ -8026,9 +8026,10 @@ class basic_json { // copy bytes in reverse order into the double variable float res; + check_length(v.size(), sizeof(float), current_idx + 1); for (size_t byte = 0; byte < sizeof(float); ++byte) { - reinterpret_cast(&res)[sizeof(float) - byte - 1] = v.at(current_idx + 1 + byte); + reinterpret_cast(&res)[sizeof(float) - byte - 1] = v[current_idx + 1 + byte]; } idx += sizeof(float); // skip content bytes return res; @@ -8038,9 +8039,10 @@ class basic_json { // copy bytes in reverse order into the double variable double res; + check_length(v.size(), sizeof(double), current_idx + 1); for (size_t byte = 0; byte < sizeof(double); ++byte) { - reinterpret_cast(&res)[sizeof(double) - byte - 1] = v.at(current_idx + 1 + byte); + reinterpret_cast(&res)[sizeof(double) - byte - 1] = v[current_idx + 1 + byte]; } idx += sizeof(double); // skip content bytes return res; @@ -8198,7 +8200,10 @@ class basic_json // store and increment index const size_t current_idx = idx++; - switch (v.at(current_idx)) + // make sure reading 1 byte is safe + check_length(v.size(), 1, current_idx); + + switch (v[current_idx]) { // Integer 0x00..0x17 (0..23) case 0x00: @@ -8379,7 +8384,7 @@ class basic_json case 0x7f: // UTF-8 string (indefinite length) { std::string result; - while (v.at(idx) != 0xff) + while (check_length(v.size(), 1, idx), v[idx] != 0xff) { string_t s = from_cbor_internal(v, idx); result += s; @@ -8475,7 +8480,7 @@ class basic_json case 0x9f: // array (indefinite length) { basic_json result = value_t::array; - while (v.at(idx) != 0xff) + while (check_length(v.size(), 1, idx), v[idx] != 0xff) { result.push_back(from_cbor_internal(v, idx)); } @@ -8575,7 +8580,7 @@ class basic_json case 0xbf: // map (indefinite length) { basic_json result = value_t::object; - while (v.at(idx) != 0xff) + while (check_length(v.size(), 1, idx), v[idx] != 0xff) { std::string key = from_cbor_internal(v, idx); result[key] = from_cbor_internal(v, idx); @@ -8611,7 +8616,8 @@ class basic_json // include at least decoding support for them even without such // support. An example of a small decoder for half-precision // floating-point numbers in the C language is shown in Fig. 3. - const int half = (v.at(current_idx + 1) << 8) + v.at(current_idx + 2); + check_length(v.size(), 2, current_idx + 1); + const int half = (v[current_idx + 1] << 8) + v[current_idx + 2]; const int exp = (half >> 10) & 0x1f; const int mant = half & 0x3ff; double val; @@ -8636,9 +8642,10 @@ class basic_json { // copy bytes in reverse order into the float variable float res; + check_length(v.size(), sizeof(float), current_idx + 1); for (size_t byte = 0; byte < sizeof(float); ++byte) { - reinterpret_cast(&res)[sizeof(float) - byte - 1] = v.at(current_idx + 1 + byte); + reinterpret_cast(&res)[sizeof(float) - byte - 1] = v[current_idx + 1 + byte]; } idx += sizeof(float); // skip content bytes return res; @@ -8648,9 +8655,10 @@ class basic_json { // copy bytes in reverse order into the double variable double res; + check_length(v.size(), sizeof(double), current_idx + 1); for (size_t byte = 0; byte < sizeof(double); ++byte) { - reinterpret_cast(&res)[sizeof(double) - byte - 1] = v.at(current_idx + 1 + byte); + reinterpret_cast(&res)[sizeof(double) - byte - 1] = v[current_idx + 1 + byte]; } idx += sizeof(double); // skip content bytes return res; diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index d022375b..8d8f11b8 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -7957,12 +7957,12 @@ class basic_json */ static basic_json from_msgpack_internal(const std::vector& v, size_t& idx) { - // make sure reading 1 byte is safe - check_length(v.size(), 1, idx); - // store and increment index const size_t current_idx = idx++; + // make sure reading 1 byte is safe + check_length(v.size(), 1, current_idx); + if (v[current_idx] <= 0xbf) { if (v[current_idx] <= 0x7f) // positive fixint @@ -8026,9 +8026,10 @@ class basic_json { // copy bytes in reverse order into the double variable float res; + check_length(v.size(), sizeof(float), current_idx + 1); for (size_t byte = 0; byte < sizeof(float); ++byte) { - reinterpret_cast(&res)[sizeof(float) - byte - 1] = v.at(current_idx + 1 + byte); + reinterpret_cast(&res)[sizeof(float) - byte - 1] = v[current_idx + 1 + byte]; } idx += sizeof(float); // skip content bytes return res; @@ -8038,9 +8039,10 @@ class basic_json { // copy bytes in reverse order into the double variable double res; + check_length(v.size(), sizeof(double), current_idx + 1); for (size_t byte = 0; byte < sizeof(double); ++byte) { - reinterpret_cast(&res)[sizeof(double) - byte - 1] = v.at(current_idx + 1 + byte); + reinterpret_cast(&res)[sizeof(double) - byte - 1] = v[current_idx + 1 + byte]; } idx += sizeof(double); // skip content bytes return res; @@ -8198,7 +8200,10 @@ class basic_json // store and increment index const size_t current_idx = idx++; - switch (v.at(current_idx)) + // make sure reading 1 byte is safe + check_length(v.size(), 1, current_idx); + + switch (v[current_idx]) { // Integer 0x00..0x17 (0..23) case 0x00: @@ -8379,7 +8384,7 @@ class basic_json case 0x7f: // UTF-8 string (indefinite length) { std::string result; - while (v.at(idx) != 0xff) + while (check_length(v.size(), 1, idx), v[idx] != 0xff) { string_t s = from_cbor_internal(v, idx); result += s; @@ -8475,7 +8480,7 @@ class basic_json case 0x9f: // array (indefinite length) { basic_json result = value_t::array; - while (v.at(idx) != 0xff) + while (check_length(v.size(), 1, idx), v[idx] != 0xff) { result.push_back(from_cbor_internal(v, idx)); } @@ -8575,7 +8580,7 @@ class basic_json case 0xbf: // map (indefinite length) { basic_json result = value_t::object; - while (v.at(idx) != 0xff) + while (check_length(v.size(), 1, idx), v[idx] != 0xff) { std::string key = from_cbor_internal(v, idx); result[key] = from_cbor_internal(v, idx); @@ -8611,7 +8616,8 @@ class basic_json // include at least decoding support for them even without such // support. An example of a small decoder for half-precision // floating-point numbers in the C language is shown in Fig. 3. - const int half = (v.at(current_idx + 1) << 8) + v.at(current_idx + 2); + check_length(v.size(), 2, current_idx + 1); + const int half = (v[current_idx + 1] << 8) + v[current_idx + 2]; const int exp = (half >> 10) & 0x1f; const int mant = half & 0x3ff; double val; @@ -8636,9 +8642,10 @@ class basic_json { // copy bytes in reverse order into the float variable float res; + check_length(v.size(), sizeof(float), current_idx + 1); for (size_t byte = 0; byte < sizeof(float); ++byte) { - reinterpret_cast(&res)[sizeof(float) - byte - 1] = v.at(current_idx + 1 + byte); + reinterpret_cast(&res)[sizeof(float) - byte - 1] = v[current_idx + 1 + byte]; } idx += sizeof(float); // skip content bytes return res; @@ -8648,9 +8655,10 @@ class basic_json { // copy bytes in reverse order into the double variable double res; + check_length(v.size(), sizeof(double), current_idx + 1); for (size_t byte = 0; byte < sizeof(double); ++byte) { - reinterpret_cast(&res)[sizeof(double) - byte - 1] = v.at(current_idx + 1 + byte); + reinterpret_cast(&res)[sizeof(double) - byte - 1] = v[current_idx + 1 + byte]; } idx += sizeof(double); // skip content bytes return res; From 625cf7e3f7b227c48b235e6b34c1198a35c0614f Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 7 Mar 2017 20:05:34 +0100 Subject: [PATCH 049/530] :hammer: added user-defined exception 112 --- src/json.hpp | 9 ++++- src/json.hpp.re2c | 9 ++++- test/src/unit-cbor.cpp | 63 +++++++++++++++++++++++++++++ test/src/unit-msgpack.cpp | 84 +++++++++++++++++++++++++++++++++++++++ 4 files changed, 161 insertions(+), 4 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 9a2c2c01..68bb3cd7 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -174,6 +174,7 @@ json.exception.[parse_error](@ref parse_error).108 | "parse error: escape charac json.exception.[parse_error](@ref parse_error).109 | "parse error: array index 'one' is not a number" | A JSON Pointer array index must be a number. json.exception.[parse_error](@ref parse_error).110 | "parse error at 1: cannot read 2 bytes from vector" | When parsing CBOR or MessagePack, the byte vector ends before the complete value has been read. json.exception.[parse_error](@ref parse_error).111 | "parse error: bad input stream" | Parsing CBOR or MessagePack from an input stream where the `badbit` or `failbit` is set. +json.exception.[parse_error](@ref parse_error).112 | "parse error at 1: error reading CBOR; last byte: 0xf8" | Not all types of CBOR or MessagePack are supported. This exception occurs if an unsupported byte was read. @since version 3.0.0 */ @@ -8175,7 +8176,9 @@ class basic_json default: { - JSON_THROW(std::invalid_argument("error parsing a msgpack @ " + std::to_string(current_idx) + ": " + std::to_string(static_cast(v[current_idx])))); + std::stringstream ss; + ss << std::hex << static_cast(v[current_idx]); + JSON_THROW(parse_error(112, current_idx + 1, "error reading MessagePack; last byte: 0x" + ss.str())); } } } @@ -8666,7 +8669,9 @@ class basic_json default: // anything else (0xFF is handled inside the other types) { - JSON_THROW(std::invalid_argument("error parsing a CBOR @ " + std::to_string(current_idx) + ": " + std::to_string(static_cast(v[current_idx])))); + std::stringstream ss; + ss << std::hex << static_cast(v[current_idx]); + JSON_THROW(parse_error(112, current_idx + 1, "error reading CBOR; last byte: 0x" + ss.str())); } } } diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 8d8f11b8..60a1042c 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -174,6 +174,7 @@ json.exception.[parse_error](@ref parse_error).108 | "parse error: escape charac json.exception.[parse_error](@ref parse_error).109 | "parse error: array index 'one' is not a number" | A JSON Pointer array index must be a number. json.exception.[parse_error](@ref parse_error).110 | "parse error at 1: cannot read 2 bytes from vector" | When parsing CBOR or MessagePack, the byte vector ends before the complete value has been read. json.exception.[parse_error](@ref parse_error).111 | "parse error: bad input stream" | Parsing CBOR or MessagePack from an input stream where the `badbit` or `failbit` is set. +json.exception.[parse_error](@ref parse_error).112 | "parse error at 1: error reading CBOR; last byte: 0xf8" | Not all types of CBOR or MessagePack are supported. This exception occurs if an unsupported byte was read. @since version 3.0.0 */ @@ -8175,7 +8176,9 @@ class basic_json default: { - JSON_THROW(std::invalid_argument("error parsing a msgpack @ " + std::to_string(current_idx) + ": " + std::to_string(static_cast(v[current_idx])))); + std::stringstream ss; + ss << std::hex << static_cast(v[current_idx]); + JSON_THROW(parse_error(112, current_idx + 1, "error reading MessagePack; last byte: 0x" + ss.str())); } } } @@ -8666,7 +8669,9 @@ class basic_json default: // anything else (0xFF is handled inside the other types) { - JSON_THROW(std::invalid_argument("error parsing a CBOR @ " + std::to_string(current_idx) + ": " + std::to_string(static_cast(v[current_idx])))); + std::stringstream ss; + ss << std::hex << static_cast(v[current_idx]); + JSON_THROW(parse_error(112, current_idx + 1, "error reading CBOR; last byte: 0x" + ss.str())); } } } diff --git a/test/src/unit-cbor.cpp b/test/src/unit-cbor.cpp index 9525c722..5667196a 100644 --- a/test/src/unit-cbor.cpp +++ b/test/src/unit-cbor.cpp @@ -1192,6 +1192,69 @@ TEST_CASE("CBOR") CHECK_THROWS_WITH(json::from_cbor(std::vector({0x1b, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00})), "[json.exception.parse_error.110] parse error at 1: cannot read 8 bytes from vector"); } + + SECTION("unsupported bytes") + { + SECTION("concrete examples") + { + CHECK_THROWS_AS(json::from_cbor(std::vector({0x1c})), json::parse_error); + CHECK_THROWS_WITH(json::from_cbor(std::vector({0x1c})), + "[json.exception.parse_error.112] parse error at 1: error reading CBOR; last byte: 0x1c"); + CHECK_THROWS_AS(json::from_cbor(std::vector({0xf8})), json::parse_error); + CHECK_THROWS_WITH(json::from_cbor(std::vector({0xf8})), + "[json.exception.parse_error.112] parse error at 1: error reading CBOR; last byte: 0xf8"); + } + + SECTION("all unsupported bytes") + { + for (auto byte : + { + // ? + 0x1c, 0x1d, 0x1e, 0x1f, + // ? + 0x3c, 0x3d, 0x3e, 0x3f, + // byte strings + 0x40, 0x41, 0x42, 0x43, 0x44, 0x45, 0x46, 0x47, 0x48, 0x49, 0x50, 0x51, 0x52, 0x53, 0x54, 0x55, 0x56, 0x57, + // byte strings + 0x58, 0x59, 0x5a, 0x5b, + // ? + 0x5c, 0x5d, 0x5e, + // byte string + 0x5f, + // ? + 0x7c, 0x7d, 0x7e, + // ? + 0x9c, 0x9d, 0x9e, + // ? + 0xbc, 0xbd, 0xbe, + // date/time + 0xc0, 0xc1, + // bignum + 0xc2, 0xc3, + // fraction + 0xc4, + // bigfloat + 0xc5, + // tagged item + 0xc6, 0xc7, 0xc8, 0xc9, 0xca, 0xcb, 0xcc, 0xcd, 0xce, 0xcf, 0xd0, 0xd1, 0xd2, 0xd3, 0xd4, + // expected conversion + 0xd5, 0xd6, 0xd7, + // more tagged items + 0xd8, 0xd9, 0xda, 0xdb, + // ? + 0xdc, 0xdd, 0xde, 0xdf, + // (simple value) + 0xe0, 0xe1, 0xe2, 0xe3, 0xe4, 0xe5, 0xe6, 0xe7, 0xe8, 0xe9, 0xea, 0xeb, 0xec, 0xed, 0xee, 0xef, 0xf0, 0xf1, 0xf2, 0xf3, + // undefined + 0xf7, + // simple value + 0xf8 + }) + { + CHECK_THROWS_AS(json::from_cbor(std::vector({static_cast(byte)})), json::parse_error); + } + } + } } } diff --git a/test/src/unit-msgpack.cpp b/test/src/unit-msgpack.cpp index 3a2ae5b1..04428dff 100644 --- a/test/src/unit-msgpack.cpp +++ b/test/src/unit-msgpack.cpp @@ -1016,6 +1016,90 @@ TEST_CASE("MessagePack") json j = json::from_msgpack(given); CHECK(j.get() == Approx(25.0000019073486)); } + + SECTION("errors") + { + SECTION("too short byte vector") + { + CHECK_THROWS_AS(json::from_msgpack(std::vector({0xcc})), json::parse_error); + CHECK_THROWS_AS(json::from_msgpack(std::vector({0xcd})), json::parse_error); + CHECK_THROWS_AS(json::from_msgpack(std::vector({0xcd, 0x00})), json::parse_error); + CHECK_THROWS_AS(json::from_msgpack(std::vector({0xce})), json::parse_error); + CHECK_THROWS_AS(json::from_msgpack(std::vector({0xce, 0x00})), json::parse_error); + CHECK_THROWS_AS(json::from_msgpack(std::vector({0xce, 0x00, 0x00})), json::parse_error); + CHECK_THROWS_AS(json::from_msgpack(std::vector({0xce, 0x00, 0x00, 0x00})), json::parse_error); + CHECK_THROWS_AS(json::from_msgpack(std::vector({0xcf})), json::parse_error); + CHECK_THROWS_AS(json::from_msgpack(std::vector({0xcf, 0x00})), json::parse_error); + CHECK_THROWS_AS(json::from_msgpack(std::vector({0xcf, 0x00, 0x00})), json::parse_error); + CHECK_THROWS_AS(json::from_msgpack(std::vector({0xcf, 0x00, 0x00, 0x00})), json::parse_error); + CHECK_THROWS_AS(json::from_msgpack(std::vector({0xcf, 0x00, 0x00, 0x00, 0x00})), json::parse_error); + CHECK_THROWS_AS(json::from_msgpack(std::vector({0xcf, 0x00, 0x00, 0x00, 0x00, 0x00})), json::parse_error); + CHECK_THROWS_AS(json::from_msgpack(std::vector({0xcf, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00})), json::parse_error); + CHECK_THROWS_AS(json::from_msgpack(std::vector({0xcf, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00})), json::parse_error); + + CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xcc})), + "[json.exception.parse_error.110] parse error at 1: cannot read 1 bytes from vector"); + CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xcd})), + "[json.exception.parse_error.110] parse error at 1: cannot read 2 bytes from vector"); + CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xcd, 0x00})), + "[json.exception.parse_error.110] parse error at 1: cannot read 2 bytes from vector"); + CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xce})), + "[json.exception.parse_error.110] parse error at 1: cannot read 4 bytes from vector"); + CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xce, 0x00})), + "[json.exception.parse_error.110] parse error at 1: cannot read 4 bytes from vector"); + CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xce, 0x00, 0x00})), + "[json.exception.parse_error.110] parse error at 1: cannot read 4 bytes from vector"); + CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xce, 0x00, 0x00, 0x00})), + "[json.exception.parse_error.110] parse error at 1: cannot read 4 bytes from vector"); + CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xcf})), + "[json.exception.parse_error.110] parse error at 1: cannot read 8 bytes from vector"); + CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xcf, 0x00})), + "[json.exception.parse_error.110] parse error at 1: cannot read 8 bytes from vector"); + CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xcf, 0x00, 0x00})), + "[json.exception.parse_error.110] parse error at 1: cannot read 8 bytes from vector"); + CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xcf, 0x00, 0x00, 0x00})), + "[json.exception.parse_error.110] parse error at 1: cannot read 8 bytes from vector"); + CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xcf, 0x00, 0x00, 0x00, 0x00})), + "[json.exception.parse_error.110] parse error at 1: cannot read 8 bytes from vector"); + CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xcf, 0x00, 0x00, 0x00, 0x00, 0x00})), + "[json.exception.parse_error.110] parse error at 1: cannot read 8 bytes from vector"); + CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xcf, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00})), + "[json.exception.parse_error.110] parse error at 1: cannot read 8 bytes from vector"); + CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xcf, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00})), + "[json.exception.parse_error.110] parse error at 1: cannot read 8 bytes from vector"); + } + + SECTION("unsupported bytes") + { + SECTION("concrete examples") + { + CHECK_THROWS_AS(json::from_msgpack(std::vector({0xc1})), json::parse_error); + CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xc1})), + "[json.exception.parse_error.112] parse error at 1: error reading MessagePack; last byte: 0xc1"); + CHECK_THROWS_AS(json::from_msgpack(std::vector({0xc6})), json::parse_error); + CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xc6})), + "[json.exception.parse_error.112] parse error at 1: error reading MessagePack; last byte: 0xc6"); + } + + SECTION("all unsupported bytes") + { + for (auto byte : + { + // never used + 0xc1, + // bin + 0xc4, 0xc5, 0xc6, + // ext + 0xc7, 0xc8, 0xc9, + // fixext + 0xd4, 0xd5, 0xd6, 0xd7, 0xd8 + }) + { + CHECK_THROWS_AS(json::from_msgpack(std::vector({static_cast(byte)})), json::parse_error); + } + } + } + } } From 8fcd01631f9383c6a8a2ff6a6286e78b2a96b5f4 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Wed, 8 Mar 2017 16:39:17 +0100 Subject: [PATCH 050/530] :white_check_mark: improved test coverage Tests for parse_error.109 were missing. --- test/src/unit-json_pointer.cpp | 14 ++++++++++++++ 1 file changed, 14 insertions(+) diff --git a/test/src/unit-json_pointer.cpp b/test/src/unit-json_pointer.cpp index a6f811eb..d14261a0 100644 --- a/test/src/unit-json_pointer.cpp +++ b/test/src/unit-json_pointer.cpp @@ -272,6 +272,20 @@ TEST_CASE("JSON pointers") CHECK_THROWS_AS(j["/one"_json_pointer] = 1, json::parse_error); CHECK_THROWS_WITH(j["/one"_json_pointer] = 1, "[json.exception.parse_error.109] parse error: array index 'one' is not a number"); + CHECK_THROWS_AS(j_const["/one"_json_pointer] == 1, json::parse_error); + CHECK_THROWS_WITH(j_const["/one"_json_pointer] == 1, + "[json.exception.parse_error.109] parse error: array index 'one' is not a number"); + + CHECK_THROWS_AS(j.at("/one"_json_pointer) = 1, json::parse_error); + CHECK_THROWS_WITH(j.at("/one"_json_pointer) = 1, + "[json.exception.parse_error.109] parse error: array index 'one' is not a number"); + CHECK_THROWS_AS(j_const.at("/one"_json_pointer) == 1, json::parse_error); + CHECK_THROWS_WITH(j_const.at("/one"_json_pointer) == 1, + "[json.exception.parse_error.109] parse error: array index 'one' is not a number"); + + CHECK_THROWS_AS(json({{"/list/0", 1}, {"/list/1", 2}, {"/list/three", 3}}).unflatten(), json::parse_error); + CHECK_THROWS_WITH(json({{"/list/0", 1}, {"/list/1", 2}, {"/list/three", 3}}).unflatten(), + "[json.exception.parse_error.109] parse error: array index 'three' is not a number"); // assign to "-" j["/-"_json_pointer] = 99; From fc9b528ec9e48a1e02c861b527fe508aec5a4107 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Wed, 8 Mar 2017 18:07:21 +0100 Subject: [PATCH 051/530] :hammer: changed an exception --- src/json.hpp | 15 +++---- src/json.hpp.re2c | 12 +++--- test/src/fuzzer-parse_json.cpp | 4 +- test/src/unit-cbor.cpp | 46 +++++++++----------- test/src/unit-msgpack.cpp | 30 +++++++------- test/src/unit-regression.cpp | 76 +++++++++++++++++++++++++--------- 6 files changed, 106 insertions(+), 77 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 68bb3cd7..ffe64d1a 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -7374,10 +7374,11 @@ class basic_json template static T get_from_vector(const std::vector& vec, const size_t current_index) { - if (current_index + sizeof(T) + 1 > vec.size()) - { - JSON_THROW(parse_error(110, current_index + 1, "cannot read " + std::to_string(sizeof(T)) + " bytes from vector")); - } + check_length(vec.size(), sizeof(T), current_index + 1); + //if (current_index + sizeof(T) + 1 > vec.size()) + //{ + // JSON_THROW(parse_error(110, current_index + 1, "cannot read " + std::to_string(sizeof(T)) + " bytes from vector")); + //} T result; auto* ptr = reinterpret_cast(&result); @@ -7926,19 +7927,19 @@ class basic_json // simple case: requested length is greater than the vector's length if (len > size or offset > size) { - JSON_THROW(std::out_of_range("len out of range")); + JSON_THROW(parse_error(110, offset + 1, "cannot read " + std::to_string(len) + " bytes from vector")); } // second case: adding offset would result in overflow if ((size > (std::numeric_limits::max() - offset))) { - JSON_THROW(std::out_of_range("len+offset out of range")); + JSON_THROW(parse_error(110, offset + 1, "cannot read " + std::to_string(len) + " bytes from vector")); } // last case: reading past the end of the vector if (len + offset > size) { - JSON_THROW(std::out_of_range("len+offset out of range")); + JSON_THROW(parse_error(110, offset + 1, "cannot read " + std::to_string(len) + " bytes from vector")); } } diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 60a1042c..300a6219 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -7374,10 +7374,8 @@ class basic_json template static T get_from_vector(const std::vector& vec, const size_t current_index) { - if (current_index + sizeof(T) + 1 > vec.size()) - { - JSON_THROW(parse_error(110, current_index + 1, "cannot read " + std::to_string(sizeof(T)) + " bytes from vector")); - } + // check if we can read sizeof(T) bytes starting the next index + check_length(vec.size(), sizeof(T), current_index + 1); T result; auto* ptr = reinterpret_cast(&result); @@ -7926,19 +7924,19 @@ class basic_json // simple case: requested length is greater than the vector's length if (len > size or offset > size) { - JSON_THROW(std::out_of_range("len out of range")); + JSON_THROW(parse_error(110, offset + 1, "cannot read " + std::to_string(len) + " bytes from vector")); } // second case: adding offset would result in overflow if ((size > (std::numeric_limits::max() - offset))) { - JSON_THROW(std::out_of_range("len+offset out of range")); + JSON_THROW(parse_error(110, offset + 1, "cannot read " + std::to_string(len) + " bytes from vector")); } // last case: reading past the end of the vector if (len + offset > size) { - JSON_THROW(std::out_of_range("len+offset out of range")); + JSON_THROW(parse_error(110, offset + 1, "cannot read " + std::to_string(len) + " bytes from vector")); } } diff --git a/test/src/fuzzer-parse_json.cpp b/test/src/fuzzer-parse_json.cpp index bd2e5e39..ff585066 100644 --- a/test/src/fuzzer-parse_json.cpp +++ b/test/src/fuzzer-parse_json.cpp @@ -49,13 +49,13 @@ extern "C" int LLVMFuzzerTestOneInput(const uint8_t* data, size_t size) // serializations must match assert(s1 == s2); } - catch (const std::invalid_argument&) + catch (const json::parse_error&) { // parsing a JSON serialization must not fail assert(false); } } - catch (const std::invalid_argument&) + catch (const json::parse_error&) { // parse errors are ok, because input may be random bytes } diff --git a/test/src/unit-cbor.cpp b/test/src/unit-cbor.cpp index 5667196a..32464ae7 100644 --- a/test/src/unit-cbor.cpp +++ b/test/src/unit-cbor.cpp @@ -1162,35 +1162,35 @@ TEST_CASE("CBOR") CHECK_THROWS_AS(json::from_cbor(std::vector({0x1b, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00})), json::parse_error); CHECK_THROWS_WITH(json::from_cbor(std::vector({0x18})), - "[json.exception.parse_error.110] parse error at 1: cannot read 1 bytes from vector"); + "[json.exception.parse_error.110] parse error at 2: cannot read 1 bytes from vector"); CHECK_THROWS_WITH(json::from_cbor(std::vector({0x19})), - "[json.exception.parse_error.110] parse error at 1: cannot read 2 bytes from vector"); + "[json.exception.parse_error.110] parse error at 2: cannot read 2 bytes from vector"); CHECK_THROWS_WITH(json::from_cbor(std::vector({0x19, 0x00})), - "[json.exception.parse_error.110] parse error at 1: cannot read 2 bytes from vector"); + "[json.exception.parse_error.110] parse error at 2: cannot read 2 bytes from vector"); CHECK_THROWS_WITH(json::from_cbor(std::vector({0x1a})), - "[json.exception.parse_error.110] parse error at 1: cannot read 4 bytes from vector"); + "[json.exception.parse_error.110] parse error at 2: cannot read 4 bytes from vector"); CHECK_THROWS_WITH(json::from_cbor(std::vector({0x1a, 0x00})), - "[json.exception.parse_error.110] parse error at 1: cannot read 4 bytes from vector"); + "[json.exception.parse_error.110] parse error at 2: cannot read 4 bytes from vector"); CHECK_THROWS_WITH(json::from_cbor(std::vector({0x1a, 0x00, 0x00})), - "[json.exception.parse_error.110] parse error at 1: cannot read 4 bytes from vector"); + "[json.exception.parse_error.110] parse error at 2: cannot read 4 bytes from vector"); CHECK_THROWS_WITH(json::from_cbor(std::vector({0x1a, 0x00, 0x00, 0x00})), - "[json.exception.parse_error.110] parse error at 1: cannot read 4 bytes from vector"); + "[json.exception.parse_error.110] parse error at 2: cannot read 4 bytes from vector"); CHECK_THROWS_WITH(json::from_cbor(std::vector({0x1b})), - "[json.exception.parse_error.110] parse error at 1: cannot read 8 bytes from vector"); + "[json.exception.parse_error.110] parse error at 2: cannot read 8 bytes from vector"); CHECK_THROWS_WITH(json::from_cbor(std::vector({0x1b, 0x00})), - "[json.exception.parse_error.110] parse error at 1: cannot read 8 bytes from vector"); + "[json.exception.parse_error.110] parse error at 2: cannot read 8 bytes from vector"); CHECK_THROWS_WITH(json::from_cbor(std::vector({0x1b, 0x00, 0x00})), - "[json.exception.parse_error.110] parse error at 1: cannot read 8 bytes from vector"); + "[json.exception.parse_error.110] parse error at 2: cannot read 8 bytes from vector"); CHECK_THROWS_WITH(json::from_cbor(std::vector({0x1b, 0x00, 0x00, 0x00})), - "[json.exception.parse_error.110] parse error at 1: cannot read 8 bytes from vector"); + "[json.exception.parse_error.110] parse error at 2: cannot read 8 bytes from vector"); CHECK_THROWS_WITH(json::from_cbor(std::vector({0x1b, 0x00, 0x00, 0x00, 0x00})), - "[json.exception.parse_error.110] parse error at 1: cannot read 8 bytes from vector"); + "[json.exception.parse_error.110] parse error at 2: cannot read 8 bytes from vector"); CHECK_THROWS_WITH(json::from_cbor(std::vector({0x1b, 0x00, 0x00, 0x00, 0x00, 0x00})), - "[json.exception.parse_error.110] parse error at 1: cannot read 8 bytes from vector"); + "[json.exception.parse_error.110] parse error at 2: cannot read 8 bytes from vector"); CHECK_THROWS_WITH(json::from_cbor(std::vector({0x1b, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00})), - "[json.exception.parse_error.110] parse error at 1: cannot read 8 bytes from vector"); + "[json.exception.parse_error.110] parse error at 2: cannot read 8 bytes from vector"); CHECK_THROWS_WITH(json::from_cbor(std::vector({0x1b, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00})), - "[json.exception.parse_error.110] parse error at 1: cannot read 8 bytes from vector"); + "[json.exception.parse_error.110] parse error at 2: cannot read 8 bytes from vector"); } SECTION("unsupported bytes") @@ -1341,21 +1341,13 @@ TEST_CASE("CBOR regressions", "[!throws]") // deserializations must match CHECK(j1 == j2); } - catch (const std::invalid_argument&) + catch (const json::parse_error&) { // parsing a CBOR serialization must not fail CHECK(false); } } - catch (const std::invalid_argument&) - { - // parse errors are ok, because input may be random bytes - } - catch (const std::out_of_range&) - { - // parse errors are ok, because input may be random bytes - } - catch (const std::domain_error&) + catch (const json::parse_error&) { // parse errors are ok, because input may be random bytes } @@ -1365,7 +1357,9 @@ TEST_CASE("CBOR regressions", "[!throws]") SECTION("improve code coverage") { // exotic edge case - CHECK_THROWS_AS(json::check_length(0xffffffffffffffffull, 0xfffffffffffffff0ull, 0xff), std::out_of_range); + CHECK_THROWS_AS(json::check_length(0xffffffffffffffffull, 0xfffffffffffffff0ull, 0xff), json::parse_error); + CHECK_THROWS_WITH(json::check_length(0xffffffffffffffffull, 0xfffffffffffffff0ull, 0xff), + "[json.exception.parse_error.110] parse error at 256: cannot read 18446744073709551600 bytes from vector"); } } diff --git a/test/src/unit-msgpack.cpp b/test/src/unit-msgpack.cpp index 04428dff..de1a7bca 100644 --- a/test/src/unit-msgpack.cpp +++ b/test/src/unit-msgpack.cpp @@ -1038,35 +1038,35 @@ TEST_CASE("MessagePack") CHECK_THROWS_AS(json::from_msgpack(std::vector({0xcf, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00})), json::parse_error); CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xcc})), - "[json.exception.parse_error.110] parse error at 1: cannot read 1 bytes from vector"); + "[json.exception.parse_error.110] parse error at 2: cannot read 1 bytes from vector"); CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xcd})), - "[json.exception.parse_error.110] parse error at 1: cannot read 2 bytes from vector"); + "[json.exception.parse_error.110] parse error at 2: cannot read 2 bytes from vector"); CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xcd, 0x00})), - "[json.exception.parse_error.110] parse error at 1: cannot read 2 bytes from vector"); + "[json.exception.parse_error.110] parse error at 2: cannot read 2 bytes from vector"); CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xce})), - "[json.exception.parse_error.110] parse error at 1: cannot read 4 bytes from vector"); + "[json.exception.parse_error.110] parse error at 2: cannot read 4 bytes from vector"); CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xce, 0x00})), - "[json.exception.parse_error.110] parse error at 1: cannot read 4 bytes from vector"); + "[json.exception.parse_error.110] parse error at 2: cannot read 4 bytes from vector"); CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xce, 0x00, 0x00})), - "[json.exception.parse_error.110] parse error at 1: cannot read 4 bytes from vector"); + "[json.exception.parse_error.110] parse error at 2: cannot read 4 bytes from vector"); CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xce, 0x00, 0x00, 0x00})), - "[json.exception.parse_error.110] parse error at 1: cannot read 4 bytes from vector"); + "[json.exception.parse_error.110] parse error at 2: cannot read 4 bytes from vector"); CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xcf})), - "[json.exception.parse_error.110] parse error at 1: cannot read 8 bytes from vector"); + "[json.exception.parse_error.110] parse error at 2: cannot read 8 bytes from vector"); CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xcf, 0x00})), - "[json.exception.parse_error.110] parse error at 1: cannot read 8 bytes from vector"); + "[json.exception.parse_error.110] parse error at 2: cannot read 8 bytes from vector"); CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xcf, 0x00, 0x00})), - "[json.exception.parse_error.110] parse error at 1: cannot read 8 bytes from vector"); + "[json.exception.parse_error.110] parse error at 2: cannot read 8 bytes from vector"); CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xcf, 0x00, 0x00, 0x00})), - "[json.exception.parse_error.110] parse error at 1: cannot read 8 bytes from vector"); + "[json.exception.parse_error.110] parse error at 2: cannot read 8 bytes from vector"); CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xcf, 0x00, 0x00, 0x00, 0x00})), - "[json.exception.parse_error.110] parse error at 1: cannot read 8 bytes from vector"); + "[json.exception.parse_error.110] parse error at 2: cannot read 8 bytes from vector"); CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xcf, 0x00, 0x00, 0x00, 0x00, 0x00})), - "[json.exception.parse_error.110] parse error at 1: cannot read 8 bytes from vector"); + "[json.exception.parse_error.110] parse error at 2: cannot read 8 bytes from vector"); CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xcf, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00})), - "[json.exception.parse_error.110] parse error at 1: cannot read 8 bytes from vector"); + "[json.exception.parse_error.110] parse error at 2: cannot read 8 bytes from vector"); CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xcf, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00})), - "[json.exception.parse_error.110] parse error at 1: cannot read 8 bytes from vector"); + "[json.exception.parse_error.110] parse error at 2: cannot read 8 bytes from vector"); } SECTION("unsupported bytes") diff --git a/test/src/unit-regression.cpp b/test/src/unit-regression.cpp index 2d799067..324ab769 100644 --- a/test/src/unit-regression.cpp +++ b/test/src/unit-regression.cpp @@ -613,37 +613,51 @@ TEST_CASE("regression tests") { // original test case std::vector vec {0x65, 0xf5, 0x0a, 0x48, 0x21}; - CHECK_THROWS_AS(json::from_cbor(vec), std::out_of_range); + CHECK_THROWS_AS(json::from_cbor(vec), json::parse_error); + CHECK_THROWS_WITH(json::from_cbor(vec), + "[json.exception.parse_error.110] parse error at 2: cannot read 5 bytes from vector"); } SECTION("issue #407 - Heap-buffer-overflow (OSS-Fuzz issue 343)") { // original test case: incomplete float64 std::vector vec1 {0xcb, 0x8f, 0x0a}; - CHECK_THROWS_AS(json::from_msgpack(vec1), std::out_of_range); + CHECK_THROWS_AS(json::from_msgpack(vec1), json::parse_error); + CHECK_THROWS_WITH(json::from_msgpack(vec1), + "[json.exception.parse_error.110] parse error at 2: cannot read 8 bytes from vector"); // related test case: incomplete float32 std::vector vec2 {0xca, 0x8f, 0x0a}; - CHECK_THROWS_AS(json::from_msgpack(vec2), std::out_of_range); + CHECK_THROWS_AS(json::from_msgpack(vec2), json::parse_error); + CHECK_THROWS_WITH(json::from_msgpack(vec2), + "[json.exception.parse_error.110] parse error at 2: cannot read 4 bytes from vector"); // related test case: incomplete Half-Precision Float (CBOR) std::vector vec3 {0xf9, 0x8f}; - CHECK_THROWS_AS(json::from_cbor(vec3), std::out_of_range); + CHECK_THROWS_AS(json::from_cbor(vec3), json::parse_error); + CHECK_THROWS_WITH(json::from_cbor(vec3), + "[json.exception.parse_error.110] parse error at 2: cannot read 2 bytes from vector"); // related test case: incomplete Single-Precision Float (CBOR) std::vector vec4 {0xfa, 0x8f, 0x0a}; - CHECK_THROWS_AS(json::from_cbor(vec4), std::out_of_range); + CHECK_THROWS_AS(json::from_cbor(vec4), json::parse_error); + CHECK_THROWS_WITH(json::from_cbor(vec4), + "[json.exception.parse_error.110] parse error at 2: cannot read 4 bytes from vector"); // related test case: incomplete Double-Precision Float (CBOR) std::vector vec5 {0xfb, 0x8f, 0x0a}; - CHECK_THROWS_AS(json::from_cbor(vec5), std::out_of_range); + CHECK_THROWS_AS(json::from_cbor(vec5), json::parse_error); + CHECK_THROWS_WITH(json::from_cbor(vec5), + "[json.exception.parse_error.110] parse error at 2: cannot read 8 bytes from vector"); } SECTION("issue #408 - Heap-buffer-overflow (OSS-Fuzz issue 344)") { // original test case std::vector vec1 {0x87}; - CHECK_THROWS_AS(json::from_msgpack(vec1), std::out_of_range); + CHECK_THROWS_AS(json::from_msgpack(vec1), json::parse_error); + CHECK_THROWS_WITH(json::from_msgpack(vec1), + "[json.exception.parse_error.110] parse error at 2: cannot read 1 bytes from vector"); // more test cases for MessagePack for (auto b : @@ -655,7 +669,7 @@ TEST_CASE("regression tests") }) { std::vector vec(1, static_cast(b)); - CHECK_THROWS_AS(json::from_msgpack(vec), std::out_of_range); + CHECK_THROWS_AS(json::from_msgpack(vec), json::parse_error); } // more test cases for CBOR @@ -670,28 +684,38 @@ TEST_CASE("regression tests") }) { std::vector vec(1, static_cast(b)); - CHECK_THROWS_AS(json::from_cbor(vec), std::out_of_range); + CHECK_THROWS_AS(json::from_cbor(vec), json::parse_error); } // special case: empty input std::vector vec2; - CHECK_THROWS_AS(json::from_cbor(vec2), std::out_of_range); - CHECK_THROWS_AS(json::from_msgpack(vec2), std::out_of_range); + CHECK_THROWS_AS(json::from_cbor(vec2), json::parse_error); + CHECK_THROWS_WITH(json::from_cbor(vec2), + "[json.exception.parse_error.110] parse error at 1: cannot read 1 bytes from vector"); + CHECK_THROWS_AS(json::from_msgpack(vec2), json::parse_error); + CHECK_THROWS_WITH(json::from_msgpack(vec2), + "[json.exception.parse_error.110] parse error at 1: cannot read 1 bytes from vector"); } SECTION("issue #411 - Heap-buffer-overflow (OSS-Fuzz issue 366)") { // original test case: empty UTF-8 string (indefinite length) std::vector vec1 {0x7f}; - CHECK_THROWS_AS(json::from_cbor(vec1), std::out_of_range); + CHECK_THROWS_AS(json::from_cbor(vec1), json::parse_error); + CHECK_THROWS_WITH(json::from_cbor(vec1), + "[json.exception.parse_error.110] parse error at 2: cannot read 1 bytes from vector"); // related test case: empty array (indefinite length) std::vector vec2 {0x9f}; - CHECK_THROWS_AS(json::from_cbor(vec2), std::out_of_range); + CHECK_THROWS_AS(json::from_cbor(vec2), json::parse_error); + CHECK_THROWS_WITH(json::from_cbor(vec2), + "[json.exception.parse_error.110] parse error at 2: cannot read 1 bytes from vector"); // related test case: empty map (indefinite length) std::vector vec3 {0xbf}; - CHECK_THROWS_AS(json::from_cbor(vec3), std::out_of_range); + CHECK_THROWS_AS(json::from_cbor(vec3), json::parse_error); + CHECK_THROWS_WITH(json::from_cbor(vec3), + "[json.exception.parse_error.110] parse error at 2: cannot read 1 bytes from vector"); } SECTION("issue #412 - Heap-buffer-overflow (OSS-Fuzz issue 367)") @@ -717,19 +741,27 @@ TEST_CASE("regression tests") 0x60, 0x60, 0x60, 0x60, 0x60, 0x60, 0x60, 0x60, 0x60, 0x60, 0x60, 0x60, 0x60, 0x60, 0x60, 0x60 }; - CHECK_THROWS_AS(json::from_cbor(vec), std::out_of_range); + CHECK_THROWS_AS(json::from_cbor(vec), json::parse_error); + CHECK_THROWS_WITH(json::from_cbor(vec), + "[json.exception.parse_error.110] parse error at 137: cannot read 1 bytes from vector"); // related test case: nonempty UTF-8 string (indefinite length) std::vector vec1 {0x7f, 0x61, 0x61}; - CHECK_THROWS_AS(json::from_cbor(vec1), std::out_of_range); + CHECK_THROWS_AS(json::from_cbor(vec1), json::parse_error); + CHECK_THROWS_WITH(json::from_cbor(vec1), + "[json.exception.parse_error.110] parse error at 4: cannot read 1 bytes from vector"); // related test case: nonempty array (indefinite length) std::vector vec2 {0x9f, 0x01}; - CHECK_THROWS_AS(json::from_cbor(vec2), std::out_of_range); + CHECK_THROWS_AS(json::from_cbor(vec2), json::parse_error); + CHECK_THROWS_WITH(json::from_cbor(vec2), + "[json.exception.parse_error.110] parse error at 3: cannot read 1 bytes from vector"); // related test case: nonempty map (indefinite length) std::vector vec3 {0xbf, 0x61, 0x61, 0x01}; - CHECK_THROWS_AS(json::from_cbor(vec3), std::out_of_range); + CHECK_THROWS_AS(json::from_cbor(vec3), json::parse_error); + CHECK_THROWS_WITH(json::from_cbor(vec3), + "[json.exception.parse_error.110] parse error at 5: cannot read 1 bytes from vector"); } SECTION("issue #414 - compare with literal 0)") @@ -762,7 +794,9 @@ TEST_CASE("regression tests") 0x96, 0x96, 0xb4, 0xb4, 0xfa, 0x94, 0x94, 0x61, 0x61, 0x61, 0x61, 0x61, 0x61, 0x61, 0x61, 0xfa }; - CHECK_THROWS_AS(json::from_cbor(vec1), std::out_of_range); + CHECK_THROWS_AS(json::from_cbor(vec1), json::parse_error); + CHECK_THROWS_WITH(json::from_cbor(vec1), + "[json.exception.parse_error.110] parse error at 49: cannot read 4 bytes from vector"); // related test case: double-precision std::vector vec2 @@ -774,7 +808,9 @@ TEST_CASE("regression tests") 0x96, 0x96, 0xb4, 0xb4, 0xfa, 0x94, 0x94, 0x61, 0x61, 0x61, 0x61, 0x61, 0x61, 0x61, 0x61, 0xfb }; - CHECK_THROWS_AS(json::from_cbor(vec2), std::out_of_range); + CHECK_THROWS_AS(json::from_cbor(vec2), json::parse_error); + CHECK_THROWS_WITH(json::from_cbor(vec2), + "[json.exception.parse_error.110] parse error at 49: cannot read 8 bytes from vector"); } SECTION("issue #452 - Heap-buffer-overflow (OSS-Fuzz issue 585)") From c8a6ce79ea52e616a32d1c96d1682ddf2545e213 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Wed, 8 Mar 2017 18:37:03 +0100 Subject: [PATCH 052/530] :ambulance: fixing fuzzers to work with new exceptions --- test/src/fuzzer-parse_cbor.cpp | 12 ++++-------- test/src/fuzzer-parse_msgpack.cpp | 12 ++++-------- test/src/unit-cbor.cpp | 2 -- 3 files changed, 8 insertions(+), 18 deletions(-) diff --git a/test/src/fuzzer-parse_cbor.cpp b/test/src/fuzzer-parse_cbor.cpp index 30fa6977..6e7ae636 100644 --- a/test/src/fuzzer-parse_cbor.cpp +++ b/test/src/fuzzer-parse_cbor.cpp @@ -44,23 +44,19 @@ extern "C" int LLVMFuzzerTestOneInput(const uint8_t* data, size_t size) // deserializations must match assert(j1 == j2); } - catch (const std::invalid_argument&) + catch (const json::parse_error&) { // parsing a CBOR serialization must not fail assert(false); } } - catch (const std::invalid_argument&) + catch (const json::parse_error&) { // parse errors are ok, because input may be random bytes } - catch (const std::out_of_range&) + catch (const json::type_error&) { - // parse errors are ok, because input may be random bytes - } - catch (const std::domain_error&) - { - // parse errors are ok, because input may be random bytes + // type errors can occur during parsing, too } // return 0 - non-zero return values are reserved for future use diff --git a/test/src/fuzzer-parse_msgpack.cpp b/test/src/fuzzer-parse_msgpack.cpp index bf2fcab0..420283a4 100644 --- a/test/src/fuzzer-parse_msgpack.cpp +++ b/test/src/fuzzer-parse_msgpack.cpp @@ -44,23 +44,19 @@ extern "C" int LLVMFuzzerTestOneInput(const uint8_t* data, size_t size) // deserializations must match assert(j1 == j2); } - catch (const std::invalid_argument&) + catch (const json::parse_error&) { // parsing a MessagePack serialization must not fail assert(false); } } - catch (const std::invalid_argument&) + catch (const json::parse_error&) { // parse errors are ok, because input may be random bytes } - catch (const std::out_of_range&) + catch (const json::type_error&) { - // parse errors are ok, because input may be random bytes - } - catch (const std::domain_error&) - { - // parse errors are ok, because input may be random bytes + // type errors can occur during parsing, too } // return 0 - non-zero return values are reserved for future use diff --git a/test/src/unit-cbor.cpp b/test/src/unit-cbor.cpp index 32464ae7..87cbadc2 100644 --- a/test/src/unit-cbor.cpp +++ b/test/src/unit-cbor.cpp @@ -1358,8 +1358,6 @@ TEST_CASE("CBOR regressions", "[!throws]") { // exotic edge case CHECK_THROWS_AS(json::check_length(0xffffffffffffffffull, 0xfffffffffffffff0ull, 0xff), json::parse_error); - CHECK_THROWS_WITH(json::check_length(0xffffffffffffffffull, 0xfffffffffffffff0ull, 0xff), - "[json.exception.parse_error.110] parse error at 256: cannot read 18446744073709551600 bytes from vector"); } } From 9374eaa013fd5f080d4c70afde83311d493394e4 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Wed, 8 Mar 2017 19:16:53 +0100 Subject: [PATCH 053/530] :hammer: added user-defined exception 501 --- src/json.hpp | 38 ++++++++++++++++++++++++------------ src/json.hpp.re2c | 34 +++++++++++++++++++++++--------- test/src/unit-json_patch.cpp | 12 ++++++------ 3 files changed, 56 insertions(+), 28 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index ffe64d1a..f1462db8 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -52,7 +52,6 @@ SOFTWARE. #include // addressof, allocator, allocator_traits, unique_ptr #include // accumulate #include // stringstream -#include // domain_error, invalid_argument, out_of_range #include // getline, stoi, string, to_string #include // add_pointer, conditional, decay, enable_if, false_type, integral_constant, is_arithmetic, is_base_of, is_const, is_constructible, is_convertible, is_default_constructible, is_enum, is_floating_point, is_integral, is_nothrow_move_assignable, is_nothrow_move_constructible, is_pointer, is_reference, is_same, is_scalar, is_signed, remove_const, remove_cv, remove_pointer, remove_reference, true_type, underlying_type #include // declval, forward, make_pair, move, pair, swap @@ -295,6 +294,26 @@ class out_of_range : public exception {} }; +/*! +@brief exception indicating other errors + +Exceptions have ids 5xx. + +name / id | example massage | description +------------------------------ | --------------- | ------------------------- +json.exception.[other_error](@ref other_error).501 | "unsuccessful" | A JSON Patch operation 'test' failed. + +@since version 3.0.0 +*/ +class other_error : public exception +{ + public: + other_error(int id_, const std::string& what_arg_) + : exception(id_, "other_error", what_arg_) + {} +}; + + /////////////////////////// // JSON type enumeration // @@ -1240,12 +1259,6 @@ class basic_json /// Classes to implement user-defined exceptions. /// @{ - /* - name / id | example massage | description - ------------------------------ | --------------- | ------------------------- - json.exception.other.500 | "unsuccessful" | A JSON Patch operation 'test' failed. - */ - /// @copydoc detail::parse_error using parse_error = detail::parse_error; /// @copydoc detail::invalid_iterator @@ -1254,6 +1267,8 @@ class basic_json using type_error = detail::type_error; /// @copydoc detail::out_of_range using out_of_range = detail::out_of_range; + /// @copydoc detail::other_error + using other_error = detail::other_error; /// @} @@ -1948,7 +1963,7 @@ class basic_json { if (t == value_t::null) { - JSON_THROW(std::domain_error("961c151d2e87f2686a955a9be24d316f1362bf21 2.1.1")); // LCOV_EXCL_LINE + JSON_THROW(other_error(500, "961c151d2e87f2686a955a9be24d316f1362bf21 2.1.1")); // LCOV_EXCL_LINE } break; } @@ -7374,11 +7389,8 @@ class basic_json template static T get_from_vector(const std::vector& vec, const size_t current_index) { + // check if we can read sizeof(T) bytes starting the next index check_length(vec.size(), sizeof(T), current_index + 1); - //if (current_index + sizeof(T) + 1 > vec.size()) - //{ - // JSON_THROW(parse_error(110, current_index + 1, "cannot read " + std::to_string(sizeof(T)) + " bytes from vector")); - //} T result; auto* ptr = reinterpret_cast(&result); @@ -13062,7 +13074,7 @@ basic_json_parser_74: // throw an exception if test fails if (not success) { - JSON_THROW(std::domain_error("unsuccessful: " + val.dump())); + JSON_THROW(other_error(501, "unsuccessful: " + val.dump())); } break; diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 300a6219..3044cf6b 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -52,7 +52,6 @@ SOFTWARE. #include // addressof, allocator, allocator_traits, unique_ptr #include // accumulate #include // stringstream -#include // domain_error, invalid_argument, out_of_range #include // getline, stoi, string, to_string #include // add_pointer, conditional, decay, enable_if, false_type, integral_constant, is_arithmetic, is_base_of, is_const, is_constructible, is_convertible, is_default_constructible, is_enum, is_floating_point, is_integral, is_nothrow_move_assignable, is_nothrow_move_constructible, is_pointer, is_reference, is_same, is_scalar, is_signed, remove_const, remove_cv, remove_pointer, remove_reference, true_type, underlying_type #include // declval, forward, make_pair, move, pair, swap @@ -295,6 +294,27 @@ class out_of_range : public exception {} }; +/*! +@brief exception indicating other errors + +Exceptions have ids 5xx. + +name / id | example massage | description +------------------------------ | --------------- | ------------------------- +json.exception.[other_error](@ref other_error).501 | "unsuccessful: {"op":"test","path":"/baz", + "value":"bar"}" | A JSON Patch operation 'test' failed. + +@since version 3.0.0 +*/ +class other_error : public exception +{ + public: + other_error(int id_, const std::string& what_arg_) + : exception(id_, "other_error", what_arg_) + {} +}; + + /////////////////////////// // JSON type enumeration // @@ -1240,12 +1260,6 @@ class basic_json /// Classes to implement user-defined exceptions. /// @{ - /* - name / id | example massage | description - ------------------------------ | --------------- | ------------------------- - json.exception.other.500 | "unsuccessful" | A JSON Patch operation 'test' failed. - */ - /// @copydoc detail::parse_error using parse_error = detail::parse_error; /// @copydoc detail::invalid_iterator @@ -1254,6 +1268,8 @@ class basic_json using type_error = detail::type_error; /// @copydoc detail::out_of_range using out_of_range = detail::out_of_range; + /// @copydoc detail::other_error + using other_error = detail::other_error; /// @} @@ -1948,7 +1964,7 @@ class basic_json { if (t == value_t::null) { - JSON_THROW(std::domain_error("961c151d2e87f2686a955a9be24d316f1362bf21 2.1.1")); // LCOV_EXCL_LINE + JSON_THROW(other_error(500, "961c151d2e87f2686a955a9be24d316f1362bf21 2.1.1")); // LCOV_EXCL_LINE } break; } @@ -12092,7 +12108,7 @@ class basic_json // throw an exception if test fails if (not success) { - JSON_THROW(std::domain_error("unsuccessful: " + val.dump())); + JSON_THROW(other_error(501, "unsuccessful: " + val.dump())); } break; diff --git a/test/src/unit-json_patch.cpp b/test/src/unit-json_patch.cpp index c021450b..9ab86ea4 100644 --- a/test/src/unit-json_patch.cpp +++ b/test/src/unit-json_patch.cpp @@ -337,8 +337,8 @@ TEST_CASE("JSON patch") )"_json; // check that evaluation throws - CHECK_THROWS_AS(doc.patch(patch), std::domain_error); - CHECK_THROWS_WITH(doc.patch(patch), "unsuccessful: " + patch[0].dump()); + CHECK_THROWS_AS(doc.patch(patch), json::other_error); + CHECK_THROWS_WITH(doc.patch(patch), "[json.exception.other_error.501] unsuccessful: " + patch[0].dump()); } SECTION("A.10. Adding a Nested Member Object") @@ -478,8 +478,8 @@ TEST_CASE("JSON patch") )"_json; // check that evaluation throws - CHECK_THROWS_AS(doc.patch(patch), std::domain_error); - CHECK_THROWS_WITH(doc.patch(patch), "unsuccessful: " + patch[0].dump()); + CHECK_THROWS_AS(doc.patch(patch), json::other_error); + CHECK_THROWS_WITH(doc.patch(patch), "[json.exception.other_error.501] unsuccessful: " + patch[0].dump()); } SECTION("A.16. Adding an Array Value") @@ -1177,8 +1177,8 @@ TEST_CASE("JSON patch") )"_json; // the test will fail - CHECK_THROWS_AS(doc.patch(patch), std::domain_error); - CHECK_THROWS_WITH(doc.patch(patch), "unsuccessful: " + patch[0].dump()); + CHECK_THROWS_AS(doc.patch(patch), json::other_error); + CHECK_THROWS_WITH(doc.patch(patch), "[json.exception.other_error.501] unsuccessful: " + patch[0].dump()); } } } From fe71e7df1f06afbac4b3312194e0755ee21efa36 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Wed, 8 Mar 2017 21:03:19 +0100 Subject: [PATCH 054/530] :memo: overworked documentation Replacing references to std exceptions with user-defined exceptions. Also changed some examples to the new exceptions. --- doc/examples/at__object_t_key_type | Bin 0 -> 130960 bytes doc/examples/at__object_t_key_type.cpp | 4 +- doc/examples/at__object_t_key_type.link | 2 +- doc/examples/at__object_t_key_type.output | 2 +- doc/examples/at__object_t_key_type.test | 3 + doc/examples/at__object_t_key_type_const.cpp | 2 +- doc/examples/at__object_t_key_type_const.link | 2 +- doc/examples/at__size_type.cpp | 4 +- doc/examples/at__size_type.link | 2 +- doc/examples/at__size_type.output | 2 +- doc/examples/at__size_type_const.cpp | 2 +- doc/examples/at__size_type_const.link | 2 +- doc/examples/get_ref | Bin 0 -> 38132 bytes doc/examples/get_ref.cpp | 2 +- doc/examples/get_ref.link | 2 +- doc/examples/get_ref.output | 2 +- doc/examples/get_ref.test | 2 + doc/examples/json_pointer | Bin 0 -> 47612 bytes doc/examples/json_pointer.cpp | 12 +- doc/examples/json_pointer.link | 2 +- doc/examples/json_pointer.output | 6 +- doc/examples/json_pointer.test | 3 + src/json.hpp | 199 +++++++++++++----- src/json.hpp.re2c | 196 ++++++++++++----- 24 files changed, 325 insertions(+), 128 deletions(-) create mode 100755 doc/examples/at__object_t_key_type create mode 100644 doc/examples/at__object_t_key_type.test create mode 100755 doc/examples/get_ref create mode 100644 doc/examples/get_ref.test create mode 100755 doc/examples/json_pointer create mode 100644 doc/examples/json_pointer.test diff --git a/doc/examples/at__object_t_key_type b/doc/examples/at__object_t_key_type new file mode 100755 index 0000000000000000000000000000000000000000..03f48e68ca3efe94e36855f9d69bd72b6854b4ff GIT binary patch literal 130960 zcmeFa4Sc0nSueU%Iwb{4QVWqWcwRC1InQScf(?h?4}l4-|DsA6-FgHaFz^6^Fo2p>)Z2kH5LJnzoC zGns;V?>*k%uk)Lo=Y7`4v%a78tY@|p;ll^_J#+AdgU@~zUEBY4w4*y>q%)2L4j+E*;P7+XgDC#UHY$N5Z5f-K@U}?f2Mc%$aU9`UiSvLY^T@%v!?g9C@yi@$$dzv_bE6 zuf1WbLAKLdd9mreCx{SlYmYK=kM_S!eS`n9h({OZ@f@^!-7+xNz6O>ZhPLVSCk z$|TwErci{#hp*oE+f+9iIPy>tEpPZ>)kMuZIu6_WHv&y!q!}_PXm2kKXu( z^U^EGkzQ1x$g+PtR$u;e`0%SQHyd$?Z>RUTHhR~;^q%rJAN2a8n8eHU zSl%n!=)K~lH@uZaT9N7P}KQnms{(=XLMmtPLSm69O z{VT?quSh};KU}Fy^zjSfdI5R?#}Q!W^&OQSbcFdEIx2Ih?BQSNP?NH6^x!xBi^w0s zjr?*)rSJ8X%KhPX5I_D~iXYuRj^AbYo$ueuO&yh2?BMx8xb@=q0Q{afcEcNSwi z_QY4b`T8R4!{N?+eBkYaU)cYTm)-Z)pZwwHKIx7B_}a(7O?~D8_e13SRerp{0r_t? z%FB4gwl0vrH%Ao{`ki~Vb2qgiC4c&`RXr+uY3~C)gJyiZ@%Hi z>wf+T&xj}pU8(#SeoT7dEgh9rK=wJ}tj~M@-CG0q*1)|raBmIVTLbsjz`ZqaZw=gA z1OMJ@;7~RBLiO000SLdpW&BeGXVMDWA(7E=xL1()yc(f1c^9*^GwEB%A2iF*&s;P zj*#S1Z4`h~d^K&*HsQ&jUA1M_R8ch2?1k5`X&y99ffmgwX`V1meuJihizb5>G}){L!2t$D zzft&_K#O1>9P=wokl!Gv;3CMNErN4f1a~uFlJzQlO`t`v2ab8(sZ#SB1QlEa8MGi+ ztFQ@@+B`wF>z1q8O=@$*(oBv?w&)-Wa;ycmY{wOYRf9Exr2(y>0g*m(z-2!(l8Jx@ zbT_k9+Kt;3zl7Z{QqF} zG;?BUO^(R923m8|j>c-8a;$Y5^ceIKoUeLJAZe~4C&bTa&+R1iZ z!`vmc-Ne#Nj!8D)3hb9-EwHK`4;l;`j1Zi!jx!PPoPrFX3(hc4?`9Ff!CQGso3r#z z8}vpOq)&eHHiN9jERIU6TlydGNqd()}Av z#Wj8-b;lzQR6Oy(^LX13#ZH~yEW6@~12dje z|6MTp?WsTQTsNfN>(s|vsSlG%QFbC*sUHNEe+DxO;_(})J04S4Jf}{D*g!zqx6|QwCnMDaT2wdSn2(q$zd=>OMU_EIRL@!#?5c4eYW+p57Ib7abvCP}K8d={ z8L4P$B3n(p1}y&udJN+68;Lm{n_BVQ)U5IK*PYw5)*+>G=#l?#>k9eG9*o2%+Mj%N zyZ=1>yLO;Sg>($(?Qwn)anc%llw_PMmb+>nQ>mLA>G=oXjf~H+=xKj7eQP(So>i6$ zf=ycE*WD{%y??Dq>2&^PbNUsyx3OJ1bUo@gLRKnTKTMuMk&mMF#x|-cU^=#u{|=|b$y43e-u_Db+c`YK=Kn~BtZ-eA zy1GTL_v)#|eN!J5#pDnuAIz0%GhaW0}z1QB}^{9dK=kM1Si*(PQzjAq=bKAcx{DV*3WB7rojcPV_mpT+gXw=@+ z;9ab|{KtP3-6<+0@_+2i2C5qCvx_+AdVPjI=9#pic{QyGOZ%$n8}n)hCX||HSP0Xc zXeY;eDqZhBi7K4Ro&|@qHU91BI{rivj>w(WbUz|PVmF(EjLyjEKo49v(XUft+_MN- z89a|TP>y1LBZ%hvA}3qK1}Ctj8%Gd_2AWnuF&ky9G^0;7flJy@8MMgalAsqJ`TIV> z6jaErTk6bpDk)>~a~*?w7Qs3cs)10>Om=h%84{tS)5l=z+P(JnDxzwT^usmDp3Jy6D{hl6`DcH!qmt+Nb54Igc4kq-R(Yu=4UtV7%vm+L z3l3a3#u9l zsk@|2Rova+8S->}qhT)p=9Q?i^dSuaRdhc2cyRbPp+9VE+4a3hZ@XJeQ%q%>SN=bK z9Hg@*+c5tqzR$Xmqr-876m7(&&F_yC^9qw?k=FkXIbTo*S;{{{BJ3dl2?0B*SvCU* zUNa(aC;I7>z9i)4FL`vUXY6_)R(`13S;$C^GdKgrJiF(k$zOnA{-YvXJBZjtra-WwWYyNv%r%_KaECy6u#2IPqT#&t+kqW@Y8JD!B4OMRjXk!;h|pZ&aG$0iyqQy zSk@C8_Wlro(XfBMmq68(|Ni4p!@q@OwzbqfH7rxv7CisXZqUq{tkrx2-)G%8?G4KY z$}eMz!XpuY&t*!4KmEENpBl-v?3F zO{c*#h(;R1U{b@Op$;t!oVaXYYV5tJPqgEqWuW++ADFtVnjLyCTowmT*q2U|i=EKm zbh24MV--yV$fiTOgJ@r%DV&obwBE(lq!TWyknvaGf_5c@)6wbteHb`fxjJ@=Ik&L0 zlnfouDbJi!zR5GaoN*;xguDXs8*oFn!OSzZmV1H#o{k}3x8?7AB-&u|6r>7P*H(QU z%>U@I@JTLwO#Z9*m~NAYIGulr-rMAxV`pG#fJnhHkrB^S9EedCIdt99Id#Y6ddF1# zq3DfR5a4gu@vBfIg*s8AGvuDh^$U`W(?HkpWCzdtK!YRCzoh32m~>{It(sPyq5-lV z1gfS#$2LJdU{It-*YekJqKe>cK^EEM)uq9C&_jRYxaAlCnc+6&Fkmeq|@JUpX=Om2=6y z-DIap_J5e{pONgBAG3|@eWnJ#yc5|kpBVh|xn!ScvR9DoCr$SENcJ}hvL0Av`Ma2idHee@UFJv*~OXNu{eze0>LLPNGhdz`!8z4()U*A9; zq~A6P-O}x3lkyHc8v}q{$M1)p0&XoK2y~%PX1@aYKO!tV2EgoEJ~Q@lz!38r?9A24 z)5v_o5|KgD@R%@28tS}h0A-~*xzvlSO2~nHpw!1uAFLlh3}`Y$;HaOmRr5@&xsoyT zUi^}~@?jP>^|-)vE(#Q{Fgt9G+i7&jDa;2GsNFgE&0<{7pq2qSlN_DTpZ~+?)R?e9 zjcd2#fUWglH_}IX-@Co5$tIex2f>lWr9Ar@D-rx<;n4*fZs*_U@aum~e>!%io2{5M z$!5_|0+UzKr2?0(9Vy#KtJ*nv~Ysb`%(Sc(L8B2l0V<0Vd)Ji=%|DTUi-(oXq25%p)kO%fG zeh7*P?q;dyjOFOu8>cTq6W)aZf;t^r!vgLVs1KC3sF-I#Whc@#Ev!hI&>a&2d zJ$=SUVjRcp07s((xD+{fa>TX#a_r13U6P{uFM0udPqeM!McRHulCvbKO;Ma5_L!0@YO*OY_e0zO&gUBXgqn+5m z(1%6)oJSwlyA>WhRhbyebBhCL=;Wg4S4I0wQ!0a&kvc~#`3SHhWk9tjuw{c`a+M{m zyVwl)tilCfD3DyMC)e&%%`P0k0QMr!%2;&BKYs)&H)}4K&^QTU50@aTnY#Dup~9hK zT9+U8Y80|Q?Fd>g!UjM=GC8JoN3UDA;iXmVW+vIJLdM=^_|;cTY+1lNooP*27WrN%O21v3t;vzld&lZl0=fmJ(E&)w?6%^m|Y z#hU6Y&D1{05Qb1)8g6pqglp8H1vD}thQwdw4|#aVSJAN@ol?lsAdQUK_tp>*{ci_p zWU`;4;>6ePK6|pzE`5t}vMYaqYme;mMF`4}5N`nX;iqA&=Z`GXL%En!?VE6KD`~u6 zjUCLkP@|1#NLtdc(u6h|aCF{cFb`skVqW94uc;oZHVtgdXCCHMyBwMZ2buw#sV4WU zPBuB(E9h1;NVn_w(^OX5i%^w7QGo&(%s^DVOF7KVJ^Z!Ijiu~65jlUu*D-@}anoLj zp`0o0A0tQ z;d&$j3jHQDK;YU`3K-!}M>thixi)_hvGT`AaH@&wu~TF>QX;UEUAE zpqeQZff5kBJCHpS9#gez)O^w*p1uH;^GOTFcTp2n-$XSWNdD$PQs2N={zPUZ)i*rd z)3q|>j*2CV1F3Y4L+O7QN}t`i|K#T`9P0Q>jQnvVb0E1{of|t~Omwe?MeKWJTeH@6 zFlFE(?qU?-^x!)x>(b*zq)AyuG5-1=;}85hP7J=|F&$I2L7-=AEEM8SqE057EZYTL zHBQ;5}wxvb#SE#TxceYi5Y=u>ve?{GJ8Q&h8sg#r$)e4N+fT6zlkTMgz z6}oFMEe}XoQ!vCdoJ7C{L9<<6r)n&H$nyl0i~Pwy2s*DoP%;Ot>yzveD7ULTYqIRy zyz<{f+Gkxm?OL^*>3*6yyQoOKsNv(cq6siPc7xun@I*c&rk}2Kc%chC#K>Hs~s>BkiG`=Izd}}Um^)5H_QWj2Uoj}|Bw>5ek#A~ zDNG&0-Q?+RjIg!!PcWV&oKSvp#aie?CEu?<+PT+ADl&dS{s$^Yn7BL>j8~ueHD3ii zY$T#sVgsDz^8fw=Tl%hE$}h>-H_?MINgdrx@d-IO{{u*Z7G1g?H38a%{sY;J_1xo; z5rlwT)VBWT{uqM9rdZc4V|V5MwH0#PdVAmV_dV}9Fk(Eyz4lBdLgUtG zWS!_ZU|pOE4rDjyq>nz?z?w1|Q0b}OpDQ2L+)N%w&y>yewlP zk;5A?N*iQ4@S_V-5SBDJBoU0Go+Ttm1K2MOKP`X}(xOHzjzb>`qb;d_Kz#rd_9q(% zSS76`7$C7HUgM>c2DBANkl_(PzTPNBo-}%`@Bsl&ULh*gN0qez5G%$ez@0ToNeKf} znXxG_cF1J>L0G#V1{_&G6bx_f1iz3CS=ZbDkojrpi)WE#l_+g4pvv=i{g*9)tB)yi z11I(3Uy2SG}tcN1lpAw=rpqL8W+tvk7dAkXS0l zj-1XTD@pc#{9UKo$g_;cA`)AM1hNQc@JM`ncVPvP>=Z%^SSs1-4q~*~_?1u6I_UB@ zLu$r^`4hk_Mt>4^BuYv-E#-k}r8Q;;vM=2+k*yFU*)jl(G=zA)1nHRl4C-oiH-p3r zrKQ>w9G!K$#Of-o&nQxTR*?cwc$}~UgK`67H?0$$qQ0$OsSDykys4L@jdgw=-$(f};MfJ2SQ z-A3kENEj1n$>0uI>ok86z{oTv0473gNmjS6M!SHYC|F$}srMT*U?zI(&`SWr24M{W z=unLbs2UlV)i~{%n=02P5l}d2Lc;{EO{IX^{O~t-sLkW7&Dk#?Fs1}jIChG&+l%(S zWZz-U!0bo)vo4V_LU}(b^qIYyU#)-G$e)j*@A>G|9djL1hp@|_jS3#>#Z0}Czc4(j zlbaXFyN3>oFf66-1JIaMPCXZ?V~Ya25wx^fEar62=P{R0Ty;U$I~V28*?af}G5^IJ4Re;|uY z8odHiE_A))RoYhF^<9EAlh@0CU;gx6a$7aKpk2aaS{K zh#Ka#8M!q?4e{<&82H`@9bsu^Kk{f`0@fD-8-?RgauTxMxI$rp{^&F5eHJLR9D5P(k1|J_(f zuD=M}w4DDd+w#B67;&5<&4?ibi*{TkR!FHjy1gVy?$eb`p`fx=4 zyX{8|A$#d)kq&+wHoydITLz4>rSng*QO0Q*7bp4akRN2v1Vb6C)l2^R&oF&!#Zxb6 zkpH*tLjFGm{!z~V9S9C)8_+!D<_6R06MAvU&#KyE&mTH=N|@;3G}eg!5&Bl zF)a&O0AGWN%iBRCH(r1;EO8z@h<>jT(Iq%gx`u}Jtv=3+U4;sA$G z0qx8Aj<`_sAG_^1m8fM}(zu?z0Q^)y+V}i@pHeB+)6#~0Zp%M(0ZYjVy2+lEI;KU7 zyo$`ngbo9#TNU)mlq{b6u;9fa-w#29#nispG$**hDOybZvceyo%D;w>)_xNwkfb?} z+QS|J{vyj>C+K7jh6%lpbuQ5`=ME~h+q$dCrNAD$g{6v}yuOo5zDWd#BRtvV{}HPx z2rB)e)9e(W{k*^G6**8h#^VV2=b=z7gce7!wjtZ~B+|hlGj$88$GL;|n;mlH*@hz8 zL1i`5aR+GJ!~p98p`o{~0$J#>UpGPMB?`5DXeq43PwIURX>i%jHmSWPQ&mtejfx4? z^Qun}A>Cnz8~|9VJq>@+T(DB*SIbJ3h^$l@s%6Hg(6Umc|N1goT*Y?vLvrax;q$+& zVC<46>YWbNEK{Y#aw|3jJ(Z)0N0Vgg>Z+Mtwu%6=0B#bm3+E3hg1P0O!+85?_r zvF9hyb<3r&?VkKL`pH%V8I2Potr>7dw2QsJRby*T$zVS^Va7uk>tUy& zBPXr1lXH$T?3$44o#a@Lstf|%CLrJ$ z8svJMonHG9pkYpayQBJYsA@z7Mdq;(j9@hr6lKcKZTXYGOJ-4909y2}BOYsXLhJz% z_3m5DrHL)R3wtMHq0;*Qh5a{%kP69d%WT)YmR2%UoF)5t_&=*k=1WyF;VPjMCXj!x zODz;DGCITjq6Z3Oi^{#R)R@o0qIp;37bjq)5J$WW1!o8Xoem^`Rf;hIj1FT0 z<_x`d5x}V!V**%hFh{U`aK@_g3-wV^F9+q=rXPY_Nk_;5CME!$Y>c2(6OCyTp8{yr z1QFE4lw$}$J7r7&Sukv8QlB5hQF8T!`aF|>64NOOD+Il4QGOOf(E$}Cgi_1OAu^{= zuO?Si(_aGb!`55+M$87Uhtwgv2hd(KZGj}qjR|;t#BQmSWXrKs`vef?e6q=})+|g! zv+ybdx6X>tS+g)dwq{|>zY{27{v)u{z&H6u{!o8&8qV+UZ_dT}-J1$S87Kb+zp)z+ zVJyz`^}#vGegc2jp@_FuqvZCM9o$~DgWKL6+#U|M$f&d*TV(UsI8Vy=VSjfsf6wvv zL4H5e-^%x1e#enh=n`KhREJ^noAAAZ{t~U^{XxKueChAmr||J)#*EFvy=!#QuL=b2V zlJ**SbPkvUZUEvZ_CZ;Cox2q%Uj$!(_X2_~0xkkB11>|^2r&Xezm6`1zOpz17ZQ2w z)t*y6?rrieN9aGN7m2BAvH4lN$YuARfaq6M?YD8&kxEC8IYy$XMkDPeFNm&jp| z)TfLIBsSE-;dweHv9Tb-9PIp!c&g@N7KP@8~lV0?tDTAcepG9$^5l{LAsLP3x3n1qzVPr5n;A*4O7=b0`k!~$oZ}T zfiMtCWtPx&%Vm>a?3ny6+C{3Y7WeIT5>-VEZEc$oQCPT!{=jlsCn6QxHwg5ZfB+^k zV*-T;B}uBxeMtp9QJ~l*b-yUiS=~Vc1Dr%x9&Kub=18r~I`Yk8RTV-%?({K5fkJ4u zZJu>wh&nG?Z9;RKz+9TGv@B~?)FzpwRaE4&CUbt_*dvbYxR|hmz_RivO&Qu^#h3tu zb7KOYu)QoO4E=KZ3luEBFz4!Y)AuV#RN3kL&^I^~Yn&w1sXW#?&T7cP@@5Zu>UuB! zh;3F8ImxcxB9Z?j;;%mv{?QEiBSin!6 zXe2zT=Xc^;d4g zp9j3_`HLvr#Miqnt*l=Id>!|KR3?sou+mk#3}3}|%%6M>Nnkr>FvVVsbZXECU2?nD zJmmRgd9c(muP0o?58Ui~7)Kvk0jCt3l^K;eZ5UvRdF21~qZ%z_E5XdHE)2HpGqd@- zv3VqK{kip<$ayuR#idUwJ)%3jk$=?G>-a7Q2O~dNP%01(h1R)di_Cao1-P` z>LUnr7FE!`|JS)JPIRDIrt_QOH~n7p>pl2A1Hb3s_Zs}(gx@>x<8jLE z{fYD64ClX0MlTq93uGKE8P1}hFJOq>kR|D5P<9&i^nxb7XwRp6h{<*fgDFFj9>x2T}nU-AFS7hiSzQZmN9;SnVvB)}s7mBm)&US?o|+IgnsL6xK4U3iuSA z!%pHfbXZ#ZBBs`TOq(WY01DKtwoYiH!|KGrJi5(11`p(kIAI8T>33YB&M>3t13K5( z6h;t?w=m~OwpmN>VBUdE7c!3La#509{+klIFBt-6^Nzc|Ty$Zaf<{uZlAh*A<0hpX zx|TRf8TKrNz0X1NTo~H}sP7fVI&p5Wymf${tl`eR=kl%tI+hrM%}*hkQP_k?CjfJu zNu8lV&DAE5A0G@wdlP|MNTZO57ds)k4KOZv_|L#L(EOnr`Qi}ASzwv`?)g*rybyKb zaf@TzCe*1^t(bzO<2(q!Ov0D}?ItQprtNHyuZln+chj$Byy1{ONB~oQr?o*QQcHZT|Decc{%*AXfhOx2q(gdh8V0mnZdUdj7^gVp4C}JKB12#T;(j zNvaW3FS1+4-I2vj23eAj5*&`zq8fUQT*=`oDyQ#AyAP5xhr<}l>PjMOj35%&n}&O) z-BPf(QAC5m;s|4BH#@;bkIV0BuyJ3Z6u+5{Fj!Qz2478X?xSO3oAts<&yJ}ahtj9O zyVp?#VAJhHssaTo6=oiaL>L6|Mji5qLn^DqcC+{)IO;h3WSCf=W1Lh#Y)^(pa2Te0 ztbL#<5q9$g0J@(s>88OCXW%FXNu$JB1~g9<=NsrmpPF|xfiRp(+w7-e`xY~9=rW5P z#IWEC)4k$^g-5*yKV}Y%u~;Kw1H?vrOKfkl`6hXXK(dbRlkf zbQa(c}-`6__9>wnH$j7EAcao1sV$)xdI_V3~dP8@w~v$V9$3Wfl=kn<)h7IgbwRhIcr zJL3^|4PIvM6*f8bKE{C!n4-qf5OP-rCU>nOKP~GxB5*T1wONf>f=GiU`~%jU=P=l- z5i4K$2z*{#Vz3;6ptYjz<8@21V5zUtG0f>xTh;4?%Cg+E+;If3Y%wO#Vz3VB8%N{f zQJ=Fx51(<$^8`zhoT3`#QV3Y}1qG`w5~KmhU}xw%?)q z09v(41U0Z~utwn8R0^ogcYSP!+WZNG%ine@Gvy09kDVg-gms+OO~dql(ib7o$9el> zXV%DEZ5s_SZ(7O4Ik%=u;k)jVW)G_pJqsY1YbQRVcIa9gqCKroDTy$R6q81+9Z@`Q%N3h+;n#Nf z4!T-~jEy+6dbG!k9PO7C+8=eAmM&=AIhz-VRe#(0yv$f@$BgVz?4wW$EIJWWsDfFB ziHS~h09%uP7};AXU1-JXu2-m+l0h)CxXyu}L(_1Y1*DV+NG8S|6hcXfi-61cfq9+h zl2ZiN^^)u>pnVV^ANUh~Yq(nkitA!A@h}t?SPb>_zYoS$h)Lrj0kQL6c~~zP<;S!P zx%|Nbmz)MLqZ}>LtK!4~k`kjF8F8Z+8Tya;I#Z6q0 zDBc%h7N5N?cH`po6&WQ1S5R{U$WLo%6M@@^b9IEkLtCUVv`=C^dk&*~ z8e-+|{V;sWNBLaEv3_V*6zMz{Sdzp^>>McHL+p7-Y0CJrIh16TpU1?$gphEF;6gwhUrXHZ}pxhNdA2GNpqiyu3CjXRsKT? zdL^|^-hW(390YonR}k0#~D;?`tf>59Zkg@R)3K{9m};3 zb6N)z<#{_Dv>#l_e+=$GRuLQ(ep~*7s@}o1DQrXdPKCDfuE)+y!+;ZAxKqQiQ4y$n zk3fJ@t)# zKOhd*Zb1|mPB$4pe|Hy*_UrF9foXtZ%37gQ7F*l8H5t5$jsqte3_dW`SY?oQL!c!I zB_0;V$$yI}^$}-fN+6E)$AOlx0BtpO1D-Z4>HLdtnW-afOEk|a_b?whf>vRa0K#k$5{4n_5dHRMJhXa`y2nm83P?}L!0a63YSeeRa}a^sZbi)mAbS`Sa7R%> zVSR!F@=o>$Ky;MRaz`QJj-r5D9fi^cx{pAZqF@`^#HQ^h3}?}ZVH;YGI>wqWy_b=7 zCAvGx3tUq>q?Xj@OijQx`k=NcM{RPl6DtXA+OiL;$W4O$SN}@Ahne5%J=>DO-lpPv zb-X0GF?x?Yz>}1kFObKA-l10n!Zt;=-xy_U3Mm`){P{@R3>wk`+)wxr8tgqV*+^s~ zm&spT z&$h(U0U(Me7c7?SQ7t^sd-3Vs>#NpJeim2CJm8}sGR&};=|qyxugv2sf5Rs=*pv{C zM#{5(L_xC=0$lhZtl>_Yxd{S$;AxzWJ$=o^%&Q5PPu`(L&8~cK4ri%FNvBy`Gp*-W zi1$uxUP0=M!ki!6IiPI$g#q+O`xz9DtBO}V1+gNmv8<4`@50JJSOl?bfIDN<7xb0} zvb5<46*}U`xTq8d5QSxaVgou-b(eT>BdEkdcbz{+z2#1CLsS}lN~qcI^mYr>dqu_U zC^H8BD#yUOkq<}JnioSx7>@pm!f3bd8H4?vs;JFet*}$ORI@~J9<|~A=*wgAqKB(? znj0R}MyK;y(0-}?ps3~q1l6?X zAqCOycB~@8DAi}2VY;GVaj|u%28+o-ehKe12~Dlfqa{AB4TX&2h9lJ$R@@dh@^Kjv zF0`O&j4ETip=)p!vl(W%F{*86F~i}T>I`0gOT!v|!-fMlaq+?x%8#`<520CreCgv` ze1r~IDE46DU%)iEs?dLt<6^VQ`V%l>wZ%K1480#BDrw}?h>R5GQ#e6e2&W`_C;qP2 z5xU?`9&~<-=6joeoZigA^w#`X#O%+5!CzLy(HAV2$bj>Ua zUz-5&wh5RHp6*+z;aVN1Bmm743lfLA;{-S^T#{p(;uB||;Fx!>pa~|WliiwtwhUz3?Xn_P8n=VkVQVZdz(JepSHGx{)ElejL^H1IxU3~p#DZXKLVDYf0 zz)2FPCp}m0HYR}WT4GDJegO9#yP(_(Ru_PsfH4CqhWhNlc*sS!n07+|*T)+ZkYb2| z(R;FSsoC%;f#su7Zln_jbl~8?H$qFmPH_?pVGfJA0^A&Kb_Z(5Yx&XE# zO>Dx%gr#+1Ve*W9HTg9O^3LBP)f$Pn^`0G4!`6Z#8u+XaYHPzOZQ4&5IswayPM{*| zxsv?jA5-sP=C^v!wq&5ICYhu@Pk_x2fapE)fQbY>U}&sh*@{4YS)j2fzs#79m4I`u z!ZDvg+7|xZ5hUr*S{I_3r8krx`x#QW?PMv?a0dX2~2^79TCmVw#mQ$a{c27ltP=<6&>9?anaO$t4q)9bh}Am zG|zi&5B@Iu!PxwXi@v*zI|`CJC)Y2co-Y}CPuDweY#Ez9yKs9LDhriS*s4`CUdlwR z!DwfWnRPQOIm%Py=nfo z3>P<$*Cbo_eXf~+r@5v4)7a;);xXWD6w1UP`j>T*vzqGDtCf0*Ez~u-71h94z z@Ny9Q*NR^6SCClg)kcJQi3kO|B7|RkN`6*6Vd9gYG3eB!S2SYov8IrdFTKoupar>RA(?@Rxz`R9P6&vSo6hV zFHScVShpk;KMGkKE($D;1Z7o;Q-ML&Z_XJ9#Seuf*tL)_?qi^w228;E5?~ad;73^r7=bKcWJ=k(uge&Y3M#O6y5UEO z3i~B}j!JAO>4bw)wb?YVT#1JY+k54>8QM`P9rBDl0E$~B!0co{Ivz;w3kM5#et#3* zpohRC1bBYK*r+jq#FweXooc$$*<1ec%3WfsfV40Um~mWzB<@@bi-Az=*l-{%-U*yj znvF##y`bpEaB-C!pAMm7+@Xf;Es%t*M15I~0m88%F}!8cw(59x#Pb?%;MV;`1NQDG zh*KU|sHg(#?ly4{q@ndi(wKJT3M9=rV*=3G7!$yC!o~z#dnA_=q8ALw$|}gE%OVj zpu#xFj-3cRuT*1c8qe0mk=@2DhZ#;kGDxOdM3BFJr`6}QgcWaJh5{iBIF0$KDr9b0 z@!}9jn1Eu&w8P$xu6N!x@%6jU-d31X>x~YZ2P5)`b18iSPMI1TQ5Jl`ZrDb&gurp< zQXye_q6n1dXkBY`aM$NEx$8YYq@U@LE0B43haZ z&%n%tO(T)-SicStqzdRmN5}Cu^?K_szH_sB0Xl$IT}r7b-+4 zd(1#uV@^?8sRs(}Xj!cRrJ9n@1Y$Xc@XA^?OR%Ifr!J3OaEwxs>H@L)Db_k%kxtut z##9!BH3X7a^%!VH%D|P@iWIGb=BdSsbXgQ?D+IPt5NX&b+zCCtYQKDO+>I!0-k5<; z+2SJ_XIRs0hozSJ1`#lP@k}d0Pb!wCZK9$U{>Q1cf}Eamc<7VP#7?w zDuHWLDWEowVR>>+ZN3<>@>l&Q1jc%e6mZE$+XD3AAK?OYyFA)mUJqfk-t&F+@qSTU z3YnSZL6Z!&1Mz~4_y7zna6oYeQ?i(K(wN~OhyIKpF1hs63b59tAI>Yx5u^h9CvS(f z?ak!Ra1lsu=1c@kAh{Md#^9G+3r6C5REoz-eCQKr55Y`el#w2tJ0OD$+_$bf*U8lI zTe`2>Rhj69kmF3q5+4}Z>wa7i2Z4rs7cK%}o0j;7T-abEOYReNq7O_Eh(VPkU|GpFc=_jfunRIFdUP{ju6yrT*fW?5}^rNw2o z5z$iZo^Sxfj7AUr3y(=3g9#AhLNN}M6hQ_wBBDLSB85e6ZV}15!JOoysY~69b zDZj9rqxl{KvvRA*RoC$$=C!p8T|qy_TqF?UnXmK;<+Kt4XI4USiKno$`#NbxFd&-a z$UJBg3H2hwQ@!Zp$`oWpA>^#M#nc%3gZ*Zl$TUG%kQ6tZ`l3@P$?YeQ+?n4>MmVWA z`?mCf+j4;&1Jhk3Sd!g7ssQ=52kf3jw6UOIac(s1vvoNzLay2`S4iJOKJ}qOANYG} z{i1`F<x|HV+J?>D>sDes2vBPf*4Ch7%K>v+ zhcEz}m@9*>9h=7pnfaN&a5>S!Fc(gxSB=6YBXz~=)By`O2o}tWd_`=Vr@8P zp)A>Uw>ZEQmb`I}X=n;roq#AffjmGOrEN#?xY3>Ycj}}l-6UF28Y`?V+Y=ObKQs^f z+c`HJPm2Lh$L2r^;w!a|rFUy!=#~G&2Ma+kEIw^QzGWHt4Fo9JkCTOIe|wQj&ux|`51*KMr)(^${&VY7YSu??S}UgaG4df5~pij_{$(xT!b2<-balEOfnVZ zu@OX$xYbX7Pd$;*oi7OgbSL6F;WniY1RMqh?+|DXD577o$DWw5hSfa9^Y-C5t_aF6 zbOSRe38uqgM>uFdf)im*(3l1?6{?#sa?*_&NPEh`8RA5QXe=uDGJRm!E1;{)5L{Rm z_IHgH`e8*+kTn?%mC~u+CJdP)6=ZJCezOF=!t6w-jHH&Eipy-m($Cg(n1L8P3){Nb z_YbTO?zxECa&X&D{x~GGQPjJ(E71Ji%v4-c42vse5szw~?uB@;+Zmo$*Jx_66t^tJ z=0QBZ6;egv=&>3Hjp7>dcnTv9A%!E)SMyvEHmG&g1{abzTO>PtN?fwTUxXXRA+*W7 zF^}4QKeY!4TRQH!>s3}U z?K(v$j5h&rP-1?vfaKsI7uXgkSfO8Jd8}At>&yYxZi^^0lwU?Va4MX5b2f z4Io=Bfq)1qP!m3M9-i>}HR1U@i)Ricw3CHb%pCoq83F+j&m2?Cm}ia|Wz_~?*sODo z=jCV&t5!K?%l2EepFo&f6;Cg<-|us0{s+IpWTg1p5(u{C0=tf;yFmaR3}8NO4+v69 zFzoo-%*bGWjxP{$RoU(~B(1m)y|~?F zRFQo4N2kON+F{r*Vh}1Ge*!{*iRsYw-D8~-TeXiFJm%I zrnc4G+j4DXlOcr3%P_8cvRx?b5;ymd%?Y8A+#x?xCBdK zHbanRcFY!q)djFhG$t@9p=;GH0%%BM0x&BxCICwUV+2^9S8!=KT7J`x!f;bFG0d>? zYjhI8dQ8ArLdE;CLrpoV0Qjon)OsAppkc>GpZw~(?Pm~zev1j!D+Jga5?1RG zRu@2R3Tp^Jsxc;jchHOxblvjw$uD(Gl7-a9zQ1xFEp-Lx?QaBg9B0Sszm0@&^{Mo_+HtEXIKA}DevwVcpNNL`wB8xz2WsxbrI z*VfG<54&Rofil&qma0fn6+7+@Vq3`TNt ztP8^{7j~=)GzRQv(6D3Z(d#1)GOQq(FjyBshNi<_ymM6BK#sa5AMAc08rbh)6+F;R zrY9$4oIZM@&Gnr&7(GAvV-WFcpJyyYf@IInq;>`aV6w@-5IHWD2a%-q1Ly(k*O|Ub zy;p;!YZ9llDogT)+;I1L?5321ynQPOMU!<*&k+M3mGW-N8gurlw=xS%uhNyh6MAk; z7~|{}P(cdRb5%`Dy$w0yw2L}UV@-aAJ;{bKgR97~B^Wk0i!LS-O0!$x^|sWRA(wE?R?NvNR^pehxV8(<3kSRa7!M(F*JO{y&Q9O*4Mub~0qj zs}a4iMtO_VDmF${+Pj21fOe#y^ufxD5Qs*a9M%gnJw2Ufp%eefLeSb?lqm?i`U0xIqvFFylSo`OgP<5ZxffbS(8 zU+n@vuP=&+Kc?kZJ|~Eli$w=PiO1IF%Y`HYqKnR>z&))&T*$dnA%no22?%UcTbD0V zEEkegP!Iu_JBUCuMC|w!YUni+7l`Ez1|J>$=qWnnhOteB)LuY&o}DchnFxwpUi(2l z0;fW^w|ctmZc5+MIKv{xJR1Zmjv~;$l~C-H6>rUR81*=`Kuf_L%TN+WO$b2dQ^-Q4 z)vJQ>so}!BftsqpSubH=s7Yf@J;^+Ti(j{{nxA$lrvO^_zY;;*HN?Sk2X?C|ZCAPE z;ZO7C1cDUzzeaGZrgYe15pWT~mjTar|0|=TBm$YANzU{%BD8@$E(sP_)R#4eSxjtl ztF(1t@ob1+sil&~eg>hd8=8`Oy|}OyT6Rr^;|!{)aGWJBrb0!<7-p$rDmnFGfKpoAU0;V!vXZ%vkH+$0802B?s8PXhNc;v%a`kt}VcA z{nMC!*-B#KI|TC{tb%?(K^rCzY`@8gHUA^rw^k5L8EX3?Zr1aTfaj$-y9hw zh*V52L15W|1Rw_+6M*K>7(touk_uX10Mm&A)ususq%$UfsaU%KHva^NGbR8IXN;i4 zc52mfA&G!!pMwg-F-&Np>4;^g{R~vdfl?uZK#v0nfOi@bXgeWNE+na-kOEkVxL9t8 z*v_#jZ&>?6f9?DXi+EiCr=W}pfGZgzC{Hf^Tm^)YOpRW)m9uFSLGM` zZGIaSPjtyK9)75AI7qJO-cHlZB?rx1-N;I7=2}tkS|_U@W-il{6EaR8&s?f+JQ)Uu z{5w6Ixh`ZZ&s_XWYQKm9Fxlka*33mDsl5k(i^6UY33rj zedek#XTSY=W$AbETa;FF{ggRbaKqXO9x z0?g$C#xUmXH%!$Qk^Nss#Uf#Tb6o_L zrU)-G8~j}VeNOP_($CuI&zQbJ(BBaKhl)PG+vxMC=@xzDl@skSYbwi*2-G*E4Q;5x zAXg~`13aayIxUvMxk5GtVDn{6z$VZY#cc{;qu7`L&UhOWfIXNo0ay(ZOPalMG?p5J zDYyOl>?aHtA<2(|A;$SU@mvEQ`)jMe6Z)di4iVGRMuD#ipLs~96F z%PP70UwsXr#k+~1HP#d(6M(E@OaQWqF#}EGtB6#{DnVe`fdn9{7!!c3VvL|Xjgtz- zngC=K1*%OGU|1RxfUJT?oYBkzkX4KcKvpqEP@1P|E9F8G0nt7O6~K&QOaQWqF#{EH zxm3s?(BnV?m?4b`KvpqEP%b2?ppXLKTP~IxqE>Yj0q{g)0+>F9)de7{7!!c3VvL|H zt1Om_Oaw)qQP4~PN&{m8kX4KsXii>WkwaDq0;3Kj09nPD0Av+o0wJr|Wr_f@iZKDG zSzQwXkX7W@G$>lV;z1_>Ww|i{=n#Zu0+`^937{p65tRBsbKZ3@2hi&8L{KjW9YX+> zZA<`Dl`#X=%Pb-l^%4YjJCFcm6=MSES;hoHR^+nZST^M8)JFW^eR_tfcuw%CAAR7*nD@Z2HvI-p}tIV*%idD8~ z;1OC@K_RP%f}D_X`dC&`eWwjZ&kte+nbdA#EY54GJ|U~1K#*1NcS}|Q0$BxrTe1qi ztB_RyTCxhh+hmnekV000Q(0C4%9P6r=PBmwER+P0Rp{N7y#gUn2CuTmJ8|Ur@gA9GuWbs>g9Rea!`Z9q{8cXjx|N>IWX$vuKFP;e(neT&jipFq_^VRdwAA{Cug`B zzmsMPN2P|QtLdKnR%s&?eI9P}h>p!+nn#v2ROudA5dPeX8P^w>oJcYhlbz}~Jg0qpG>GobO3b|faXBY|Ah zSCMYl@sGeFyl7mykTZMvBA}ASdtz~?R3=3Vj0zO2*aIx4aUQ(JWC1pzzW z)16MR3}8r1_yTbkXuK>II{P-0As??q1ZghQ;w91D&}gAL1yCi%1pEd#{QBWol103$ z2MbhH&pvgl8WGgYu;Q2B>&DzaLPwl$u%CgZ=|w~;WRV~+;6MW4w#Ec7%NipnPt&A= zktGn@6g~xt{ER9}9JTX{qH*U*`z8UlK~0>XG!N-&j@AUU;6lzhsK7nw)SO&`7Xdg-_zB34vm_ zz<0d1T76v6w_c_&VlXd*5Dx>{c+vJ3e#yiNF~p&_0ZhgHO}2r%dpW3R>3T? zei1gm*QYsoVNpb{M8<2l)m0PW2KvJW+8$Faf4-)GW{I%AR%jn4#PWbs5;j;gGsOwR zIC*GCTk>LZnsCT*hZI&A%lkRd;-qCf4(fB4n0pk@2MTb|SynrwK70reKCPiZc zNh3E#P@WQws1!1YXt9nFx%pjf*v(k!)@?PL02qle198h?L@K6)Ah4k%i}$lPjS0Xm z#smn;Qv#`=B?PcuQ53xs3|7{wCV9XRw2d& zFz*>NP$8?OLI!~W2NJ-%XH1|?6J9PPsi2SoI4Iykx*_7&gRe8(SxxQB3t_;gq>5`T zri~Glr-XrWk%^$li_W6}wx*2;`Cc1tLgk+|&dhWEc~` z*fnOLdg(=^qF#c4l<+%V$l9)pcfA}jNr8A=C=62|=`lyJm?OxtPKL~UI0z?ETrbm# z?#DY|Wz4FBz^xjzW<1V^z#fMoi%TTq4(hmk+8YkVDUn9Ev(zI%dnh?*g6Tn4S`*Bs zQmb{d3SxQcMC62w)5ptG)py!p^!x+h_1N=eEVB7jpVpWcdhD==qEb7l{W<;?6O2l| zSA(Tkp3=KK!IXo<1j8Vu35FAF>%-Qo_$O%T?U0K{6y9STyB$ubhp-JZaAQ#>W z4%HnC5vx`^?=Z!9)mmXG-py3G+H8^89v%WP1HO>~@xu#)(D5VmDzkxedGLf_B|yV^ zD9`L70LEiX0F1|&05mPe4C)&qh()7wz?G!0e@%U|9Y=gI9rAvr3FhNaHTeSGlf>#L z?!)4-!O5dJR_*tLH?CD5{QUY4;0vbVm?3$&`k_4k<^-C0@bl^5=W#_-?>Xm%6JBtI z-{e)O-F`);tW?45i}?7t-f0{sRFyusvA%}(mIG$#^{e0;>pKcrTn@|791tTEn(11^ z2ju~*Y6sWWKkoR2Lyj0ZSM%G~^R4{7mCgTCtqij5iyY#rwByjXLs?yt&<;-^nurXdhI<*eaS}wYqm~t?QkO{KS1$*YxaJoH~hSf%~Dgsi7rX z9ir?AJkWij+B5OByWxCdaA~f%U*vBwYKj0TvMLH95MWg8r(JSUQ1!Zrpno{Jf{Hn4 z@0GIyJs@d=MJESph%~r*8a|~IvsE(N7aN`;Q-a<0S4VDeGt}5Me>UaN8aunUS6l~q+*2!gBaXrIQ z{s5f)0rdqF>XoOjCCG~R7Q5+)t4#&shSWHT76zhm;IBBPMx7HR2WPInTQq&_9~%I| zLOB#KY=>VwJRS<4P&paHwyfh#h_o?0QfSiiaxAbBJ6a`UTh2m5WGpK#YMM2%DMjFY zZjKWt$^xsRgZrARq`9b&am`inz^vK_VzY?()ib_IPL3(%E5;EQk9b++J!nu5KQ#?r zja9AJ))@8t?>&6)mr-}PT__%mdRL9jd->;%6I@aU=G%p?dUIq2rDiU46gCZ6THAHhMpFRu;ikT`4c<)54{EY~`&gWb$FtloaL<#$K zRbS+SzI#OE;^xx;yt!vs?r}Lm+*m+!N&H$D2RRJy(ZVkooCcf*oWtD%v&OnJ7-z@he0sZj9hns+d$& z7uaJI6VE@9Il>4r_@nHwtoKTH*LL_)4j#7EY{^Z3vr(NHjka@ z<#d}t-Nt*qvO^08vcWmLiL9>U%Wkj)aTjQ~3*3b_HtB#jHrauI5$uPt09@FY;3D8R zL4fr+G@6or0@$SmhCwt+P z9-T+zwBz5SJHzT%pCy5%>*$(^&fqOPx+9t15@NOe_S{`8^?um1?b>-_iz zVkg-dCW9pk0Xo3jpMwr?F;h8qicOKORwME-v<+t{^3 zhABQ=9qXB6XEs=bES?3hX*jk`)R*G9&*Fu<0Xg1*@n7XvsAz4|!afKWfo;#;`B@ux zVP8QIJTU(Syw-uHVsA(5Doyw8f|F@GER^8x(c28Tec8fZ-M8Fz{QWEn%%@CGy?`vBT z-JSRJE#<$Vn~>uYX?B3P55ljarJYu%cCM>3GhKI;|F+wPEuq7}n5OVA)|>rcCTrD} zT3w}D;=RCk?_c0w{OiK7KLQ9fwm{NUNmB#A*5*p_^MB_#ZJO~7c z5$}wi{0bEPjbReioiL~nn&#_jL++=70r-|`RtO3Xv69PS3BVP2}7hD^JHT42vu zzdwqFFo@H3Sy#ME2SuO?0eh!Y{ny^E@)z&7MZrSz9|rr4axo#LZ@b-$57uF5k~tOA z5LSfeYzK1{e;XPK{4FJq!%i}HlQ9*c^E~d3>d9|kSUveUj58F2a}le7y~;JLu<&&s=8P){%$OFAJhBM(Rb3%uzu2A zR~$pQ5_QA@A&LVahCwfdg`;n?2f%xKt%%|nTw9Kzes%4X2o&-}pDS?teS?gT#KzW; z^HpXRaWQ~gikwqAGLE>#dTEv}yiX<>n@8{ku#XuGfMwu|>b!c}1a~DA13}D;;}_!j z(moDZ$O!vbrGS_*0d$oD83v}nV8zt8r5n`oS`y=R|!kL|tx zt;pNG?S0<%-gos*?ftCRw72*FJydS*m(q9V_P)jur}N*#$G@_@e@!&b*WUAk_t@Sq zIC^h;Qz$s!`2M+P?9|?eU)|o`fA%xw_I^EmcW&?JIpTEwA$+vAx0veD&55gU(eomo z9awCi4<^ryfT4kz;RRA&_@b9cxf;QJ%j8sAHA1e|nYEkXvOtCg7Tw1*6I=z4UssI`gF|z5YNRozEjkd4FMt_k^35Fnw+z zl;=mh*S-txSXgpvBaF#fSL@m_4u#uvtlehJ_GbOs7Kwvy4R5Cyz#7AT-cE^QZvkj; z#;qDG>Ijevy`|T6`~;RoQEkPI!L!s@z^0JE_GWW-5rB!2F#+t28Z(G{f+HzepKZ{; zIP2|u-|-foA{BOH6|Mi`toOFp3_};!tCeE$oMXSR_OsrN0a0mG_=54VmR6KJ>LvbmME)b^>@#j|nWuTffgZ{3@5#cLEjN>k4eU<#M6 z@85FJn|pDsLdM42qKhT)y$^b?fI?v<5d`K;K;RyX(#nM-6|DFKS_i$|5dTjf^jE(+PEMfQi9xcjD%5xaS}HEO+kt z;GS5a-`@M*eb9TFIlJMh+~bdf-eE*$KF+P8Fd~x!=Bn6@f2^C+u30y^zKqTFrC_~q zTmKF=%-7#;BeG4Th3MMQe%Z34*xJ zzxAxztb@$hPZ(|$kz+hYR$sK=vK?22)dg@RnlRgpwHb?Hw%f2DL7DSUsigG@fEH6E zg3@=(4#h{B+l|FAfbD0nzY`o{9FYp97z8$zWG2vNEQa8nCO}YTill;WDFCZ|5vZxi zg|V1z#=h_4(6jd3I@XV@w5Ao2LesDcVp?$`aze)GW79F! zHx4ntA%EkiIIa8yV}T{Zbc~-#?fq!z$tM4Tg;O1s$|~ z=~yr5w3&|W93FeJ47g=F_7t-a?AcOuJ0jHQe<>SPMeJ zqNubiu}BA+s5F;xR}Khe9hB}%1EH)N=wyRbm-rWzNogt+{aDFr*8?voppjW%f!Z)Y z{+7#_VXK!@Yu1z7^5g}gfsASUAazW?znu3xAJwZ=kl*~574*v2T#p#sWr(*klYXrG zt&L`}v>2XBhK}biQTU@%`AK|OU%Au!fj2yw^iILduCAL3Qw`&tu6rEZn_B2^^V`_ zMZ(#x-ywj+!Vs;uDxqr^tpP;#`D&=?>4Li)&;6k$tN*9{xv?8V~R@w1brV?SM6)6)yr5Y2! zg3y?O1OXmYETo+vFzY}9NWz!^W<_HJWkJBl9d!>8Zia>@HUh3#>6%2+X{(F-aLwWb7$1M&TGj05DwIdVH@Rr?KyP|YA5J5h2K z4+iC;X8;pgVoS9?0MrjFh@VLPpnw|Zpa}X{IAjcC0v6&&>>>cpY0Ti_>g0My^@`=L z$+O(pjOtJOq4Po?tc@zH0d^CN2|#=>CIB^ssS0dA&pl&5g0g5brl4=y09?;+)ZiBpsB$l%cyZ?>2jRH1@lT@TB&>p`}f>;YqfAbh2`XjDJ)d8xQn9E zBVq{iHFUW^ed3y8(`m*-v2dbPsN8L5~IB7ra|Dw(-^v0@%1WCV+Q1jS1k@O=AMSRAt64 z0x)JYCV)%AjS1i-RAU0Tp@SIg`Q&H}$_qI*R_wQIKVi6IQ;vL7b4yc~So-Xr!FfYB4H4y)Zwg4(A1YQkUyqzJWc`DMOH#u!d$If5>W z0UIU;flPoix*Ya)?8DZ=zKM;y^*g&*Sb9H7-S)~4y+r-{mKUQZKlNe!zT|yy${+l5 zTS{&9vlc@;q6I@u0PM|}0N9%`0SsJWUXd0{QLjkbuGT6uv|)9s+ChNUY)b_7(!LBl z|BBRFxhf(R3@r%EI6DIWw&P1Aj;aDoK;<@>B$FnP%P^UgRK*=kXH6OgqHD1VYF1aRtE*vmixo*p z5!;Fi!A0oeg?MG+WsNsl_mA}dJn#FRnKNf5Ezl~%Wqxzcd!Fz5e)rq^eIG-TT6aml z3ay4(?}ApQLMzjemJqWo#|)()DWMgb4zbL*YqN~PAZ4V3F`rr&lD_k`N)#OxS{40m znIj2)HxfLzLi_5|6!U_tVkU_l*Y{*;*ruRr- z$wrJwds_xecJ?$5jH&}duqLEP2-;IcLbS@>qbfpdIb;<_hQuv7f;6ErHZ3N%kqjj{ zP+$`#AxZ67j_irWU_`2;y)A*g)aRjvP+8>jjhNxwfn`5uQhbp5Nx6Otwfg{9oM?=7f?i4bq*6-rKu^YKxVuxFV1F{A>LS~PopI{4Aj-DkE)h|on45k|Pf|F@ zUWVYtak+(P#Q=t#_@T$I+g~ zHwSJ=XVJY%km(?P1bqe5ztK+mP*v*uWfGC%rMZ<7T#sU%6wd`1XnFy$pg<89*F}|b zZPJYsV2I@a*>jp)qM_Dv$ia3B3aRMp=tISy(7n5xT?0pf+KuB<{NF4DJE5lVo=G z?@{dEna$;!z$NteGIYs=u!`1=NugyyJ~952jBaiYH}@bKV!AIA^x?G+z+(Db?y0xY z7Vajcb0!{=o5xvoCiSLm9CIzH&Mh+8*L~{-3i}f8u{~Pf3Nbr4hIAu`c3&n{UkI+g zRwM)|r$`7+Q&A*ja9asE#MRdlJVir_aCVC7m8qVfp)o;&?7mFUA^X4*b3}MF6tC5B|2dwTqYWYl+>PETT+@Zvq0Bl9G5!_bb&gzmChXv=hCx4zr$P~ zBI^9SwvK_VoLQh(YZWauqRs-9X`Bf`9VJL_ImE<-A|dFADK|RKWR4sxAa=q9~fvmCb?2=FkT{L&9Fbvrm+; z7??rM(+C5&Cqdj(f)2hSA?SHXg`C*WmJJhDg5>;a|5Jnea8=YaG_cWR7?!vx1o|s5 zlZ6glqpAqObg&|l6+>50e}F23pNOl55R3*yQPxe-i4>%(sOasDs@_(yphyp~UNLwA z3=0Q3V-D0OhGlLYs7;aZicTaDx)oj!P8vXKpb5a~A>$TT&9!qsKn=R0u|hiiDt*P$UG0 z;3;21Q1TQBL2@Y~agH7bv_yi7OerNmDVV+k7CG_J0bK?4Uz^ELS!MSMGccX;Y^%ESyWGnWRQAT zvNMQrpiB)3K{KOB2)e?GglGntt13dUX;P68tV5Hd^ttl&qPLC2kwF-eRLKQ)2EoZu ztRaWiqm#OAuSAR?Q0d<_Jwr~Zr7Iu)Meay@>-j9ZnxWhy?X_?ob(;_@u_+RQnQ=uz zFp?9rSqR2$iiF@eBt;}yBkdXn9+e;n>Rum6I;9AZQlqefI!Fl0o+2R_C@WGTDK!_C z?36MN45%R?m>f_f1oey}A(~Pnsv-pQ;EIIc99nXOYFruHBYN9N93yRpBz4UqjQp;a zVU1&??WNWtXdzT{@ICbvOfm1cQjFR4X6j_>C&zs1v7^nlR7D5|{fd;Z&7MmI`>wpC zSI}6EkYtq_7BsY35VDPpm0;zFWI^yD0Oq>La;*o&T?=#JV$ zO|DgwO4xMQs;Nd%PYTIxx}(T~9E2oHiu%+QCBAfJqRY-wOQe=V0MI>fZ-Q7PvUZ8) z2jkXUbxTRQnfZ>4Oq|Wkgk-IRl0!5zf0R*O#dHAm;fJ)DIUXax)Cw>l%?tvQH9~2V zN>$~lD4x+jo~a@op`xID&m;n-MK@h02dgDu7R&7A$Pn&PW;^xmiGv%^JdIUHvZ9g(A(4@7 z@bM~PdXfe|O@nrfL~I6AC%p;r^-uZrmYo8D=BvR87=&O2#UHMOXvFXY=9W62IB?*Tm>~KDrYC-fXEoHi z@NB~Kc|1Ai`PlLn*yoq}_F3Rge%4EV*94R zSGhG3``#x80uMnIpMgo}V8`T&6le-`8gzQa-~-FC`EL2(L#zbhQ)&)QMFO*!brgY_ z(jAvRl&Y_U8cr1e3-0|IK1|`WIhR^zn^qv(1le~NY}#o1Y4^WkvQZZb=?sw2=FxB z^B@3sTf|0E2nHdFgrH-fNQv}3;;>}*Jd6VoH6#QB0YyTvfT2i;_B;kuMF{#SiiE(l z#7RO3xSYh^bk=q$8ILq!HDXSIC{(5`ZUxLkhDNN0mQM8 zS22^$jpfdBQ*}c}8;neyjl2NQjqx~)#bJB^bO3Z1bl5qeK|3(6#4P<9|L;qkxo;q})llAn= zguxg`j&};~>x{oM473B*NHXOysn>lJYtF_B5~CWuJ&;|AlkmO@Rzz_~cE>N>B&sk{ zB__vwAWH`lXA=z>#jFhvgSrTVc}ulWFDA!GheZOBMMZI$8BPen7+sMN6c0h$gbZ#QQ6yw=+g3#+rFR{l zURSFj2ew!h0X*C*MxX6US`{qWrOY@mstyP_yp=RM&R8c&+9wF@r}G$$iZ2O4I8#p* zOC+xCb94mRry`fYp#e3uU!65kI-*@4At=f^MJoi%3)(CMHzp|(g5$gukvQtFgpZD+ z`wxV6-PZ=&}Qru?3mc2i==ClxK^YV{1ZUj)YSrdeUJp6SYK}hEE9U zKSe@N|0z-;<-Y`$>}gu#K)o6gf`&zr5DbD9kvMAvx`IFoN!q8&3y&V0bda1wTaZ5z z%XZ@$gD5Q9=IDPst#2eUGh z+@dKagkyyJvSxUGDgv4+u7c)8-H@D(bw@B%x}#)pPyf_aVABE|Og;^XtkMrIWQQO* zyBgga_(YwI)G3zXq_R=T7Rf~BE;X~NainfmOHfuUokK)$4C5yat`H=!A|Xg%MM|V} zPz22`9mavVYDfq&vmzmw)>kA%OGk~W2yyR3Q2Dlmkm7_c8 zLRRAE)EN@B(%{GmL6Z{gKg2urJuInC-G#n1v!hrWhc)?}nm&dpk|*kiWxn3*AxS7Q z7xF0?OpM6vpq>ScK9C|I7)UE3aW=)2I!s4B0r;3yqe8M5xlnJYo)U@r2}jf$2ezvr zAsEmr5`x)RMM5;{CsaiUsxd`EY)Q!k{irQY3{cM&rWBdVT~d~Ry+uVzBo4Pb;?OuytA>Q22c$>{N`oRH8i$Rl zA_Vh(iiDumB1gD7g`_w!hk9t(RFPzt)tV;`+m4@Dx zPek>kcqKZhI`EV;3Nyi}pc(gPaLQtp_3Bd%LA|(=sBd}yw0mgy7>X#6+Yz8SH zDk58}o)C@0X;l$|Wf?_6FuEW|LkS_pdRRDmSgJ^}%j(J#ho>DiLmcYblAa?B0aD?} z-2AEp^`H7L1Szga2BtRSvt5G4CPgNvDm)XGeh$u=N4#1KfhsJ?6 zwIT#9wIU(N){2B^9L7~e2u5v+gaDBv^twoj^(f^mq3c~%G*28(Vu~eieK^5MBYQQ+ z-i5taXOe{I`Ot83Qcb2rFDV2SnIaNr9QJ{`bQ}^O4!s%-A?V5}5`uO`krIi+2rSug zXdKwDehEQOPpt?+wpJuWroh9|w zz-034Y~_beT!opcT?vBTU5R@@e7h23Bp8uskKz2C2WHha?_0WQRyoEzJH(o;)&FE^ zM(bW`p`s_*E!P`zx(hKkA$F8tNm-)75NeNRX>V6;B_*}RJoYoWRgH*A96SzGy74;B z$fcV|YH|MM^Q7|5!~o6_({>ra?w+kelX$YV;q85Z87>a~suUD#oY&|U5nZ`KKr&ffZ%}^u+MM;qm%{0TR zBE-EuOUIFEgkXOTO+rvY6-hQF8IA$=WF?;a#}RMxIdzlvhD|4h%F?%4mX!j`Et|L~ zA;tzD)O)J7DKcw^$-O#(JV}BssUjivN@8+a^mGXkA~T9C4H!x3UDf?hdioC_;x8ii z7?p#hBc1?iKD8hOJv~K2vKJCH;(f4WN4#;MM6C!xL!(GY_ClgYyjS&wpmIe>NkB)6J0?dg_v0y478|Nz0*D9(dqR?oEXA9_XjwkoIEZCboal!5fj5ta1tEV zkXhMI*B4umTWg1^5x^4RK?!!tD-wb!Aw@#a0tnhH1cFJ5gg~f65s4#W$4WF3Es%7? z62RX^u_0@o2)d6{ME6nk#s$eD9eCfPNC>)iibxzy>b7b6fg%r*jtK&& zO{)zdU`CM;G?U7x63Jqduw-X3sT%9 zzYn1{alp;$+zcRxtnJ-3tK=-F9o^H86$IUAAQ!oqQ-O+&b39eqEX42>S98ob4-sE6 z0B47BN5(+uMy8xr6I9}2O8-t&YoOeK+6dj+WZ+ZKfF^Do<-yk6_Ao?>TRw5qXU!&m z%|I`Wj6ezLxndkjG*L6)*QJ!e_L@yT_&HKDKr)QgaRf9lJoQ0DxbfZ2($GVv*{m+< zLA9DKQ`a#fb)^IK7|RWJxVz5>IK1-YQ9tIAuPJAF`taYiJS7mg?D7P8gMBDKkEI@a z!{4ktL7L_d;)|B3Q&_|fR-!IP_C|^NHhM93iCP7v)LgJAsd@xU)UV$reg)~5DN(h@k8YCF%{hk1D4`Wt|s``$I5JD!u$$sEsEvkizl7W;xBk zGJ&n7^LE0-FieytzKqOUx@jxkkmwL?+DeJ-GDA<0)h_B!8&@slHe#hGm}M{}J5v`7 z?!>;iwBX9|r=xt#f)66NLT}L>PF?(Z3C;kZEx7V48Exs#=S_3pW5kaBrb=Xun*}T-7 z^Ey9!3`3FJO%5@yO)_N2W_9;IxXYuL@$}zUvnxTP%yxaOU-Wzxjvix zjT5DJ1$N7oR(neC`h6;?vH%(gN2Dj$3Oh>9+=Kf`85gCW%$o@Dw@R{ZLz>+Ej3&Y; zlv1yO<94#<`XK!oYy_l7@B#Bq8k3AdB-biQu7`zdNgGw0SGsX5qh^}U?P^m<_R;W| z5y{e%eg`IR>1qFX4Se0I`eF=oRrGx0W$J^dVNysSW8tAWaE=a*d`{s414v@fh-Z@!cEmpg@1Bx3`j;q@Eco|&QS+dW zkzF_Ot~odrA%Bp7LII@&)KYW4?M=U_VXtIxY_9OHWN^%*kiiALzLy+!U#2lg` zA(-`3L{fTJi7iAiRzhqGQ4oNk0WoTeEDS3Wg1I_HN@T=uLB$^N8wVQIkPsAiMM5xT zph$?0`1@2v2&N7d39;o25*8J4Km-ZqW`x+;QxMh#RB^i+8709qfRLo1LFQm}K@WF@ z?6uk%SHnW=(rE98-Ya@}F40DHmQpgclK(sFEO?io{epI+`+-CZ?q%bmm$O)2C#H5x0uaqO) z1=C9)1_Dn)67$KyNBe&Tfs5X~gO@&1`oT(JV7BnV>kl%gd!d7%W4irQ==>ufqA;5r zPGKcm6%4!ZYZAO1?0E?K_Cs^d?}OQUwt5ixJ-~T?4Vr5mqN7xtRGKW>bVtFD&qz__}HcP3KQmE6NXD#BUD~;8x)u~#~ z!|7QdGXta*CF`!kM{}x$9dyrTVSEktZ*=KTU1nK32Xjq_w=1dfe?Hj8km_swd+u;#D;>tVFYJktO@SxikM;^yFq@$m?XMN?t4ep= zxNmUm9`WKa>?7B&ZR}_)#pMP=oB4I%djpBD(6x7vOQcR<&)|ytg^xQV5P>~Iz}nFs zOx{7GrOb2W^gS@KNvxOd*i(yhrJy@_E5nOKkiZWR3_LKf`F3c*1hdkn1i@0VVy=qY zJSo}|l^D@Z0HtJ2T19Z}j9bD4HN#IVLrd~*-V{myP?L7w{L(dP3|5{Zlq7uKYr#wc722UC*aG{0TejB5EO@t zF+kX?CPaFv*eS6UXdv-MuA{F;&>ell*sEQwWm%X=Nn z8dF1?#fZiu9G-)CMZmT4Ga~HFQ;nz+L{G1IM06f_7(;McF5P$v-G#sgeQ|i9J@%|_2AfA26V?}x}91m4TND@ zs=gVjcbVkpt&2Dpmp2`nyXs2sI=X+-)4v~en%={K)mDi|tT{6k30Nhe2Z}6--l%hn zf{Gi|z=(Z4fiXS;Vvf`HL1qJ({#@7o4~tD!dSk=ku}? z%mp7GngwJ4c1p*p#Kdl}sbAZCqQGlo;0ysW1{TBcp%seB5mZ2i83rlcSj&LUlwo!! z8es?}(wf3@4eH#n2(T3PHhv^q@J>An15-=zQsd%_noNBis%WEODltBoI1RLPBPZvC zpCM7`PkE5#Mdx?aIf|}{8<1UrGJ+yy>5zp;Or;BvR?fOn00%ROL*~L=yAWBD16A)i z6rwtte?b1_a3}Zv zyTKcu9_-ye)cf?gz56p48leg`^C8<2q5~LD960?N)P@I6ZBJZfYWAa`v?>>3IN8ay~_xt4?m-h$cy;0u(B<}%vZ<6=T^1emh!}9*HygwrE z+vL4P-XE9u9rFI9yhr5yX?g#vygw)Jt@8diym87$^5kz}j%abifz!VRI^0IZsb3-@ z+AiYP5Bb&0ukZ2e+V0{PFRX2QkY7LL*RT1tmtP0? z^&r1q&(`^Veo0$@H@}urjpeOv6~8{muNHo>U(wdXuMhL<1N<7`*FW>?{rEyoyM3Uj zc%P>RF}eiLCFqt}pT)!0=xYSOz(bwy5?qX^rOrZ*gp&Z&Sw`?@qqB|RJj5}LeT!fY zBIrEO&mN1{H2{}@evM!uf(5xp)p-v9SMdLbI==<***dp_TRQL;=oJ41fX0q{6~Jr3 z)1`1xer-_UIPX`W)(Bp3L!JJM0ba~`&VB{}`!wyJ=SJ{~-$zOr7iSiqLGT6yMt=Oj zgssI6rS%TH+Etxqzj%{Xe1``oc8Et&P)`G1*V`uU$H!@SmeYeW*kxoi5-?ky4;mx5 z2rpWeACDs;+y99`74HD>s=8(U7=W&w4jt>%V$0(EZ8gD90qEKa0_N4vfXdHf!4iJs z$CKX^i;qJXtdM*Zo1XqE2R^!E)82Q=!+!c>o#n{$pS^s4_$B&Dlu#C=? z;KlF;3EqWQyKU^i?T^wflf>$SXY%96yYck{Dum!68?gka6+ujsl@EY=b|7C7On`cD zXQ;&?z5Hyxz5%BfQ{^UA*N@Wx_Rz@vc$g+XBuL;z9;XSufmc7M{L*>sE4B(hIMehe zD(t~?5kL4laQoH8SFXcH1)h%|gBLv^mtQ9MKHQ~`_Yu4aA!0xGQGz8VWWOS)fCcJ2 zL$Dg)HH#rHdgIp-c{H~cu6W*z zFc5qVG+i&6{P&1$pCA7WLmiCbD)7swx){{67{MWE0QJ=2l@o7wz{eWcXI1(>IvB<4 zLQwgo%jdJIgP%)%{XERJ*j(|0k@7_<+<>Q@AN<*IIBeqZEXLuJNJ|>u=#0bXQC9XM zHRZ>THHhmQSf9NUavx_I4fuc4O>rb*FY^+-vs#e~vsq2jDDRk`{cC`00O&w5&zJm! zt#diF-vc1OhXLroaY!_aJ;4i9-Ll>c6_&4sq^IKbHd_Z_y^%@Ccrf8R8#d*~4>I)` zDqM!wJ!%r!wk_;k*WKM>wRd=%db_*2MWrbu3eBNdtbM&I^{(sKAcms7v6w2ghI(RD ziiRTXE!|eArzaeZ1>3vUhobGFu9!7%-q}6T=CiNu>F%1>7KvD$!PtgK*n8?C@A^gwu_c~#$Q^jo9do!<7Y_E>wUqy0VMsJEjXF=CAN_JqCNk#IB= z>yBP|wfBbhSew^kn2hU|@N6j$R#SI(M>y1FwY2v%hoUXv78vbp3P&wQSuh-pc1Nu% zB__jt&EZI_9dT;Sv#$1{RjKqh`z!n{<>9J@t>u+fm7(g2icob^xO_p&e1B!De|}|o zQ){{3TRzV}&u?`j$)m=_>%tqnXIN)=8MM~!-mVrW^P$+>GY#9Fp~yMsgktA-y|j@r z-Q61tcDDwjp{{jd>%BKx?>*Lf>p5QQoyS@&y`2%O-K6!r?8H6m{N`|1%iJ?z76-^! zR~~E8MI?XPoJ7r8=Z8BZu?;X6i}r@C)=)=J*g7vALFbDf!7$=0g6N8L_jngA^7hQ@ zK}vw}cX)duq2{m}YVV4L*C8_dXv7D;&O`2E^ah)HTU*0X+tOLya_cPLJfFqq^m%Vz zr`B6Lxe1_9%~xib`Fo1 zLUH*Dt$B#4e$OQyYxop&JnKD{&x>9tsCBw!{p%`^)rh+79#HFS%X$QKgmJXmW5v-Q zT@E^m{?q`d_d?71IVk&r?`-f`Wy{chf(~DX#v8N-U67v>Uu{|5H6AO{gw_&tVjc8B zN6^chbt&A5VAKaXx*nZJ(BX}iwFR{7L-6-9(A(fXX#J-wtBv?x^oT(lzh_y$A$|-E zFwUGBeggi3#(#ol2Q>0C%Niy9IlAXm{{`CRwFvJY;UDQq9_uR5QOtx5fLgV1>)jrS z+gGDGew_#PJbpE=wmc)~VbmZ2%Fqf;zs~a1oM;_Yf1>4yppy|f$?}vTE~ihiJfm-c zxzjAq@M)lPEzkH|Yxcz1mdEF}toVGGL)>}K!KBAhxEZjDM(fZmLHvy@w~FGctXY$Y zyRxg`4&rS58rTbhhGDlG=6k@*`z?<*4mT2TBWZaWZ^8G!L+#7Zy9fO21Rvi7-ep;n zyJ6=+*nJ56On}daEl(VA*7)C`{|)*G)E@z#Keep-pF!=hW^3O zrwu&{Ws&hyX6PFYJcq1PH3GxYt24jB4jL+>#3bB2x@ zYWs_7Lw_^GudP3Em4;_Jq|~-%>wArTx4xYZ?EDkIRP)oQp|(4=y^$+af80>FIorRQ z8T_;J+oAlk^W&k++5EjiUx|TNV&Ih+cqIm2iGf#Q;FTD7B?exJfmdSSl^A%b#lQ(~ zEnBmsVey&^BV~;zU3krfkvXd`T)lV=6y$B4P!kMZv9ckSJ&m^1A6Rd#iu#6)?)+fT zzn~ctZ2`-vRME`2vbyF#d3%e|SyShB%ZkqtJJiWczFF}t#j1SSO201{Z0_p|HMOtz zm*cyqJ=olV8FQHu$DDad)Pn2!`hu9N#!NkCvSS;9>wQ+75=!dH@M{o9LCkwLUl(j{ zyDqq{7gN;mxMf2}3zU1Cf=%sREtp@8QzFUO!F&#*J$EkCoWS5@Z14p!9oZe^ z+;*_FD-fuVb$|epX0C?6F4v8$`OEm6k@&0`jb=|}#`mXVZX`9cCD(jwd^lrkVD?O2 zWfC4qNqHEJi2ocf(ma+g+iXoctgbd3Dy$PGN98f-3TfMnF1kZU`7-5oCM#J%+iOgl z?4;Q& z_YFbJlCO)k>B0p{?ywczp6yI!M|X3mBV3JzmtZTV{$qjF%w1Y;t>}2R>8keb9`UYn zT{s4t>f|0{2u04{(Y+4S{Ic}2q&!f+teJjJqz#aBe>oV2Azj3&YhKv^wdS@^G#HD9 z+G9PaHUdzq4t0<>1ds}W${-4SdyGzY(#^d`*iCVC@(4QGDM3#i2^T|=2$nHnkLXL^+sAZQ$H^)SH~w+550m|INI6Xg;i~-Gxv)+x+0;V zsg$s~Y_BNO(ReYlx)_i(O009nn%k%|kkLUA&ZrM()LWxrmvdV(ni6z6G*Zm|#HC#~ zpwkds9q#Gv3{!clCusDU^4560dO#SH5KCf^1_=tDHGcn6Mee7`2^y{EGT;lcBPt2M; zX7bq4V|~8yqT!;*@Nwg>WxuPKj{AI*uk%e8$B&&ZoA8V;iWg13seZ&W>KUIkinlL* zjBmVn+A~rdFB&iM`pV$F*LbU*%iBBY@r}M|xX3qtp?9)q!as4;#8Jbqnk<pGJv1UwtuB#K59n z^=G=*h=qURod(CxRD2^Ayr@57aP}F- z)Zlkxt(*GO2LIXMG6*1$pAWrW^?e2}f{>P&holzupD?(7p5j}ug-QG~H~;}f;t7K< zexqW_A<+I`4W2akM{iR78lURl2*C*2j~IM1(v_I<3$(vWux0rw6n_Qjb_z6XydG=! zPF(KI#@7e0g=2zI{w`JXA2Zlqw;eZ_@*{Nb&B#Y|&tB(U z?856@_!hxPZ?hhJr@{96@FNCK8~uF-Q%;5ceHQtY;iJ3?@%xYuB|e%I`;qU6DZfJf z9~f+PDqezoMtz^b6_EKLZZ!A|h@U&peCmDQmgX-IO z?i$67M*n_;;|8~)oJ;r(J`v@b*!zU~_Z@>94Sx7?)u$X6<7eYq#gR$HbKk9a!r=cg zxa=pY{~?rXx?gYbM^UbcDGx^XmR+T|@u!M^XYjbeBUh`w^)uC9dyV26gRcoHrd%1_ zdrhO_`kyQQqQR6mqyBXvqyG!Vj~GljH0s}rdWHN^9*uZzv*MaRDgLp+5ra3jsQ&1Q zDp%tTE1od;QGRUd=3s7%K{LEMU3xh{06%VzkKIFZGpUQT{!!?Sh4USx> z_{3|Cdyt<03Sa~WfiIaQG+S7 zNc;tZDZ5Dgb%QCxNc^B}-{7AZOqoXN|H)v=HWD9=dYb-G#*z4q22<9N_$-4d^GIA{ zFl8T!8w{olByp?3l!YYTVB0tNW`ijkN&Qb5Oc_by|FG>F{7r)?GfDlu22*yD_}2zg zhLZRhgDFc%d_tFok200Srx{GyO5%kEQ^u0`VuLAbNqn`zl({6n-eAgJ5+@9%3?}i% z45ln5@xL2PnM~rH22(bZ`1=M^Mw58TV9II||J7j1Y!aVS@?XC*IAZX}(9WLns<>s#7mgWh%N0Ihuq}5u>v}bB%O4iOk5agA%Qs#?4EJm~ zHWXjMww&33!L~fvmkqY%*(QkLo-H4TVlUX1U*oPs>f7>l7Z_}r@PvpxWw537N3USW zgF*?P)_d{7pKQx-NHMkK_iY#cg9~#Hkt(VAx4ZBX7an$D?$~nLKeIR+_qyPCP0buXN#SUHB#!zRiWXqX|;W z@IrrMF8zmG_%AM8dUQ5FrwY!vccu&b)I3s2f9JdGuXW+;T=;!1{9zaVlnZZDoc3?G z3qS6{lP>%l7ygqAKkdRtl}LP~{X5=;{~N)61kd;Jpvz{VTV|mvW<8DvT`mjVEDK#H z3*8wDT@-5)&rk6D6wlA_{2b4dcv5(#@catTukri_&mZuhnYaFo=P5k<@a)I)7d+@L z$}Z6V2l`h$&)_+LM=}>uzX(sU=7Jii#_eY}J|4sK3p~HXvj@+&@H~L$w|MZ2JD+KQ z-{ZkA{(pEqji->q?6nSSArqF@E?qjAPj{KN)rB0$@fpT|+#(mA^a|(@TCCy{xr9 zRnF9yvd9O;?A1LeoDc1@!i-cTyZK8(JC`1}SQ? zAk@-c!Q;WzBdsFK_oj*yNHmL8+J1a#$}chv}kMApD>PMFy{Q zY~l`b6L($1g~31*;@IrKU1)Z!2MQ@JCAF)V+EoRb+QOmVPZsem%p9pa`h-WHn@26c z9X%+@xur8e0kfdr-RP|e1_BM0LG2Euvw24kO-p@{V?%@>-Ew1aeHoq4u26hIQ=9l4 zwFyKhU4m#3fuW-=!85Hp7U!=!3My-thakPTtf9J~>aI{j0M&xj0Edw;C{o7us{@B7 zK!~QJBbb)ZaX9T3Bdk>MM-iMWpa#e8iL6|f&giZT`*&^u{gv)xNFWcz2+K6ZQkGri zC13G|n!EhW(9WZz!LZ%BqwGr?y#R^}^EA?w2wl(-?&+z6Ny>*V2$H1*kx+ZIZW+|r z_VAZ(D_@MfQw}W??+|%~^k7({9@4&g{tE@`5ps9sMJ0D_2(@RK7%*w7$V`eX)D)>e zid5xG5s}jMR|hfa5HZX_2ssjphU<`o4FM*vNwS4V8<9*#!c_$8pqHCcFZIyAtkQO2 zOBg->?hS!@$(NdugI(P%$QyxR{rZ45oLWXrA*RVATEpba)Q)N!TG8gUK>7wt%AdvP*p)NtIFeo;SZH=ANpqFmBX@RiDjd;U zk*5lPT5iKuk<+qC3||;TZpR=%#`A%|`qgWKm#o1tJZw}my)FBMD90A6=8i*fxxm5Y zf;?P8Etks$`MI2FF!9{kUckO^0dc-`O><{+hn?`P5%k!5qVs!VErErt-O(Gc1FjCq z5onDtm73SDXIdeN8tPCoI3R;;c{g@7YzW3e>*|(Yi@i*)5tco*@a(eId`F_q&y#3S z%T2WT`4g?0{h2&iCz~LpbF4G_ zg*E=o;;+2$#^;rKI%6HC4K9|+0~ERXHNmA3yWT9Ad0Ly(l?~-+EUP)YT7}cms zq&p&NW>_eO5Fq?6#*+?LVCAw)JL(%Q4hB#`1sdw5fhfR$&X?YmqnDL87#K}SFqRc& zFeGO!=6q(JBtWtW@5>95_ve|{P>_(fX3lV{3pKQPG91Dp!*zGzv2)#u7gJbfT(-!Q zSatwc7G?nRR@kOuvrCoM2s2E;IcE1zbN)RR+R6`S*(lR#Pap1!_P(8c6I~$4nL8-Y z*8wWe=>TQ-W^8YqLObcPt-DW`=WvXyu95ASW0zm9TdqO7b7=H;NR+Jpi60rEpjYoQ zeSmy5)w3$yOzwhd>YwXMs+4ck$e41GX0ul6CdbhBh0RP=?L9#`hT6IS751EEeCIRg*wBAv7BnyQNQBUk-t@&kP+TFGM+Rn z*3z33oUFFTH&dXq1tw=S=7^i6wxLXxGc8mmXbWy=!!ojS87k6||M&_t6*;R*IaBy+ zf|w6k<40$>o~v{+!pz^`I{D0*1GO8{teBb6RCdhFWGoZr+%`EUTGCPHPArEjIUc>R zm0niMUjE`~*$;bkpPj#6*hDyJhs8PQC`6|7kKD64%~vvyV82}Z_Al|ZVV$?E)7IeS z5n`DMH6eCaseWfIe`K0k=Gm7*rysI*n6EUwe$6SA+mMxlobhkI;csq}osRviic%Mi z?RC#1CY`eox6^dsn7O)(S8H!T){a+VhDn03K!r(gtd-mVBx)5#cpyu$JWCwL7XQg3 zY_Lj%4LVTM2AUuogjsE8DSJ`HogcMedTmH$DbkV1*=;UQ`FgwBug5J>xB^Nd&+nAH zpn(}84gwj_Er)Rm^H5GBC=wjHMkUf#4Go;p#1B!(wE@PZ-x--N@k*M9DoVuYww9pA z^dSUX@)xeZ|IC{ldS}Ei3xy|X0t$<#tS&R-?D@_AXZMh@>eat{(&q_M?$lWUGu%|b zylp68-ueodx0ZtCjdjwCzszR_ozH75owIr=$@TIWmqbukQ_Y=pL2T5<4c~I{M>b`*qyF@I0cmVI?vqYJi4SloNd!jh$Hwm?~FF zELNBf60@rtE(zlQ^67Ov*Q}SDf83J33(Za}ZhgeI)~v%k?4~Oc4X>{|NHPQhYl1u| zpAs$F70R@F5M_RDR?8Qf8oMckJ)F(qV5k+7d?r&I+^O|BY*ZPyI>N27V6+=oOQT4l zJTjYQnuT;pg%ERAVdI`PF&1rK*Y?~TD3H06tUjMiYy%N9<7yO;|a6#pq{J}o_P}c^rSY^bV z3NW>+IzxT9RSXx*7}>A_yQ9I4`HdPMhdeBR=BGa<$=Oor6>;+?kYujprBSMWE=LOj;+eshK(bxUf6Be zhV_?YDQ14xGdK&l7|(F-VWw-ETZ|#N`2rVXGW}vK#?O`S0xHJJ`pdEyrw;|q zF2AhJ=59AE9NswDc8GK@lsep*UH#=u`Cz!GITQ)EFb5RcDNvcBWq8FU+TnHK zXkDM_?iSXO^LZlIW_Uy4GwhSx(Av=rL6d@-#>s3b*u`yg_`{S@b!=w ziUnmwy}kV)M(2mSThX(>XvvabxUV@JfgP^f`*KvOa#X5wRQ$Qt{0nkS&CfM8KaZ)( z9E%HbEiTBlxL{$f#a!lbYqdMf6}diE! z`Knwds&jqx`*P#Z@5^PwU!G4b*H{1iJm&JGrhk4uySY*6Ur><~et$)7BKs>Un5ms- z)fP^bldq2zsB8=OwYSO=( JywMTA{}1@W;O+nb literal 0 HcmV?d00001 diff --git a/doc/examples/at__object_t_key_type.cpp b/doc/examples/at__object_t_key_type.cpp index 83646f15..a66bd9a3 100644 --- a/doc/examples/at__object_t_key_type.cpp +++ b/doc/examples/at__object_t_key_type.cpp @@ -26,8 +26,8 @@ int main() { object.at("the fast") = "il rapido"; } - catch (std::out_of_range& e) + catch (json::out_of_range& e) { - std::cout << "out of range: " << e.what() << '\n'; + std::cout << e.what() << '\n'; } } diff --git a/doc/examples/at__object_t_key_type.link b/doc/examples/at__object_t_key_type.link index 4a050ecf..1cb99d46 100644 --- a/doc/examples/at__object_t_key_type.link +++ b/doc/examples/at__object_t_key_type.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/at__object_t_key_type.output b/doc/examples/at__object_t_key_type.output index 79cff2d7..654b9eb6 100644 --- a/doc/examples/at__object_t_key_type.output +++ b/doc/examples/at__object_t_key_type.output @@ -1,3 +1,3 @@ "il brutto" {"the bad":"il cattivo","the good":"il buono","the ugly":"il brutto"} -out of range: key 'the fast' not found +[json.exception.out_of_range.403] key 'the fast' not found diff --git a/doc/examples/at__object_t_key_type.test b/doc/examples/at__object_t_key_type.test new file mode 100644 index 00000000..654b9eb6 --- /dev/null +++ b/doc/examples/at__object_t_key_type.test @@ -0,0 +1,3 @@ +"il brutto" +{"the bad":"il cattivo","the good":"il buono","the ugly":"il brutto"} +[json.exception.out_of_range.403] key 'the fast' not found diff --git a/doc/examples/at__object_t_key_type_const.cpp b/doc/examples/at__object_t_key_type_const.cpp index 0e8d9d72..99eb6f01 100644 --- a/doc/examples/at__object_t_key_type_const.cpp +++ b/doc/examples/at__object_t_key_type_const.cpp @@ -20,7 +20,7 @@ int main() { std::cout << object.at("the fast") << '\n'; } - catch (std::out_of_range) + catch (json::out_of_range) { std::cout << "out of range" << '\n'; } diff --git a/doc/examples/at__object_t_key_type_const.link b/doc/examples/at__object_t_key_type_const.link index 1ad9c07d..a07dbd59 100644 --- a/doc/examples/at__object_t_key_type_const.link +++ b/doc/examples/at__object_t_key_type_const.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/at__size_type.cpp b/doc/examples/at__size_type.cpp index e31d61d3..07e363ab 100644 --- a/doc/examples/at__size_type.cpp +++ b/doc/examples/at__size_type.cpp @@ -21,8 +21,8 @@ int main() { array.at(5) = "sixth"; } - catch (std::out_of_range& e) + catch (json::out_of_range& e) { - std::cout << "out of range: " << e.what() << '\n'; + std::cout << e.what() << '\n'; } } diff --git a/doc/examples/at__size_type.link b/doc/examples/at__size_type.link index 00e42d30..78ec0ca0 100644 --- a/doc/examples/at__size_type.link +++ b/doc/examples/at__size_type.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/at__size_type.output b/doc/examples/at__size_type.output index d1f68bdb..4f0c9e54 100644 --- a/doc/examples/at__size_type.output +++ b/doc/examples/at__size_type.output @@ -1,3 +1,3 @@ "third" ["first","second","third","fourth"] -out of range: array index 5 is out of range +[json.exception.out_of_range.401] array index 5 is out of range diff --git a/doc/examples/at__size_type_const.cpp b/doc/examples/at__size_type_const.cpp index a8a43ed8..c29db4fb 100644 --- a/doc/examples/at__size_type_const.cpp +++ b/doc/examples/at__size_type_const.cpp @@ -15,7 +15,7 @@ int main() { std::cout << array.at(5) << '\n'; } - catch (std::out_of_range) + catch (json::out_of_range) { std::cout << "out of range" << '\n'; } diff --git a/doc/examples/at__size_type_const.link b/doc/examples/at__size_type_const.link index 0fefb628..c5f66fe0 100644 --- a/doc/examples/at__size_type_const.link +++ b/doc/examples/at__size_type_const.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/get_ref b/doc/examples/get_ref new file mode 100755 index 0000000000000000000000000000000000000000..d2dda9ba81796fd46a0b3d8cb861339285e07f15 GIT binary patch literal 38132 zcmeHw3wT^tb?%WLi3kZJr{v-`AxzSQI6$3|{E`zw%Jjd}&Gq4fF^cqS7=G5Tg74 zYwa^{C%(oP++Ll}=d8WWwI3S)!>^us=|amYS!h{_1(s!%;SFA3S;@pz z2wB!2;uHE}#3S)&eLla&w=2IF4HM!y z{L+jy5Q#Ljw{>L{n9r}SM)|$X~9zo z>AFWEZ7q?my{+}_Es>6R=V|%*?8uM#{fGVI3LgQCM0U4lTA@(BGUTUx9D96-=kvQ# zljIwTv_xM{z8!|&l*vOpn;+|Yok=5zL>i;P$OS8AibQJhxII<&t(%)VkWT5+R1`ENdyiQa}>Hs`|PscTu^=gEn0(ox3moLj=_xxkLr+a$PDc2WCd=27)crV6#Lw8r_4K2;} zH#F{TX#~#aa=et|^|jWq8=fxj7{2_Y_g(kBzBoh?n`>Qs5i(z7y~)P``7A?ynU8iX z`(ZiM1Jk7cFIs`xmt@LD`3-M=^YZfL5TdzWP=DuQxdLIseRSU|s|B8kpC>yawhqFt34m4a{p`UIX(QnAgDBYhb(W{Mg=q zY8V{G!!MH0c=+ih0Cp&8F0db(`n(|} zPAD;ScvuKPN?<=KX(0jaW=R>>j6sHHRcNjWX zy1*MY>->B)zI{2EFG6(W&z=OYaN6z{0(^V~Ou)sLGX($$x)CCOK1HFkgOj zycxuK(mlJUyaSgan+x3Sp0b@|-qY6#o1j)NQ@>j8!xczz7G2}rgFt_depo&?YJBnh z^7o&@&IK}!RzkzJ14Sm6QrAx*R@={&h`^E3zK46KOZo<0hh!&JhHv1&E#N3r2a}8r z9GU2yUf`rImPDsW;zB1?Dv94^;=5UNDj7lyd#>%)Ov-R`Q;FXljo3(f#j$?9_A>4s=`N+Z30;)ytZ-_^8#a2$z@v1Ac`48F++9^57c}+yU>gm zUz8!wA?XK6ABstsV){@FtiZlQ-i0itUyBJ!F)}<+0!OLsq2xP`%)zVNewWI$CpWB2xyT7iKF`)Zy&7=Cu;+)zm%DeT`M)xG(% zG$g52Uj+F2W7zhyQk8giYH@;pu zy$ULq&I*VjfMMW-F`?k?9b^AT-^a^w;BqX$Hi&%)!`AV&XrN_Q>pfmYlaOog_MaMK zJ2}$BY-fah#CAud+x1G1LE;GeQQ*KL^c0wXB|&IZyVnWt%aX|?RF!6>9MTje3}Z-j z581sZAa?gK!oz?^0Wls&qz}6WNWpgRco22Qc*N^q5|#P`!uA74UH75xw0X4cc%5$4 z7q9bz*TG7;IxyUnNbJaziLoX_Zd?YK874g(ksH^6X z-TPn}vJPfUD$Qzbsoi@Fbp~{{qaWaPXxX<)nxMQp=H$B##Q{9Y?y(cCMN9MyoVk!7aAC z+P?k;`@y+I(1doafm>1H$w-S0h~+Jbt%xOz+YG=BBi-8S_8Ne(3!p&Lz;p*i6#H za>r}$sgQOU^ndjE6Bk(a_-*iKkcTdSxH=dX4jc|z1&*2%-qo6X(z_mgOznQ}@sMxy zYaa7{_`=+r?+*lRcX2c3tL$RZyxr|+hjHJYz!K)$lWh5PUgp?@teB%6WCjE84=9cZ zP|W{g{ZJ8(YjyyRadrUB>sEM|<>W3bAI-OK*6Dr@S!eBr?&rMMY6j!(6^tjAhf0YJ znLTmAVlg2Yhq4NwiJ{M%(dBBOh>9KM?)LrZ+B?L%O26zz$D4=95l&X}`(4EO}_F@$LZ@K)GP$aXtRZKnfz-h!5` zEdv!OPBYtKHXIK>i7HyX`<`4F=%o+hhM#mE1ivYJe9^;9LPYOai4%Tu`g^FK2>QSc z)LUwx0}+_Lq~14QVCt<(f*x1hy&P2TW$Oiqp1h5y^7IG6l?xG>Z=BR7vc=Mk-QxBx zeu=i6)d2kcFW=3$4o+?AgdxQ%gxDW_KSpuAqQl>A%uid zNd*E%QT~tN^ndy=x?{Hde|~+o{>4jiQWp9-`VTS4FPI~9JLE6Zw50?((}N^PN}c9` zaheAfB5(z${2b>h{}iR9JP=PAjtX!_PBP^xm7i1~U=-zl1oP_jKf-ck%l|D})^z`| zQ39mmz-9ke0R&Um0aDd!+76|xQxXB_|B4Af|5uE_^nXT({wFCV<$-w0P;3JR`2Hn0UnTmI>s(MJ9u@78w5TCI0QHh zI4pwI995~tSoTdK6($(WPmt#*;~t^8bw`;Fc^E}`_Fn~gTo{sUd0fsZzCN)Ka%#p< z{shXxG*ELA}R9rKKxGCTspa(buIHU5+syxFi`wJ)q6CQGL88I@%xOFNI(;*L| zD9^tJNvSq*|8d$)&b`*9zn+_W6z|7hoSWMM(n~Q4>d_MZ`zY`%dmQxr5obG`hJCXx zU6bve_Y7^iSxqJo_Sf5(<+x7;+xBP@CmchzosN`WUQF~my!SWV4?iF z-wMu0o$jw$MXF4wx8`MJh*u(lnqDnrHK z09TbN?e;n=jD%JfDit0kM6eNN2*@3_!oYxiZOE^o}wFzY#u+sNLs2xB1Z*7pOQ%RI(+9N z*(W3l(mqJPWjqW zm;iN0V4po<;p=5aF#+hCg6XbA+HbIAREqlm)}_F*G%+%%wVnV-Yk~+gv5XnrlmI6c zQcQr8s#HutH3nU344I~lDh-1&B@&<>3)n4XWX`4p#Ch;nC3XW&3xo@*OlBCt(oFEF zM4n^bo5Q%nF> zUzEsA3UHFPq>U)>^-?R5NdYkYnq2_=0L28rCM!merHPS#t#uzjS`$Q|i9XFCz)AHh zCIF%-ra&x10-1cvU>I~LkpQe@6%*j35{eOITLyALl@(|+VYMjo9P{4uUt}2cuq3ez zZjxb$I;h+fn8|)%n2}CuNYe!K9pBWLq^Vnu4k#qXN!PW5SwgsPvz@iJ`$5>gy`{Fh z2eZLF@H$dQmG_kL7T}~j#RNF18N~!(3IR5rd<cr0$?)|I?>|mwO!q~TAw)Z=x8OT{7-GqWO;+5W z7Kyv7NZd_D;x0!VgrlH&Y`wq2Zly2xfQChGN5dlbJ`Ic9_b}`xXQipfy^jkzN;$nF z2*2F7%Be$<{Yn`5eF<@K9V`iPIRSZL=*4BHAwC3r5T+FdnTI@h zu^a>(#<6hk4@-J~2@5SAdEfvJ(kQSPemT72PJfLDkJ|&B0mLijql1&cjXC5n-8C@d zvIHChoB%$79e=nHJUF1^kM)8q2hC}Y@yN|-SV8@^;sVH0E{>hq@^7t^U92#+mQvft zHl=nB*{S>M#19$i5HQdLa3Z-IU(`V`c(OfA!}>r;`V4|z!^9sB<3c?{W1aj zldK*H@U+Pp5#6{9^1!7V%I=18yRp{ZIgBur(2WH)Pf12Y!tn)e;SDT9coQ!ih#TxZ zB-a9rOO2J2?!ZIQScb|`3XW5-!#*B<3@XNc>0=mim|C$t`WS~B_S}7sH~g6Fmp&%@ zrCb)FAn!H5paBv8-M2SOMHqiWkGg_|DdqtUEakNC3KM{S;8tDki|3$oOY4Vtq@I$+y0SK}fR*$aUp>7fL*-Z0k!d=(qx~ zH7=O;3pZ6OVm*w_P^c#)$be7A(hU z&T*KtW-D2ir4=C4mCt)VOqOSM-t+js80Fik#b3bD6xa{i#m{?wj*TTW7e|qH?bLtZ^PZYh=}yOu zA+*M-_seX0*;WC1`f?Yld@jy=zJ?8pEOIJwHH1dVg{MDc&~1 zOduxF8L47_bOtJ=Tk6qsYIGSeuO`jMzv^VRo&_8%?MqzR7lvrI;aU$ehvV@O6{?UOSAT+B+KzB(Xl`fEOMMCJMv0v;_&{sknGdc)gw{|pf?`BN|$8$)*z{GoI;iz5@jyk~iNNVhTE z|Iv@4F~+}vxU^<@N(n(pjJi#o2tO6Xygp^8aIvv7vjZpBeK2XS`Uy9K!K4;xD0z%k z0t@S4_$ib_6uyt%G~`h^rLPP;uKZoXWbuv5o+dle&+j0rojcdJ!lNO7_456R@kSj`LC0*VN7JBhvS9 z-@sucfuam09!wJ1AX%g>A&RI{x0!H8Lsfe3lb$UHY`%FZm421#CojD!2<3^VY4=uI zf#g$KvNl*&-{~I@e|WK^adRR(I3B*wpo4C0u=io?b&oqwc(=dGj5R55`eVo;Gffc1 zI;ku06^B&fveW@GY!^~9XbQ$-WG|zny2;}tnbUg|yVOtsd3g&^5W*^FK>IjP-1}JX zZ-7t)@O~&+bSUgU&XKcpoI`_rWnwOz(3(4<$Fneg?6m#7(~gENS}j{QC*} zPfeke4Sg7J(wG=*W2}%fa8!VJBsjGi0@j!O9OHV05XzT(41SlECY=5>)>G&S>}P&2 zy&kNSha|#xc4Yiumro+AmLgpm*i~?oPzOGu^Sa5ZUEq2gX-H`L)ok2x$-ve?nIf+l z8B!*73t5sskYK!#B0WSpk3pgcN2VVZ>FEy~?plaB7sBrgD+=jm74I+^- zR~mUxBaJhx(*Y=NSISxu(^zah@Gv+2^lB*@yGbYoR6D=K;d+P313A2VFQ7_|9UR?( zsK5gQ08PL=e|!lvoLy(6~-((6PUWnsX=wqT)` zzsOUswF)HuI?<31f_xC54otU?VG<8n@b5ot}o ziFCL;0yQRdUQpUIQ37z*U}z;z~rXhqH{J zq&&-%3&6;ibQsa7ij1XK97ICp`h%!c`;R4HU{)9zx4#qy4>}x?G*{D{Gap3NzWh-9 z5c!&Q&bUnHQhQe}E7-0LV`2^~bUET6CdWhpnFW%HW8~DVSe+ZgBpFN7&&Sl>Z&hyvL^nTWpvL|sy_p`2j65U@aom%{BdL5_t z*WV^w|C9LZ_fMI6XYOY`jOopGFXJqS$+Zei;eCOf`GZi8j2YU&v%8=5D)7PR+d-bc z0lU-t>-=#DGknxvUyg{hzmD}~&iyPjm|5#(?q6o#`8&tw`~Ctnam$67?%O|i?NP`v zckS2Z`%U>iAm1_h9+U5J`JRyPgnS>C@3-aq9r;em_fh%sw*TCeJ6qGgF?1V40UGJIGISk76%5%7En}#jp(`1BKSQr& z=#vax#LyQQqP1)~#?bRHP)*-s=oCXgVdzH)LDPqSmarDyMO}OZ@4w=;#__%iFa42) zllZjbmFE^z`z;6^2DI+Qhd#07`K&>B`E&8-@gbRcO7=k?cCn;rjl;4JCR?D&Tu_c}yaoB83x0DZoXkmW(VSio9;O=KfJWPXH* zgHJD@JadPyUw9ouUs7BH{6(DS7%_qOb_V#&9p1U{uH6W%!uu%-dK@oohxJ7wkAp8u zJVu1K1S!iCMAk!ilKqs(DukHw5|J%HE>-Q2zmer;ipht8JCXcNbPDT}__RqhK68i9 zRsHwxPe7C1R05^ng;Ex?DirViM1BV3y+F1>kcD5z=YBusT}XaE5P2Q}LI#(h=GH#} zxkcmg+e(OG^Q|Gg3Ljj5@t=}?TGw>lZ$M6Y7@$y}VZ=M}`KD5yySi_#Y-?$6YK^wF zt!|9Pqs=Ylq4?g8SR~fj+1?pOf67l^tH1Tt+}?)NXjfOPGmhWF+7s<;j<&_E<;!pA z>TI~-?ymN>+%f+!Jl-jz!`(-q>D$cdQ{EZ0`)VMkI2%i2+etcS{R@ zy{bsIXlG}1Z+0$?c==5F(_h)SOiWg`=C+3R){bbrxxOV9+!EUr>x{KE#I{lFT@Z42 zEFS5M?RrZv+7R!Kwgja<&0Rr@vRW#-C9+uNhsNsL+goDMHmk9@t0CIi7;6M+YdwB# zEu&cLF40h0mCLQ=*3KZCob7uBUXF)r48F|Z0JPF2wYBGqK)`XsUY|R@`sWd0n6TmG{An0=NZ5%oE=@U33P8oI^bXx z@&nGGwU^eQ^CRoU*6XY$k|CP)vK=eYZB#6Y|Vyk5MBJh1Rs)C`=aj{i01~~Z| ztE2-qF9Eq`f=IjCT2S^z&|(~np@L%i9a5LFYhQQt5^b^vm({4X)MnF$44IeENAfuYq|D%xhp?1M?b~*TB36<~1;{fq4zgYhYdj z^BTz3z>=%Ww%oe)*1AjAz5CXVrCabD>DwfENmV3r=jN^P6_H5!%7*stc-Xq5GuWRM zyQ+cvE!7R-isr_!B`Mpgx5hL3S5{(hfkwY`b9o3qJm1q3t#95_UV(5|bEKgKyIZn- zCBO9^wh|uplI1C>EBlJcnahyZwhe`r*w|9L$4VS38Jsl$ zR!UTI=eEKOIR%`|)Ss0%noUP$YZ&_J+@U>Wqjr(iE$t1_me|_vu2^JOv>_G`Z=rUy zYgnCiImDIC?Ol=jXjg1CHvf<@yvORdI-9b)TzLi3p=jOmt8Uo5HBw&Q&=l>A#5<$S z@h)_`FyhumTUcE? z#mTbo+_IE<_n!-%&28Mz4!0UP51j*^8_Ko&4CeA$q5Y>F9VixUE#85`tGi-t@y@;L z%fnl?u82@`Lub#$7i9P>ShRG7dszI@oV;dG`HaVM64Qfd1x$1oHqq;&jrGm(NEH8f z2LFqXW*9YWtjC)=+wWBkO%%_+ssX#{o#DG#{$vixT9`H+fZ=AQo=m4^49A8FWj>{8 zFxT-|XKQm?G#-;tJtOh3{n3aSt2q0R|IGv^9ArA1A*he-Zf=WUI5#vgC9!}5L$ac8 zXX5)Z@w+->dF=W#iBfekI<(+YFxsyFugI2IS9fcS(Ss5lF=s5DaS>;Tk*%>-Xp}U= zM4^M07*qL_L~zcuAYzrZMw{Dk0=b~I2^ESpbjM@(Utgk}N|xPemGgXIq1BJWr3Krp z&O%dtrZdUICDwvM!)<-Kq;R^hCPAA!2Ru-cX{oSvykya$sZW(yixwqtxVmW3{XAk_ zkWDdDlAHf|tBM=b0z*FB_a^pTz&p;5-8*e$vm6vrpnb z^Ya`0C4=*9l<7-wSW9_%MoRn|;8(snkv@mC4L*2>rY`_ZU~wXSj(H1l(EI0Y4F>nm z)slwZKd1Y=!Toc+zb9UTL5YXuKZTbU&Uqe8{Kp#4V&(aFd=bwid9F|-jo+Aue<%;< z?Fjv+`Geq`jko0CU(3TUSeQ${F%LhOho8v9f1QV44Bcm!e@z~KV;hkrB=|8IHt zH}ddr=ixud!%G+E%JWBpCm^CeS8IB@Jwkc(Z_mSb=HYF5_<=n9V|n=J6i=7;r9Aw9 z=Ha|Y^bWkWc+q9$I?`4^uD-Y9eJ9>`;pJ~wyc_R(@b1L>Uc3>!b$FwAV|aJtZNl4( z_inuJ!;4{J?ZVrNmlt~4@#3k%o45_1JMrFy_jbH^5_}u+X~FxumMv&;ui>m0+sM4F z&>PBiWQx;dyFR2SbhHY&6w>5!DRc)T;wk1kpsey4ES|H-(jI?i+00a3fS-c}^SNZ! zDouJWM8hKaw-zjGb$kwaor5}NuZMqI$z++B#pG;Oz-O8LtcsR3Z>B)A$(V`A-CxYa zXRnGgvFX8aX3OCs8Pn_FOj^zYSfew8T~@w`%w6&#B5xOzF=w^`DwH`Jo`{isI4py+ zmZ@e@Tx30MQnDA)1uJM1k+*aTE8OcV&)@6|*Z1sg+`1;ht-+PiM*Q!bG2D@CX?}mK zGaO!DUftZhsX2sA!FOzn(3y;-XHuk81+ai0;PXY7f(>84BLTeT_P?#IO zk@9?_vn>)1Z(SWx8=juw&zBZx%x!?*qJrGe*Ay{rJoi}H z66@-!1|?5HRz@h&%8qDrXZ1$JZM_xIc+!piA5nI0D`)hMgB|SS$5izZz1A*Ahv&XA zph`1MWm;s7w#X{9NM*4Wk+Z1swGnvw9Y#30^WGV&MiXugvw2OEtwGytsxL?LY~H#m zQjN62mio_L+yAfHc1>dp=LhY3!?oxH<>lIuBW>-C=o{fk?VfNrtDl=eOv-a=VTyI? zEHhKQQ|H-_^Gmk8KBWF+)-FJvi=ks$4eeQ~HtlZ0JuUDzF`Cz4G>hfU)0?sD=ZS2& zdMn4-p{h>K<6lWebR&P3I{PjvdaQ&=JdGYJp~v-N6@6LH6W?Hc#mlTI@_Iyd^Pcb; zb2CKdsMD{8R%YpRWsz}@xWd)Y%Hpe`brE#&aJV@f-m_&}~Wy7b9-SD8OTOW#^) zX!)HA$c~ezEX;uaR}!zJgMEGoyndS7i026-D|I z;tKoHisF4~ExndvO{I$_b}+wxr3a{ywjvbk!TJYxuk@x=XE#^-)tgx7HRcZ$!>hvK zwY;bhXX-1|Yjsw=Ru`!k;tK1vx_G_TpQc_qW;SoNBjv004+?RaC?4C5YZSQrb{_3* zl!xTLndlK$S9;D{*(%O6*(%J|OQgBAqlNn~GGA|tY$@o+dEUeCc=JlOi125+)wOv$ zFzIo2zQ~uqg_W6Te!mi~l|Kb;i?+thGBT~X%mkSJD{PFR<}DjfdyZ03%(he%*p}Q; zSfrS=TbP#R_p76F84er@5%jfit=f_Er3Le)+jyRdx2Cx(k~>S}Y*^$^N2_sxYQ`ilQ&{V{WTac&CPx-M^Ty-0O7gp6h~UCU5qq*-}yQ#78VKp5yVy!6Zc zu)HOmcqZG*;cV4%Lu|{&t!vLf&VuDeftZCuxRAuJZF-a!k`xqlJ{SW027{Rw3#LWg zKde|8iF9s-C2*hnYUPgII{f3oMEZCzj6%g!Yo+qXX?c0<9=d(9 zxfpJ4ei_mgvG!g0v&;F_kd5cPf0K6rlO8kYgB~+8W25wfjQRRKOSdXB_3UbC7rS}; z`t{~kH5QWiot2Q4AG5X~rlPzsYIR{$Va63}3aD087DTNqWWTbI{mMf2s|tl*wSp~x YQ}lEPF-EoUdQKrtW|j%#ZWEvX1IimE#Q*>R literal 0 HcmV?d00001 diff --git a/doc/examples/get_ref.cpp b/doc/examples/get_ref.cpp index d0ddb2d2..a152d347 100644 --- a/doc/examples/get_ref.cpp +++ b/doc/examples/get_ref.cpp @@ -19,7 +19,7 @@ int main() { auto r3 = value.get_ref(); } - catch (std::domain_error& ex) + catch (json::type_error& ex) { std::cout << ex.what() << '\n'; } diff --git a/doc/examples/get_ref.link b/doc/examples/get_ref.link index ef560c42..c5bd9c03 100644 --- a/doc/examples/get_ref.link +++ b/doc/examples/get_ref.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/get_ref.output b/doc/examples/get_ref.output index 50bc0df6..3811afa2 100644 --- a/doc/examples/get_ref.output +++ b/doc/examples/get_ref.output @@ -1,2 +1,2 @@ 17 17 -incompatible ReferenceType for get_ref, actual type is number +[json.exception.type_error.303] incompatible ReferenceType for get_ref, actual type is number diff --git a/doc/examples/get_ref.test b/doc/examples/get_ref.test new file mode 100644 index 00000000..3811afa2 --- /dev/null +++ b/doc/examples/get_ref.test @@ -0,0 +1,2 @@ +17 17 +[json.exception.type_error.303] incompatible ReferenceType for get_ref, actual type is number diff --git a/doc/examples/json_pointer b/doc/examples/json_pointer new file mode 100755 index 0000000000000000000000000000000000000000..83f8491caf97ba3c05ab47ff0005c6be85579bdc GIT binary patch literal 47612 zcmeHw4SZcymG4Q@LO|dK1dU&CRX*B5X;PpK)Jj@<;YJdOw1E!Txk=O8CXkOyJ|M^x zZ%wb}a6FnBy-pOJiKFu-<2X*7@iGy(ytH5w(dWdG(L6P|F!nH zA14XpK$*vHUhgmGtiATf+G~HUz4qFBpZw!co3D=C6uJI7`WF96iqR!8(&~j&oVf8vsU3DPrgBd^rr7tdYcRp;zjLINF451 ztgrKqw$9zLw$43Wf)}@M`TLaK5tEL1kf%bD?017GODwi^^Y+aJAxWhp9iHC6<;MYl zANw1|M2^>3tg}7Vb7#k{uJ%}>w|iN7Cb`gKdjH4%@fIHejK%hLL%fB!dc_{&$b4a@OZr$A8Q`8^+} zAAYM*UZ$fQEBrB^N<0jc{=f2K)Iv!WI3jGB~4lDoWs-FFM#=Iu&;;Nff!i+_{ZTN9<)j(n zofIrsQVT}jy|RLWiO1<$uxc$>lq|WMW}qIU3^l}@lz7^y9l~|CK|K{qHHl}cRY5{Q zKf$sE69RU2n4H_GA+p?!0Rs0R+ueGXk>H>Zw?2R>M36^9mLSmMxeNu(Pe?+)g!-?osdYU_}KS;R2L2r$M3WXsi4u~b;YRx7Af zP))FG!GwUFZ6N2EP{{I6(U0J+f!i+_{ZTN9<)j(n{Y5-bun8>~dG|gCW#2EDc>E1w zqhS47uqat_Hx0K-!D@(Mj48W21Y{f3Q?XQ&hP5h4DCj3xwqQcQ&Q_CiJ5@=RPt$^} zhTAU~{ZTN9<)j(neFKXM3HA#xqy;1I-q$e``~Kqb7-FMfjN?7Qz%5uWz@rQz+x`}E zw&-ww!+tzU$}NZ#2$;JPen>NgNP!@)mucR*M4D1fL(v>eQ^Pb*!ekK$n0pw0$ftFQ zG^a4lXO~DbplK*j0_41z7M5p;G*@aGj@4kA&oRwgmq=5pX*djmX%bBHb$QU z;wH=vWKW|1(sgt?>TJ3|y7gi29F_s?9zypcF?x9aM8bBe+z<4AxC;y;g5SMT zHO$A33`X5mJ#CB}Nk*Oe!Kht74EIpfi4cz98O1Y(XMEYwP(Kv4{}I`hAv>sW7CE31 z23`x#=0}4BI~fK;^J9qjP;)U`*&Srd&5%1UTbg4xD;|dt803ovnehexP?DOH$ZoWz5HpOhw&QhdD>tkqNZf6xwVCZgh?a;Vd2x z&pe)m<=bpp+iV8eq0OerL7jG~DW3xrQ-YlggVAOT@lI|kW-A*eTW*Hjc_*=w$urtX zJkCIDxEaTLBLg2pV6d7uBa)W%%92H)e*2JIFyWVDi`u6I6#Vvi62Yi%iwx%4XM$ys zF-4cO&$p?{pnbm1w03G7{=$ymKJ@VJ*1$18&~rdtxkiy#$O#s51EaZFPQ>1Y>ZZ2g zouh85RNkxAO})`OQQcIIy?;|WvvmJT-C@dxC*gKvg%;ia2$yR+9}d>U^M9}uGoM!? z!pxr)4=Dq3`W}xsejxe)J-NMWbbejS2N#sgSCT;f%tfTKCY|llZ za}EGnmaH!aj+963eG5^$Z$9b(I(_qADZBc>Bjr0E-h*GI@?nGO8zf4eE$)eeK8l5Q6X^@3|v7U>TIC{Nd`M9 zBIvuKaXGGn(=oEc;-9XN)CO}!9d%UAa}?XSxh!mU&8oO-Vog{ZUZ z=i#0QTmW3CcTTEztok(v>+GrMC!)_hi}d4YurWx_QFN4XNZ2vZ8wEZMNjrt`3HSp- z+B*AYyms)0VaZndkNWdIMwNbF)SrX>(|dKUA1xjq`;X8>9<>*uuG>n%=B676>`Vo~ z;nZvKX6F#8>|_MknbnNqD6lh?iV@V>s0y9ICP5dJC$GY~fqb$~Gf_sJ6m>y-9r3Ai zi~^D2WDEkrf+HAOx=H98a0aL(=wYXYU#!Ul`wsm3#pNa?z5ypi?USR9%hI674Qh}8 z+Z|F&fb9+omKI>AM-&rar$-ed*c!U;ciE z5MxTxx&kmYDki{oXB1OVTLunyfWllcfiaCFz;?$K6JWa&iV=kFJN@8uCD9g$Frfkt zNGv+AV1-hz0ZZDNC6lRg#76(J=C0&I9s`#|phyr*re*=Oan#37%`?DJ03}Q@0hkRj z0ib>b*eLUP>M==3P9HnI*U~GqkDJYe?A-4=erv(T- zR5tMJ+R%Zmkox15V9B2JzI-F9)t)?@`bR`4fpn1yb}!wF`3>W2S}2><9yQ%cczC~s zxHr0v82R9zWZv$oJxm4*g>aG;PiPlUV&;VJYNYj(c_i$(^g}LIQN~g4fr%1Z!sn(Sucg@I?8x?^hi(Nx z-D(wNrA8ls{Y~O%L!hT$8l?*7ob_e_(gNN&sK>dxu-p+Hs0^Lkc30v(cij?x8<`nb z$_4S>AUS85b0ElTX(UrwHn?{q+)rzd2&JGJTXQfrff?oEqX?b*!AbP+hvPhoe>n9k zyw`G60BI)7_zu9ZEiR z^yxz-rt5M4H|lOW?IwatPZerf0JSag|#-#tPiXe!IB~Z4IA1$jZ3geIkZdFj{xG4 zI-)URuC6e8s#Pfqu+3t^tT5bu4HF=BRG$Q~B!~_I4zd?8PuTVeN>kO05Am#?1X3?m zp#$3?r+vlYuItdVs5rb2qq-#Ox=E2YSTeY$eoW^3daNQE$FLNcKvrB$5MuREKaFPw z&+M|hqm5XIWXH(>cSzG?;!2GQa7O^VPyQcaau{wMR;Kfq1|4>r{)Jvpy$4IPw0Rw4O)L2nrfAsj~cmAn!D!d0K0z zcp_!lymygeEhsJ}rlh4J3x^}m+>NU8u;5y_rx9!Yh!=Y3DX!Ib*Rug+vYGoZi>@=A zH@#TywMx^t6Do1XMJT5PbGT^5l$esXE|SbLcwhpmSkXSL<1#_ic{f-|FDMD&5we3r-%eKu$drp9P){hT z>bY7#szyM%T0JWXp!5pxJsTvMh)Niz)96q}&%q)M`pn@}7v5_z zGMhxZ$sCBmxH}U>!_z?nT)yh6k%Q#gI@hF zg>ZuW?0+?}IQ z_tOP+iZ*OT)gDgm#T&IC7(Dx>_7%Lq+>4mQjES|Y$fi%LAXRILlM##t6D+Gm*3sDr z;noM?rZ$l}dnY6j>>Ezsc%9t{DU_8yVn1Vl6V%y>>S$1t&gFgWUD8zAwP03u8fQTe zV-SzsI1e}vI5QccN!J?K_ZSug=fLGe9^-1DaVEDEh&;x1K;*Hx?+RmC>0nQ=N_9J5 z`x*Lx6fY%R)Nxr?X#Az7GBsyGg%x@Vz`tSyg?dUk=;WqjBFHr=QC$I8uPV00Y~_f8 zQiepnf?^0%Y7zndRHz0q=(8F|P^h4g3OcR;=+uH`5(3-?0#tp;A@sOPLDH7-gCYZ$ zHyrf`q~jk47-0;o<|LwSr8`AC>dt?tf)&odlsdVJ5)lMTf@GnZj&1CQ3W}8wos;K5 z!B?COfhi>*0LB!>1i+Z0m;lvVX4OZ4t!kE&)3!t?rkDp>uuLjhrJ7%5hrQI*A__e6 z!Ct9jt>kGLav1LrOx9c&FPWpF$jx@QN}f{dUJKG2iKZUclOVR=7-#<@9O*ZURA5r? zUIZ?-Ux#VWv-|#41`FqROj-_5jvJ?b*88ejzkTy^-oCuZo_rxr-UjW3`;W{*|G>yy zAWN=)+(4?sUG*VMrl=fCd!Tq4_0KmAo4x%Pv74_W5$7+MF*p3&esQG|tc%dk;p@oI zhWeo)hrAy|z>n)InKInmRsYO605%zAQap{0h}EBXrgjkED1?Gm1qlTr-Im*#*Rd?y zk*&g=+eGee`#t z6-%zY?;;VWa=O1x_x?xN_08Vb1iWqcLA?D9!YTUD)RPbLKB_ckN#lcLB4-cENAlA% z(_s2-1w33$w_*-*o0Axpx*vAs7K*K znke$Fv%hO+w!$Hnplb1<8M+cKZ+Z(0nlU*EzzzaKe($Hk7%6Z!fdD%A!e+)#(FS!m zwGD3wEts$m+(1EvBbtonW~s1wZwF~rx5)t$1Fw^ABr}P%{DIIz9>0A~RFtGbs ztLW=Y5~*E}p7QV)$s-Y60o6h6&~R=&;YEp1LmzRgibQfyllKTPviNxd{NF;*x9*kE}7JH?m#JpKq7 z)!NdnA2%WK(*lgi&tHA`2Ecxyl^POYtyP4K(w~#%GDMTsjX2)&W#XyOKs6nR9mLl^DI3!Hkhi|%IB!f&zSdoO;jdv z@H8U{QXYY$z}JLTA%@jc!j{s6v8P5o#Su`TUdxdXaKlZzoR%#+cvuW*3IRpy2e2*r zkz;^ZzPR7#YuScCxe^c%wD*NIOn_?HtJOz9a8sYeQD_0aI+ih7nW-01&ahWWDn>_x zO;%}w7QMoy7gHjU z0&NJ#aB;|cH*^A!njm#{K8&^r9X9lWrZfuGeH6UG90yCDoC%;sk7od2)#F(Jht|Eu zX+S(n4+)&r;<+qXt^l!kHs`Y(oYBIWt;xva((m^@o?Ape&K{2?x8L0(kIN$Zaalwg zM$2@~#9aW-%$#T?fCA_0{HN??^xcXN^U!!Vd^tB!EwcYJ_&;1>WDvDNt z-(5@i-MoZfPJb&%`v$~A(OAp?S?{Bl-Q;1Grk1;4lb7x|v8Nm3TYm%C~Q z=|Gst%$pTY%p;d$vkAUDMBopI22ylfIadGyC&KT)`>dj6Hv4pRv=Wf zW+tHE&P^{yipvD(sPV<0>(uyy-5{l{K&F=&w!7JUG6Y67k^oz6N=urSbb6W4*aG}% z*UY!({7i~@pb9*RnNi@^%e;8L_=z^dc?HBN?Q=CDqQCe^TX5#9*11H1*!?K~eY>9_ z@Hk#*rK5^yhuZnLfVm$D9k~S>Bt9reEUYI#y)VdRLNLDl_a%F==F-8kM#f$w%Exp zdY}L^JDbxsikb5@Fo;UYvNHno2o97|#ROoGC?)_>SuiyqK)x#`00vo+;{pmL!>CkS zY6KuR9*BU$N=+dEW^Ba-_{TEL#ZQq8!${;y21B4slL&yVUNHesLlqOCk|C@<0&vPp zF#-N*z7+ESV^ImqDDdm0K|Ejl(U8oL5+4M})dUe_IH=4B2p-uO)-VMk8InlkO9n$A zp^*fR9pI~Nz36Y|*K8z)`Gt%hP#h)V?h0+zo}}&a$X;J5-X}pxe3QN@kV-MX3J%+c+3c7N#f{4fp#>%O;5?f>i!wY4F9H8^QhIInwV#*CKt8Pxi-j~$Px!C&) zbyw5<3A*<`LS@Wopq22p-M8UAcQ4oyRk6VDt|k0#UcwL3YEjxZzz@t(Oa;h#rz=mw z_bPP@-_KqoNrdlTs9X5{5#2DRNmEaHE0_fP8!Xb>;C`X=^4NEn_x?dlbZ|UILgujX z0q~bUKaY42@_4AZ+xsBs@!0!@03qo2Q4kLUj$i_SK$c_Bqkv<;$3pj&;V7+~q|H*+ z=T2;h;OFfVY;ovTJ+|v{*cvz<5g!Ro0$4bpK)P__lFGrM2pfkY{DToRPK&U>0SU(( zl5l9OHs?7~qylahz|F$4fYzMNIxTF*dk z+VctnuQQy;BiQOQPxIs&CYM#+q*tFKwbc3TrV|m<&NgwM+0N_`kZvHrbgh^iRe-rt zG1-yUtE^D&`z2;ZfNie1DzGzy8b%Ph?}UNb64+spq=vYE$e9S_%BVzj1wf@!OaKmS zDW*W=T{V1sd1nX=X(R!V-iir;imDhvp}Zp%w7mdYh(C5P`Rk+jd-|Zi-aLmb=5G35 zQ~}&e&j+|aLpkW4`VF|f%D4GSxH+ut=)q~&TQNW={s<=Jd@CriqiH~DoB@zz8f`)` z0nktdOAD~m(~1eOb%E^&DA-e$1Hi~`H4ErhED=;}8KoMmKLjr{BkGt4BauI)8Uixy zL(D3xp0X++w{I{{Pn}X1G+aQD27t^-9919ymUCgu0Jl=Obra&57J$AbSVjQ)mSO_X zw-h4?-S^IcpOy^po`GCIL>DUuxm-;UVHcAWdcG1}OfdoIVu~ryO?D*myO<#`q>%)m zizz0c=nQ9J7b|1dSO^Qy?tmkQno9RkaB1tO@I zR!Qg|=}0Ih!0aMBx|yu!EvAvk?+S)MgC-Gx&0@s_VB<$Ig2JvqDk!o7w?Z4ONt;xg zP(NdWPm2hXnFa~V&I*=|5`!6EJXIgvD94dQAC&1H22ciF=m#-kXrP?kVKUVUgul6; z6CcPj5SKb$t)0b?W~8}C)l2zuHRMpVgvYXle#sPE%2`Hu!A4BUGj-0Y*NA?Yof2+w zJ`m^En25Odew>5{?>x2W>yC88s~UMCmV&h8TB_A>*|$QeAorNZ*ttz2iM?iu%}LG~ znc5C6|2@b|e=tST5{4*I*7rb`0KI_+<$X??7y^T~0Ds0yN*flRKh6x&)6NV@m;yU9 ztQbLI9~_hNxuXKoJO!ESgG9j5xFpom$i>Hyi;Wc82Zxc!?}LWGh!PM`bPO4yc1*(v z3i}|bpiUIH4O(PnHSI}~jB>0;4X`Yvr+-c+Yog3b%(R4|LrWBMGN;%~;@{k}Q0b^* z0Y&=}oRKt)05_v>D{2-G)r8ED)Khv;KsG5L)en%XNg}98wMnRHqIiR%_$-xC`;f?& za)v;K@*u!mQC#wDv`V>3jV&N})+#}cU{X`Yc4$lmenmBjXVF;_|B7KcAz=zI{V7IJ zAn)B)CEf&(Q+J7|(W2mBvTPM_yFezi9f&dqm7OEVL{CDcO+wv;%8kda9|9aI>@Q4% z=nAw6up$&sZ5JGi16GU68vo-hr0nWw(qV+ldD6qo2RCqu(cwkLedD+Vi35L;3BU=! zDZr@+RQZByaShA_jVT~qBOp^PAX^ELD>)IA+ym=*1s}I5f#Z>quSOKV9sOHTuBmf7g6eGZZI59sU+_2=9sQ}0g0U{Uzt7Yl#3VJ?{*E2Du#M9iHOb;;(r${AAMu43e zQA~iH8C8s+P`Z{0owNmzs|g~gi783(yWW;uSU^n}oyiawQvw2h$J>&mf+8z$O>dZ0 zIs$NZP%!~eh6GCs!0~Fu1eo0toI8LrR50pO^UMQu0CGhpf+AO{&Mg4Vt(X8<%@k80 zjX#S-e&ZVgWg1BU8c;C-P#P5zpp73^9|6U8BVoQLN7$}a#-b84rN9giGg#6M5~cu` zu?5Qrurp1H39vJ*iV+m{ff>oeofeRu0?5@15!B0|B-DG1u%Rg?0NT4^3PegwAd%k{ z41t735&*?eF#)(`r5Hh>+#nTHfC5+hskup<)+S8%OLJ$Gf4z(BWVIOCR))cLE>Ti~ zZ4^6OPRoac;rVm-PrI=NY=;wL|k!kd|CawHgYvyd3vN%QJ~2wrF|6d4ZPq z1!nrpLsH)iLq?o$=FZ}odwiGzb`9Qu4}Q-fff;BP$IfBH!S*;Vzi=jk_=jQ(d<`^6 zn1BZMjicsHl~TwjOcv(O!WK3xDl?oZqGU)dP|=N%@sMj`f@Wkvsvp-{^w2rZwnqnk zSrT3IYiiMU|p)T!N~KWhx%|uV&C(ZJskVI9Tr zU60HT^=|<1{#;CM<}RN`-6&Y-10iV@J;?<#-s@kNFAP3UDKzw36Zmow>)y_kNfRir zGt{HOq5v&dh587}L1IR$glMvImmME^q_=5Ud}$=IstL~r&# z+hO(-(Sh9^4Bl808ZZpjNOU;i1$n>~c)@t_g`cg$RYALvqjHqP z4DK=bLyo}OY~ya79NZB%WdV9W^;sOaarkAJ#)YW;9WVM?vuyR{t7IoPpLF})H5WLt zJCL*Y*C@TlG`yaSfeXpV5rSI->M0upt%z>ByP4=G*r7lNqL243V}umTbs4UdkTLZ%|`^mV*KIZUUU)6bwBzW0A4V96%TW8mB_z z2+(VCVf7J!v%-og&|@XeiBab|w6Pu<88}o1$_+xzeE!m{5zLsM^wQc!2@4%~LYrLT z!*(ImPlEju?Nnq6!uo(=@*Ns=7F`OWsZ0Bb4gzgLV}P>tSR{7iguF3FnX{7SAf!0~ zX)fGouk^1+tEcIz_)vqEW=x2s1wd66PrU(cr^h8s4?w~)tzpyRX^v``%N^#Pj1gbv z;D%ZFn>#Yb8sJ}*Sx_npPLJB(*$fh1+JpVI1(e zf)Q%vgc^xr^Q#Q>m9T8=j3G57g|3(ZAtA0n@)0;k0jB`*kjxm#$ajGFEui%pVMA_w z;KCg*-RVb(hTtATWXK^tS!RgQmS8pEx(va*Jt78CR!HXdBVqwq*4w-WrWhe63Wd; zObn``VzS`&BTsg6OA^-_x%tSQO;MM4mJ^~8a|fqHF`$Y%D3kQut7Japclpd=9R*OF zr78`eaIW6@u-yOJf5f7+!vTN9sB=@wQMOVT%d8e)E@CXqyUV$Mm z{XBP;XbZ-Eb~QPJ051o`#|EULj2JS;rGbMLQqSB$E;KOVQW06o&Esq;-jz4aG#Qza z65m=s&GfSvp0nOJpy#0x%8*bD*Rott)2KhUGYq;0`H4bsI3>|rrJ^gMCDT!RGy2qK z^nCgV``(#R?&Py)x8gc?3HA^u<9MPOwI!dF zMA7$&-LT$?mO-xKQ)#9%=f0SRq>!WyV6nDjf@2C4@Q?yzD349=8-zF>gy0?m90DA! za~g+hap)D-kVoJi0UQM!t#{7gC)uC}>KLi>m3{JOPQi;^P{)OKWBV|s2=;Wjy)4&o z>u?DhGY$N_K8~mTkFeU8SpQ)CQq&%;Qqvi278RxVei=QDwC1ExEL_+M=<^QRt8oy{ zK{U)UX%Liv#gtm@Tih;vsJVL7Ts>+Ik6k~88XPmVIu184d86$bCkkto?G&h0dPPyI zHfAb^AKhTnVX}%0*WD@y`zz{TpfIzLjn6O0v{7dtq8%F7_C&|^t9N<*m4gv|bSAKV ztYQ1uX0Jm5OthRk@HH0!(K0i1AucRw3UmF}>~I);#cXXDOfPeo9TlVCII28t+*}?y z@EP)*yX(Orjs7J-k#HEr=eW+OhA7brMkdx(d1n9*9stbdNxgzLA3ikac{wAdCzKO(P}q1Js%>0l(Vno+qfCOo2Ydlx3*IJfP2y&N z2u3bP0{vzGk)-yAD2s1Yo16wBp03OtUiPHiYD+&=Sv1GqQ#RwhLw^DtXOs&HVhM(SNL+#?h=%)J(~Ry7#xMb~@L5*VJAqFOsnK zFo|+c_1122+0v zpScUkC0ZvkW!FC7k2Qm)pnp`y99Wc;DW>$cfH*3EK3>1NmCum&aegUr5Q)9>d@j5j zC7KR}{f}^@?|j%)g%0Xpq7-xABX?*xS$Qg(-kW?H8r@|3I`}ds&LA9MPQO2T^eBvz z1?TtldkI?yG2o0a%V^)y^|$WaL(3gS^1;+!vaACVF>lD%E-g(s_ouqTVUP2Bso2Vg z#l!#j#*AVLY9N};g$@N9#dsC^uzoqmw`55dpmHU*K6K?UDGAsb=zZhCJXyD3mcR;q z_%4kl%$o7zU)eM(cwH~xtQM*l^w@4ukw;DbvH`bc>1j_M{Z@3Nv|Z@H*`zxu*{^b{zZ|gl;!)6yB_t%l#*1?oowT-w^&wJF< z94JuT12J_{G+%iz5<9E9;F7AetPgALXY0)QQ8gHOzu+`rR=>E*1=d0jrAgb{AQOaP zT14M6;$3#}7ayF(4zhpl(iphhD9QU@HX}ljge4=oBOuHojs`2K2|INzVCVo(jS4G{c;M@Q`U#`ZfnPC%7Gs}NGjL@r=_VJ8 zyv!hySh+z|CPTxhoaqIogZ+q}xj|$JreXhyK~(GW4}znlYi=7)%5*NJ&#A%?CjOl* zPohREkZMX(O)FCZ{I6v|Lbt-(`z7cs*0=0^zkz1v^!;XI-$KDIQO+JSjuIW+`S8%u z^L}!YWgR`wllRZ$eMH{Bl=r;6pOW{#%KK?~FUb2jdH+V<&&%6_8hG@)74lvw?-S%* zChwEw&3~(S^gRAwm80jC%liy@pDFLx$h$(`ub1~5<$boi!}5NMyw8>Q`SMMoXiLQs~nxgBcbUj4Zlrt2egeV(o# z(KSNXC6It~AE#@Ot`E`mI9>0j>tc$`e!2$e>ZI#?bnT|=d`vKNH__EjR|8#tOV?F& zeS@xd&~*wY%5`+DrE4`^^>m$0*R6D&N!JJII+?CNql-PXb%L&^(J5NLN7oU$o}%l= zbcHz6&d|mG3^Vslx-O^d5xTCU>wdcafG&=wqxa5~mQJDsIIYbWe{^tv=pUd@Sq-Qe z+?2JxOXQYfvi;c@^5h;s+-gat@IdpvRCah(v%eWE+w1n~)wNXQM(G@hTKRb z4dhK=R(^j69~?`!zKWh@abSSD^#qZ3;q?x{X8=ow@FIa%!6aDv7!XFg3j1ZN{x=X8 zi2U{-nAGk(1+PB^lB4!K5K{XSgp_W@i`4#_$RJ)hYA%pF5hA}6uuc9pxE|zo`8)B- zQTw_1@f3hOGowg7&H2n)#r5Ar(~4BH?r z1pWq<_=F#F^xie47ZBOx=S;^pf%-3$6Mk0Heg97 zUJo-Be^uog@)bb-PZs=gZS#arkq+Few$`%l&cbUmQfQzknZM*K; zaH6ZNvp3!y?&$034eyGF;~k0KJHuVwaPMvF4Bye#+Zw)L?FHethwo_axjcLU80ks0 zxAj^VU9`5RyJhVyJzbp_wI&i)cYII0JKot6kM(xl8t=S0(bcmvylGSTg1axU;yo?R ziFmlBwYj^wg}K2qm*bwU_V%tj;=4^=l@~Dg3#u-#63yK`@mRdOyQ^E>VReU_d&Ab{ zVe4keTmRX48@RA8vUY~8yQ?e|`)=@)Xm18LJ@|j^`Y+5oyL!WWE|LPSUW2;B@7n6l z_O8~B=FZObyW_phZS7T61-V8RhhLVg7us|^nDZhAE&E%|@UT*TMl z;s1o^PjCq0!G6n{e*?xH`cLvg%Q_Rws`<5u3z)nZ0}gO#6Xe(*TGm)C{$Dj<@;z_^ zPPbduIv%3^|J_-25Kp*ikuZ@Aq`PLzx+ zn1G`tlCblR4lCdn+uPqCOT@cj-fD)q>(1Eil~z)=9+lB0R5rW+JgKvfmZU$Nfq=CvlEw zw6&l%yW2YV*0x-`Ef%b<2>dQLQ(|xA z`fImU#e)oooplG*yb@m?fUg8drjrId1g3?06$H_~v8mNZ#%Y-vjM4IV?H z9WAWNp<_X@IgyB?iBWWt96q)rrQjpSf@WK1PrSP~(qZ@xCp`#@y8t895XT(8!`PY@kB2+TA1Z%K7wo%#UPejS{X8y55d^Y`;X`S_jJbt zzHmFy0K=Aj+Tl#wAvDSDqr|;D_SykM}DYYif#Ey>9W`gsO znYJ6ZTQ{#JX??I{@f2-+xx`u#n0H4@@@*2azF4wy<@9e$td%S0@wQeJ zhW&F%AnErJc^=!s;9`Jm%WvY$**8{(D;vtg;qXLhxN_*6p|YCr(1}B5&YxL1TY`}J zH_xxAtSp~BAzU+G7EZ31EgdSIT@fByIbAk#a#*QY*2&oZJ-Z}nja2AhY1n=x&iypvw-|iL>{or;;QqeIvj(3x;h({7zof_h7}KAO4S(X?ha=u? zaDP8#*x>%Y%KsD``$lFTHa z=M2uhSmLK*T#z5`$r29(Kl@cl%ijl$7<_uahI0UiP@n!j>RX8;zQ2DLHMqY|hibDz zC!qe#zTZa-?(Y+RiFgS*1%Ax`t9W?eyawZj_}A4xCpW&YID_PO9$fmTcsziAD1d)H zfIkz!Bgjd;l-^JP=TWu7@HN=TF2rvS;D-Zv2z6Z;KNi407r>tk;ODF=PG1?oqXGO6 z0{EW<@V^h>j|T7`1@NZ>_=!B+B7a(6Jak@&Um3s~1y3T1{&og%{x5U{=GwbEkp4pf z{LcdTSO7m9z~wMb0l#w^p3Cpo0sPqj&I2+vcsAqNg6B#+Tk%BjT!rUqJRAcMhjLi! z8o)X{^?0tu^KLvix?pX?vmH+(o*VGoi05WJ@4>@^I}m?X6P{)~yYRdd&rUqM@x<}$ z!Gm87p5z9+uEW!Sry37_Nj?z4Rtuh23eRCHZ+lrrTj7d+u|g71D3*>?sDm)ZVr6Fu zGK(V>lUb~M#Y8U)s(UQhu`v-8PFO{?40{^Ab%dPB>6(N5~|5EA6D!G8ji&@0K$mD^n zjbS=Ztk@=<_b*)E=Y4Yn`gdE{FOe{}qR)pFE#;T=_WNyKg396@af?^&@CX>x>GQkU z(j{h*?HoR0)CHvRHWn*fa06S$GGIaqO#;R%ZwoLU0V9C%)zv?cCCu;s0^eWRFEYS` z*}S2m z2o7wdac?ak1-HkxSH`wip%XS-gTA;#4J>tF9&)Y?I|0iox$72-lxw%GTf|QBU|%Yu z+`xE63e)PoPQCGfEr?*pS*kQwu6y~aEw45(mzUl)+cpPgv84*MwbIDMYjW*lqUG7S z#Ab0L|;!UHU(R5jrHK(Q!LTk+ggi8|X zr&LaG==e7^F_GNX$6K9C>yk?^2t|3sl5j%|%Ht(Pg1ia?G|{|UPD5afBCq8LQzyNw zOOJYB0OuyATA%uQr4DI=Kdwu22AT*mEv&l0vZJahejCnI^~!0&NL$+rM7=oPwFk4o z)mL6=jz3^p+|?ehT;x-=Zn4+Ji@nw__S&%6t9p^wx=R;%UA&lyOR5&dx?~+o2QyGm T?I}{RMEETcDX$x0Bj(=$@v}rg literal 0 HcmV?d00001 diff --git a/doc/examples/json_pointer.cpp b/doc/examples/json_pointer.cpp index 140eac3b..3b23dfb2 100644 --- a/doc/examples/json_pointer.cpp +++ b/doc/examples/json_pointer.cpp @@ -19,9 +19,9 @@ int main() { json::json_pointer p9("foo"); } - catch (std::domain_error& e) + catch (json::parse_error& e) { - std::cout << "domain_error: " << e.what() << '\n'; + std::cout << e.what() << '\n'; } // error: JSON pointer uses escape symbol ~ not followed by 0 or 1 @@ -29,9 +29,9 @@ int main() { json::json_pointer p10("/foo/~"); } - catch (std::domain_error& e) + catch (json::parse_error& e) { - std::cout << "domain_error: " << e.what() << '\n'; + std::cout << e.what() << '\n'; } // error: JSON pointer uses escape symbol ~ not followed by 0 or 1 @@ -39,8 +39,8 @@ int main() { json::json_pointer p11("/foo/~3"); } - catch (std::domain_error& e) + catch (json::parse_error& e) { - std::cout << "domain_error: " << e.what() << '\n'; + std::cout << e.what() << '\n'; } } diff --git a/doc/examples/json_pointer.link b/doc/examples/json_pointer.link index 6602f0cf..78881dd9 100644 --- a/doc/examples/json_pointer.link +++ b/doc/examples/json_pointer.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/json_pointer.output b/doc/examples/json_pointer.output index b81c8a20..33e2cc68 100644 --- a/doc/examples/json_pointer.output +++ b/doc/examples/json_pointer.output @@ -1,3 +1,3 @@ -domain_error: JSON pointer must be empty or begin with '/' -domain_error: escape error: '~' must be followed with '0' or '1' -domain_error: escape error: '~' must be followed with '0' or '1' +[json.exception.parse_error.107] parse error at 1: JSON pointer must be empty or begin with '/' - was: 'foo' +[json.exception.parse_error.108] parse error: escape character '~' must be followed with '0' or '1' +[json.exception.parse_error.108] parse error: escape character '~' must be followed with '0' or '1' diff --git a/doc/examples/json_pointer.test b/doc/examples/json_pointer.test new file mode 100644 index 00000000..33e2cc68 --- /dev/null +++ b/doc/examples/json_pointer.test @@ -0,0 +1,3 @@ +[json.exception.parse_error.107] parse error at 1: JSON pointer must be empty or begin with '/' - was: 'foo' +[json.exception.parse_error.108] parse error: escape character '~' must be followed with '0' or '1' +[json.exception.parse_error.108] parse error: escape character '~' must be followed with '0' or '1' diff --git a/src/json.hpp b/src/json.hpp index f1462db8..4b2af732 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -301,7 +301,8 @@ Exceptions have ids 5xx. name / id | example massage | description ------------------------------ | --------------- | ------------------------- -json.exception.[other_error](@ref other_error).501 | "unsuccessful" | A JSON Patch operation 'test' failed. +json.exception.[other_error](@ref other_error).501 | "unsuccessful: {"op":"test","path":"/baz", + "value":"bar"}" | A JSON Patch operation 'test' failed. @since version 3.0.0 */ @@ -3652,7 +3653,7 @@ class basic_json @return copy of the JSON value, converted to type @a ValueType - @throw std::domain_error in case passed type @a ValueType is incompatible + @throw type_error.302 in case passed type @a ValueType is incompatible to JSON, thrown by @ref get() const @complexity Linear in the size of the JSON value. @@ -4088,7 +4089,7 @@ class basic_json @return const reference to the element at key @a key - @throw std::domain_error if JSON is not an object; example: `"cannot use + @throw type_error.305 if JSON is not an object; example: `"cannot use operator[] with null"` @complexity Logarithmic in the size of the container. @@ -4365,7 +4366,7 @@ class basic_json assertions**). @post The JSON value remains unchanged. - @throw std::out_of_range when called on `null` value + @throw invalid_iterator.214 when called on `null` value @liveexample{The following code shows an example for `front()`.,front} @@ -4408,7 +4409,7 @@ class basic_json assertions**). @post The JSON value remains unchanged. - @throw std::out_of_range when called on `null` value. + @throw invalid_iterator.214 when called on `null` value. @liveexample{The following code shows an example for `back()`.,back} @@ -4698,7 +4699,7 @@ class basic_json @throw type_error.307 when called on a type other than JSON object; example: `"cannot use erase() with null"` - @throw std::out_of_range when `idx >= size()`; example: `"array index 17 + @throw out_of_range.401 when `idx >= size()`; example: `"array index 17 is out of range"` @complexity Linear in distance between @a idx and the end of the container. @@ -7032,6 +7033,8 @@ class basic_json @throw parse_error.101 if a parse error occurs; example: `""unexpected end of input; expected string literal""` + @throw parse_error.102 if to_unicode fails or surrogate error + @throw parse_error.103 if to_unicode fails @complexity Linear in the length of the input. The parser is a predictive LL(1) parser. The complexity can be higher if the parser callback function @@ -7063,6 +7066,10 @@ class basic_json @return result of the deserialization + @throw parse_error.101 in case of an unexpected token + @throw parse_error.102 if to_unicode fails or surrogate error + @throw parse_error.103 if to_unicode fails + @complexity Linear in the length of the input. The parser is a predictive LL(1) parser. The complexity can be higher if the parser callback function @a cb has a super-linear complexity. @@ -7099,6 +7106,11 @@ class basic_json @return result of the deserialization + @throw parse_error.101 in case of an unexpected token + @throw parse_error.102 if to_unicode fails or surrogate error + @throw parse_error.103 if to_unicode fails + @throw parse_error.111 if input stream is in a bad state + @complexity Linear in the length of the input. The parser is a predictive LL(1) parser. The complexity can be higher if the parser callback function @a cb has a super-linear complexity. @@ -7158,6 +7170,10 @@ class basic_json @return result of the deserialization + @throw parse_error.101 in case of an unexpected token + @throw parse_error.102 if to_unicode fails or surrogate error + @throw parse_error.103 if to_unicode fails + @complexity Linear in the length of the input. The parser is a predictive LL(1) parser. The complexity can be higher if the parser callback function @a cb has a super-linear complexity. @@ -7228,6 +7244,10 @@ class basic_json @return result of the deserialization + @throw parse_error.101 in case of an unexpected token + @throw parse_error.102 if to_unicode fails or surrogate error + @throw parse_error.103 if to_unicode fails + @complexity Linear in the length of the input. The parser is a predictive LL(1) parser. The complexity can be higher if the parser callback function @a cb has a super-linear complexity. @@ -7260,7 +7280,10 @@ class basic_json @param[in,out] i input stream to read a serialized JSON value from @param[in,out] j JSON value to write the deserialized input to - @throw std::invalid_argument in case of parse errors + @throw parse_error.101 in case of an unexpected token + @throw parse_error.102 if to_unicode fails or surrogate error + @throw parse_error.103 if to_unicode fails + @throw parse_error.111 if input stream is in a bad state @complexity Linear in the length of the input. The parser is a predictive LL(1) parser. @@ -7963,9 +7986,9 @@ class basic_json @return deserialized JSON value - @throw std::invalid_argument if unsupported features from MessagePack were + @throw parse_error.110 if the given vector ends prematurely + @throw parse_error.112 if unsupported features from MessagePack were used in the given vector @a v or if the input is not valid MessagePack - @throw std::out_of_range if the given vector ends prematurely @sa https://github.com/msgpack/msgpack/blob/master/spec.md */ @@ -8205,9 +8228,9 @@ class basic_json @return deserialized JSON value - @throw std::invalid_argument if unsupported features from CBOR were used in - the given vector @a v or if the input is not valid CBOR - @throw std::out_of_range if the given vector ends prematurely + @throw parse_error.110 if the given vector ends prematurely + @throw parse_error.112 if unsupported features from CBOR were + used in the given vector @a v or if the input is not valid CBOR @sa https://tools.ietf.org/html/rfc7049 */ @@ -8729,9 +8752,9 @@ class basic_json @param[in] start_index the index to start reading from @a v (0 by default) @return deserialized JSON value - @throw std::invalid_argument if unsupported features from MessagePack were + @throw parse_error.110 if the given vector ends prematurely + @throw parse_error.112 if unsupported features from MessagePack were used in the given vector @a v or if the input is not valid MessagePack - @throw std::out_of_range if the given vector ends prematurely @complexity Linear in the size of the byte vector @a v. @@ -8792,9 +8815,9 @@ class basic_json @param[in] start_index the index to start reading from @a v (0 by default) @return deserialized JSON value - @throw std::invalid_argument if unsupported features from CBOR were used in - the given vector @a v or if the input is not valid MessagePack - @throw std::out_of_range if the given vector ends prematurely + @throw parse_error.110 if the given vector ends prematurely + @throw parse_error.112 if unsupported features from CBOR were + used in the given vector @a v or if the input is not valid CBOR @complexity Linear in the size of the byte vector @a v. @@ -9908,7 +9931,10 @@ class basic_json m_limit = m_content + len; } - /// a lexer from an input stream + /*! + @brief a lexer from an input stream + @throw parse_error.111 if input stream is in a bad state + */ explicit lexer(std::istream& s) : m_stream(&s), m_line_buffer() { @@ -11288,7 +11314,7 @@ basic_json_parser_74: @return string value of current token without opening and closing quotes - @throw parse_error.102 if to_unicode fails + @throw parse_error.102 if to_unicode fails or surrogate error @throw parse_error.103 if to_unicode fails */ string_t get_string() const @@ -11685,7 +11711,10 @@ basic_json_parser_74: m_lexer(reinterpret_cast(buff), std::strlen(buff)) {} - /// a parser reading from an input stream + /*! + @brief a parser reading from an input stream + @throw parse_error.111 if input stream is in a bad state + */ parser(std::istream& is, const parser_callback_t cb = nullptr) : callback(cb), m_lexer(is) {} @@ -11701,7 +11730,12 @@ basic_json_parser_74: static_cast(std::distance(first, last))) {} - /// public parser interface + /*! + @brief public parser interface + @throw parse_error.101 in case of an unexpected token + @throw parse_error.102 if to_unicode fails or surrogate error + @throw parse_error.103 if to_unicode fails + */ basic_json parse() { // read first token @@ -11718,7 +11752,12 @@ basic_json_parser_74: } private: - /// the actual parser + /*! + @brief the actual parser + @throw parse_error.101 in case of an unexpected token + @throw parse_error.102 if to_unicode fails or surrogate error + @throw parse_error.103 if to_unicode fails + */ basic_json parse_internal(bool keep) { auto result = basic_json(value_t::discarded); @@ -11921,6 +11960,9 @@ basic_json_parser_74: return last_token; } + /*! + @throw parse_error.101 if expected token did not occur + */ void expect(typename lexer::token_type t) const { if (t != last_token) @@ -11934,6 +11976,9 @@ basic_json_parser_74: } } + /*! + @throw parse_error.101 if unexpected token occurred + */ void unexpect(typename lexer::token_type t) const { if (t == last_token) @@ -11985,12 +12030,12 @@ basic_json_parser_74: empty string is assumed which references the whole JSON value - @throw parse_error.107 if reference token is nonempty and does not - begin with a slash (`/`); example: `"JSON pointer must be empty or - begin with / - was: 'foo'"` - @throw parse_error.108 if a tilde (`~`) is not followed by `0` - (representing `~`) or `1` (representing `/`); example: `"escape - character '~' must be followed with '0' or '1'"` + @throw parse_error.107 if the given JSON pointer @a s is nonempty and + does not begin with a slash (`/`); see example below + + @throw parse_error.108 if a tilde (`~`) in the given JSON pointer @a s + is not followed by `0` (representing `~`) or `1` (representing `/`); + see example below @liveexample{The example shows the construction several valid JSON pointers as well as the exceptional behavior.,json_pointer} @@ -12033,7 +12078,10 @@ basic_json_parser_74: } private: - /// remove and return last reference pointer + /*! + @brief remove and return last reference pointer + @throw out_of_range.405 if JSON pointer has no parent + */ std::string pop_back() { if (is_root()) @@ -12068,6 +12116,9 @@ basic_json_parser_74: @brief create and return a reference to the pointed to value @complexity Linear in the number of reference tokens. + + @throw parse_error.109 if array index is not a number + @throw type_error.313 if value cannot be unflattened */ reference get_and_create(reference j) const { @@ -12147,9 +12198,9 @@ basic_json_parser_74: @complexity Linear in the length of the JSON pointer. - @throw out_of_range.404 if the JSON pointer can not be resolved - @throw parse_error.106 if an array index begins with '0' - @throw std::invalid_argument if an array index was not a number + @throw parse_error.106 if an array index begins with '0' + @throw parse_error.109 if an array index was not a number + @throw out_of_range.404 if the JSON pointer can not be resolved */ reference get_unchecked(pointer ptr) const { @@ -12225,6 +12276,12 @@ basic_json_parser_74: return *ptr; } + /*! + @throw parse_error.106 if an array index begins with '0' + @throw parse_error.109 if an array index was not a number + @throw out_of_range.402 if the array index '-' is used + @throw out_of_range.404 if the JSON pointer can not be resolved + */ reference get_checked(pointer ptr) const { for (const auto& reference_token : reference_tokens) @@ -12283,6 +12340,11 @@ basic_json_parser_74: @return const reference to the JSON value pointed to by the JSON pointer + + @throw parse_error.106 if an array index begins with '0' + @throw parse_error.109 if an array index was not a number + @throw out_of_range.402 if the array index '-' is used + @throw out_of_range.404 if the JSON pointer can not be resolved */ const_reference get_unchecked(const_pointer ptr) const { @@ -12335,6 +12397,12 @@ basic_json_parser_74: return *ptr; } + /*! + @throw parse_error.106 if an array index begins with '0' + @throw parse_error.109 if an array index was not a number + @throw out_of_range.402 if the array index '-' is used + @throw out_of_range.404 if the JSON pointer can not be resolved + */ const_reference get_checked(const_pointer ptr) const { for (const auto& reference_token : reference_tokens) @@ -12386,7 +12454,15 @@ basic_json_parser_74: return *ptr; } - /// split the string input to reference tokens + /*! + @brief split the string input to reference tokens + + @note This function is only called by the json_pointer constructor. + All exceptions below are documented there. + + @throw parse_error.107 if the pointer is not empty or begins with '/' + @throw parse_error.108 if character '~' is not followed by '0' or '1' + */ static std::vector split(const std::string& reference_string) { std::vector result; @@ -12447,7 +12523,6 @@ basic_json_parser_74: return result; } - private: /*! @brief replace all occurrences of a substring by another string @@ -12456,7 +12531,8 @@ basic_json_parser_74: @param[in] f the substring to replace with @a t @param[in] t the string to replace @a f - @pre The search string @a f must not be empty. + @pre The search string @a f must not be empty. **This precondition is + enforced with an assertion.** @since version 2.0.0 */ @@ -12556,6 +12632,11 @@ basic_json_parser_74: @param[in] value flattened JSON @return unflattened JSON + + @throw parse_error.109 if array index is not a number + @throw type_error.314 if value is not an object + @throw type_error.315 if object values are not primitive + @throw type_error.313 if value cannot be unflattened */ static basic_json unflatten(const basic_json& value) { @@ -12585,7 +12666,6 @@ basic_json_parser_74: return result; } - private: friend bool operator==(json_pointer const& lhs, json_pointer const& rhs) noexcept { @@ -12634,9 +12714,9 @@ basic_json_parser_74: @complexity Constant. - @throw std::out_of_range if the JSON pointer can not be resolved - @throw std::domain_error if an array index begins with '0' - @throw std::invalid_argument if an array index was not a number + @throw parse_error.106 if an array index begins with '0' + @throw parse_error.109 if an array index was not a number + @throw out_of_range.404 if the JSON pointer can not be resolved @liveexample{The behavior is shown in the example.,operatorjson_pointer} @@ -12661,9 +12741,10 @@ basic_json_parser_74: @complexity Constant. - @throw std::out_of_range if the JSON pointer can not be resolved - @throw std::domain_error if an array index begins with '0' - @throw std::invalid_argument if an array index was not a number + @throw parse_error.106 if an array index begins with '0' + @throw parse_error.109 if an array index was not a number + @throw out_of_range.402 if the array index '-' is used + @throw out_of_range.404 if the JSON pointer can not be resolved @liveexample{The behavior is shown in the example.,operatorjson_pointer_const} @@ -12686,9 +12767,10 @@ basic_json_parser_74: @complexity Constant. - @throw std::out_of_range if the JSON pointer can not be resolved - @throw std::domain_error if an array index begins with '0' - @throw std::invalid_argument if an array index was not a number + @throw parse_error.106 if an array index begins with '0' + @throw parse_error.109 if an array index was not a number + @throw out_of_range.402 if the array index '-' is used + @throw out_of_range.404 if the JSON pointer can not be resolved @liveexample{The behavior is shown in the example.,at_json_pointer} @@ -12711,9 +12793,10 @@ basic_json_parser_74: @complexity Constant. - @throw std::out_of_range if the JSON pointer can not be resolved - @throw std::domain_error if an array index begins with '0' - @throw std::invalid_argument if an array index was not a number + @throw parse_error.106 if an array index begins with '0' + @throw parse_error.109 if an array index was not a number + @throw out_of_range.402 if the array index '-' is used + @throw out_of_range.404 if the JSON pointer can not be resolved @liveexample{The behavior is shown in the example.,at_json_pointer_const} @@ -12773,6 +12856,9 @@ basic_json_parser_74: @complexity Linear in the size the JSON value. + @throw type_error.314 if value is not an object + @throw type_error.315 if object values are not primitve + @liveexample{The following code shows how a flattened JSON object is unflattened into the original nested JSON object.,unflatten} @@ -12810,13 +12896,22 @@ basic_json_parser_74: any case, the original value is not changed: the patch is applied to a copy of the value. + @throw parse_error.104 if the JSON patch does not consist of an array of + objects + + @throw parse_error.105 if the JSON patch is malformed (e.g., mandatory + attributes are missing); example: `"operation add must have member path"` + + @throw out_of_range.401 if an array index is out of range. + @throw out_of_range.403 if a JSON pointer inside the patch could not be resolved successfully in the current JSON value; example: `"key baz not found"` - @throw invalid_argument if the JSON patch is malformed (e.g., mandatory - attributes are missing); example: `"operation add must have member path"` - @throw parse_error.104 if the JSON patch does not consist of an array of - objects + + @throw out_of_range.405 if JSON pointer has no parent ("add", "remove", + "move") + + @throw other_error.501 if "test" operation was unsuccessful @complexity Linear in the size of the JSON value and the length of the JSON patch. As usually only a fraction of the JSON value is affected by diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 3044cf6b..d2f6346f 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -3653,7 +3653,7 @@ class basic_json @return copy of the JSON value, converted to type @a ValueType - @throw std::domain_error in case passed type @a ValueType is incompatible + @throw type_error.302 in case passed type @a ValueType is incompatible to JSON, thrown by @ref get() const @complexity Linear in the size of the JSON value. @@ -4089,7 +4089,7 @@ class basic_json @return const reference to the element at key @a key - @throw std::domain_error if JSON is not an object; example: `"cannot use + @throw type_error.305 if JSON is not an object; example: `"cannot use operator[] with null"` @complexity Logarithmic in the size of the container. @@ -4366,7 +4366,7 @@ class basic_json assertions**). @post The JSON value remains unchanged. - @throw std::out_of_range when called on `null` value + @throw invalid_iterator.214 when called on `null` value @liveexample{The following code shows an example for `front()`.,front} @@ -4409,7 +4409,7 @@ class basic_json assertions**). @post The JSON value remains unchanged. - @throw std::out_of_range when called on `null` value. + @throw invalid_iterator.214 when called on `null` value. @liveexample{The following code shows an example for `back()`.,back} @@ -4699,7 +4699,7 @@ class basic_json @throw type_error.307 when called on a type other than JSON object; example: `"cannot use erase() with null"` - @throw std::out_of_range when `idx >= size()`; example: `"array index 17 + @throw out_of_range.401 when `idx >= size()`; example: `"array index 17 is out of range"` @complexity Linear in distance between @a idx and the end of the container. @@ -7033,6 +7033,8 @@ class basic_json @throw parse_error.101 if a parse error occurs; example: `""unexpected end of input; expected string literal""` + @throw parse_error.102 if to_unicode fails or surrogate error + @throw parse_error.103 if to_unicode fails @complexity Linear in the length of the input. The parser is a predictive LL(1) parser. The complexity can be higher if the parser callback function @@ -7064,6 +7066,10 @@ class basic_json @return result of the deserialization + @throw parse_error.101 in case of an unexpected token + @throw parse_error.102 if to_unicode fails or surrogate error + @throw parse_error.103 if to_unicode fails + @complexity Linear in the length of the input. The parser is a predictive LL(1) parser. The complexity can be higher if the parser callback function @a cb has a super-linear complexity. @@ -7100,6 +7106,11 @@ class basic_json @return result of the deserialization + @throw parse_error.101 in case of an unexpected token + @throw parse_error.102 if to_unicode fails or surrogate error + @throw parse_error.103 if to_unicode fails + @throw parse_error.111 if input stream is in a bad state + @complexity Linear in the length of the input. The parser is a predictive LL(1) parser. The complexity can be higher if the parser callback function @a cb has a super-linear complexity. @@ -7159,6 +7170,10 @@ class basic_json @return result of the deserialization + @throw parse_error.101 in case of an unexpected token + @throw parse_error.102 if to_unicode fails or surrogate error + @throw parse_error.103 if to_unicode fails + @complexity Linear in the length of the input. The parser is a predictive LL(1) parser. The complexity can be higher if the parser callback function @a cb has a super-linear complexity. @@ -7229,6 +7244,10 @@ class basic_json @return result of the deserialization + @throw parse_error.101 in case of an unexpected token + @throw parse_error.102 if to_unicode fails or surrogate error + @throw parse_error.103 if to_unicode fails + @complexity Linear in the length of the input. The parser is a predictive LL(1) parser. The complexity can be higher if the parser callback function @a cb has a super-linear complexity. @@ -7261,7 +7280,10 @@ class basic_json @param[in,out] i input stream to read a serialized JSON value from @param[in,out] j JSON value to write the deserialized input to - @throw std::invalid_argument in case of parse errors + @throw parse_error.101 in case of an unexpected token + @throw parse_error.102 if to_unicode fails or surrogate error + @throw parse_error.103 if to_unicode fails + @throw parse_error.111 if input stream is in a bad state @complexity Linear in the length of the input. The parser is a predictive LL(1) parser. @@ -7964,9 +7986,9 @@ class basic_json @return deserialized JSON value - @throw std::invalid_argument if unsupported features from MessagePack were + @throw parse_error.110 if the given vector ends prematurely + @throw parse_error.112 if unsupported features from MessagePack were used in the given vector @a v or if the input is not valid MessagePack - @throw std::out_of_range if the given vector ends prematurely @sa https://github.com/msgpack/msgpack/blob/master/spec.md */ @@ -8206,9 +8228,9 @@ class basic_json @return deserialized JSON value - @throw std::invalid_argument if unsupported features from CBOR were used in - the given vector @a v or if the input is not valid CBOR - @throw std::out_of_range if the given vector ends prematurely + @throw parse_error.110 if the given vector ends prematurely + @throw parse_error.112 if unsupported features from CBOR were + used in the given vector @a v or if the input is not valid CBOR @sa https://tools.ietf.org/html/rfc7049 */ @@ -8730,9 +8752,9 @@ class basic_json @param[in] start_index the index to start reading from @a v (0 by default) @return deserialized JSON value - @throw std::invalid_argument if unsupported features from MessagePack were + @throw parse_error.110 if the given vector ends prematurely + @throw parse_error.112 if unsupported features from MessagePack were used in the given vector @a v or if the input is not valid MessagePack - @throw std::out_of_range if the given vector ends prematurely @complexity Linear in the size of the byte vector @a v. @@ -8793,9 +8815,9 @@ class basic_json @param[in] start_index the index to start reading from @a v (0 by default) @return deserialized JSON value - @throw std::invalid_argument if unsupported features from CBOR were used in - the given vector @a v or if the input is not valid MessagePack - @throw std::out_of_range if the given vector ends prematurely + @throw parse_error.110 if the given vector ends prematurely + @throw parse_error.112 if unsupported features from CBOR were + used in the given vector @a v or if the input is not valid CBOR @complexity Linear in the size of the byte vector @a v. @@ -9909,7 +9931,10 @@ class basic_json m_limit = m_content + len; } - /// a lexer from an input stream + /*! + @brief a lexer from an input stream + @throw parse_error.111 if input stream is in a bad state + */ explicit lexer(std::istream& s) : m_stream(&s), m_line_buffer() { @@ -10322,7 +10347,7 @@ class basic_json @return string value of current token without opening and closing quotes - @throw parse_error.102 if to_unicode fails + @throw parse_error.102 if to_unicode fails or surrogate error @throw parse_error.103 if to_unicode fails */ string_t get_string() const @@ -10719,7 +10744,10 @@ class basic_json m_lexer(reinterpret_cast(buff), std::strlen(buff)) {} - /// a parser reading from an input stream + /*! + @brief a parser reading from an input stream + @throw parse_error.111 if input stream is in a bad state + */ parser(std::istream& is, const parser_callback_t cb = nullptr) : callback(cb), m_lexer(is) {} @@ -10735,7 +10763,12 @@ class basic_json static_cast(std::distance(first, last))) {} - /// public parser interface + /*! + @brief public parser interface + @throw parse_error.101 in case of an unexpected token + @throw parse_error.102 if to_unicode fails or surrogate error + @throw parse_error.103 if to_unicode fails + */ basic_json parse() { // read first token @@ -10752,7 +10785,12 @@ class basic_json } private: - /// the actual parser + /*! + @brief the actual parser + @throw parse_error.101 in case of an unexpected token + @throw parse_error.102 if to_unicode fails or surrogate error + @throw parse_error.103 if to_unicode fails + */ basic_json parse_internal(bool keep) { auto result = basic_json(value_t::discarded); @@ -10955,6 +10993,9 @@ class basic_json return last_token; } + /*! + @throw parse_error.101 if expected token did not occur + */ void expect(typename lexer::token_type t) const { if (t != last_token) @@ -10968,6 +11009,9 @@ class basic_json } } + /*! + @throw parse_error.101 if unexpected token occurred + */ void unexpect(typename lexer::token_type t) const { if (t == last_token) @@ -11019,12 +11063,12 @@ class basic_json empty string is assumed which references the whole JSON value - @throw parse_error.107 if reference token is nonempty and does not - begin with a slash (`/`); example: `"JSON pointer must be empty or - begin with / - was: 'foo'"` - @throw parse_error.108 if a tilde (`~`) is not followed by `0` - (representing `~`) or `1` (representing `/`); example: `"escape - character '~' must be followed with '0' or '1'"` + @throw parse_error.107 if the given JSON pointer @a s is nonempty and + does not begin with a slash (`/`); see example below + + @throw parse_error.108 if a tilde (`~`) in the given JSON pointer @a s + is not followed by `0` (representing `~`) or `1` (representing `/`); + see example below @liveexample{The example shows the construction several valid JSON pointers as well as the exceptional behavior.,json_pointer} @@ -11067,7 +11111,10 @@ class basic_json } private: - /// remove and return last reference pointer + /*! + @brief remove and return last reference pointer + @throw out_of_range.405 if JSON pointer has no parent + */ std::string pop_back() { if (is_root()) @@ -11102,6 +11149,9 @@ class basic_json @brief create and return a reference to the pointed to value @complexity Linear in the number of reference tokens. + + @throw parse_error.109 if array index is not a number + @throw type_error.313 if value cannot be unflattened */ reference get_and_create(reference j) const { @@ -11181,9 +11231,9 @@ class basic_json @complexity Linear in the length of the JSON pointer. - @throw out_of_range.404 if the JSON pointer can not be resolved - @throw parse_error.106 if an array index begins with '0' - @throw std::invalid_argument if an array index was not a number + @throw parse_error.106 if an array index begins with '0' + @throw parse_error.109 if an array index was not a number + @throw out_of_range.404 if the JSON pointer can not be resolved */ reference get_unchecked(pointer ptr) const { @@ -11259,6 +11309,12 @@ class basic_json return *ptr; } + /*! + @throw parse_error.106 if an array index begins with '0' + @throw parse_error.109 if an array index was not a number + @throw out_of_range.402 if the array index '-' is used + @throw out_of_range.404 if the JSON pointer can not be resolved + */ reference get_checked(pointer ptr) const { for (const auto& reference_token : reference_tokens) @@ -11317,6 +11373,11 @@ class basic_json @return const reference to the JSON value pointed to by the JSON pointer + + @throw parse_error.106 if an array index begins with '0' + @throw parse_error.109 if an array index was not a number + @throw out_of_range.402 if the array index '-' is used + @throw out_of_range.404 if the JSON pointer can not be resolved */ const_reference get_unchecked(const_pointer ptr) const { @@ -11369,6 +11430,12 @@ class basic_json return *ptr; } + /*! + @throw parse_error.106 if an array index begins with '0' + @throw parse_error.109 if an array index was not a number + @throw out_of_range.402 if the array index '-' is used + @throw out_of_range.404 if the JSON pointer can not be resolved + */ const_reference get_checked(const_pointer ptr) const { for (const auto& reference_token : reference_tokens) @@ -11420,7 +11487,15 @@ class basic_json return *ptr; } - /// split the string input to reference tokens + /*! + @brief split the string input to reference tokens + + @note This function is only called by the json_pointer constructor. + All exceptions below are documented there. + + @throw parse_error.107 if the pointer is not empty or begins with '/' + @throw parse_error.108 if character '~' is not followed by '0' or '1' + */ static std::vector split(const std::string& reference_string) { std::vector result; @@ -11481,7 +11556,6 @@ class basic_json return result; } - private: /*! @brief replace all occurrences of a substring by another string @@ -11490,7 +11564,8 @@ class basic_json @param[in] f the substring to replace with @a t @param[in] t the string to replace @a f - @pre The search string @a f must not be empty. + @pre The search string @a f must not be empty. **This precondition is + enforced with an assertion.** @since version 2.0.0 */ @@ -11590,6 +11665,11 @@ class basic_json @param[in] value flattened JSON @return unflattened JSON + + @throw parse_error.109 if array index is not a number + @throw type_error.314 if value is not an object + @throw type_error.315 if object values are not primitive + @throw type_error.313 if value cannot be unflattened */ static basic_json unflatten(const basic_json& value) { @@ -11619,7 +11699,6 @@ class basic_json return result; } - private: friend bool operator==(json_pointer const& lhs, json_pointer const& rhs) noexcept { @@ -11668,9 +11747,9 @@ class basic_json @complexity Constant. - @throw std::out_of_range if the JSON pointer can not be resolved - @throw std::domain_error if an array index begins with '0' - @throw std::invalid_argument if an array index was not a number + @throw parse_error.106 if an array index begins with '0' + @throw parse_error.109 if an array index was not a number + @throw out_of_range.404 if the JSON pointer can not be resolved @liveexample{The behavior is shown in the example.,operatorjson_pointer} @@ -11695,9 +11774,10 @@ class basic_json @complexity Constant. - @throw std::out_of_range if the JSON pointer can not be resolved - @throw std::domain_error if an array index begins with '0' - @throw std::invalid_argument if an array index was not a number + @throw parse_error.106 if an array index begins with '0' + @throw parse_error.109 if an array index was not a number + @throw out_of_range.402 if the array index '-' is used + @throw out_of_range.404 if the JSON pointer can not be resolved @liveexample{The behavior is shown in the example.,operatorjson_pointer_const} @@ -11720,9 +11800,10 @@ class basic_json @complexity Constant. - @throw std::out_of_range if the JSON pointer can not be resolved - @throw std::domain_error if an array index begins with '0' - @throw std::invalid_argument if an array index was not a number + @throw parse_error.106 if an array index begins with '0' + @throw parse_error.109 if an array index was not a number + @throw out_of_range.402 if the array index '-' is used + @throw out_of_range.404 if the JSON pointer can not be resolved @liveexample{The behavior is shown in the example.,at_json_pointer} @@ -11745,9 +11826,10 @@ class basic_json @complexity Constant. - @throw std::out_of_range if the JSON pointer can not be resolved - @throw std::domain_error if an array index begins with '0' - @throw std::invalid_argument if an array index was not a number + @throw parse_error.106 if an array index begins with '0' + @throw parse_error.109 if an array index was not a number + @throw out_of_range.402 if the array index '-' is used + @throw out_of_range.404 if the JSON pointer can not be resolved @liveexample{The behavior is shown in the example.,at_json_pointer_const} @@ -11807,6 +11889,9 @@ class basic_json @complexity Linear in the size the JSON value. + @throw type_error.314 if value is not an object + @throw type_error.315 if object values are not primitve + @liveexample{The following code shows how a flattened JSON object is unflattened into the original nested JSON object.,unflatten} @@ -11844,13 +11929,22 @@ class basic_json any case, the original value is not changed: the patch is applied to a copy of the value. + @throw parse_error.104 if the JSON patch does not consist of an array of + objects + + @throw parse_error.105 if the JSON patch is malformed (e.g., mandatory + attributes are missing); example: `"operation add must have member path"` + + @throw out_of_range.401 if an array index is out of range. + @throw out_of_range.403 if a JSON pointer inside the patch could not be resolved successfully in the current JSON value; example: `"key baz not found"` - @throw invalid_argument if the JSON patch is malformed (e.g., mandatory - attributes are missing); example: `"operation add must have member path"` - @throw parse_error.104 if the JSON patch does not consist of an array of - objects + + @throw out_of_range.405 if JSON pointer has no parent ("add", "remove", + "move") + + @throw other_error.501 if "test" operation was unsuccessful @complexity Linear in the size of the JSON value and the length of the JSON patch. As usually only a fraction of the JSON value is affected by From 1ab580d6e99aab3b3c8162354a6faf340db5ee96 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Wed, 8 Mar 2017 23:12:13 +0100 Subject: [PATCH 055/530] :memo: more documentation for the new exceptions --- doc/Makefile | 2 + doc/examples/at__size_type_const.cpp | 4 +- doc/examples/at__size_type_const.link | 2 +- doc/examples/at__size_type_const.output | 2 +- doc/examples/basic_json__InputIt_InputIt.cpp | 10 + doc/examples/basic_json__InputIt_InputIt.link | 2 +- .../basic_json__InputIt_InputIt.output | 1 + doc/examples/object.cpp | 12 +- doc/examples/object.link | 2 +- doc/examples/object.output | 1 + doc/examples/operator__ValueType.cpp | 10 + doc/examples/operator__ValueType.link | 2 +- doc/examples/operator__ValueType.output | 1 + src/json.hpp | 224 +++++++++--------- src/json.hpp.re2c | 224 +++++++++--------- 15 files changed, 269 insertions(+), 230 deletions(-) diff --git a/doc/Makefile b/doc/Makefile index cd23a9f7..c900bdea 100644 --- a/doc/Makefile +++ b/doc/Makefile @@ -59,6 +59,8 @@ doxygen: create_output create_links $(SED) -i 's@< ObjectType, ArrayType, StringType, BooleanType, NumberIntegerType, NumberUnsignedType, NumberFloatType, AllocatorType, JSONSerializer >@@g' html/*.html $(SED) -i 's@< ObjectType, ArrayType, StringType, BooleanType, NumberIntegerType, NumberUnsignedType, NumberFloatType, AllocatorType JSONSerializer >@@g' html/*.html $(SED) -i 's@template<template< typename U, typename V, typename... Args > class ObjectType = std::map, template< typename U, typename... Args > class ArrayType = std::vector, class StringType = std::string, class BooleanType = bool, class NumberIntegerType = std::int64_t, class NumberUnsignedType = std::uint64_t, class NumberFloatType = double, template< typename U > class AllocatorType = std::allocator, template< typename T, typename SFINAE=void > class JSONSerializer = adl_serializer>@@g' html/*.html + $(SED) -i 's@< ObjectType, ArrayType, StringType, BooleanType, NumberIntegerType, NumberUnsignedType, NumberFloatType, AllocatorType, JSONSerializer >@@g' html/*.html + $(SED) -i 's@< ObjectType, ArrayType, StringType, BooleanType, NumberIntegerType, NumberUnsignedType, NumberFloatType, AllocatorType, JSONSerializer >@@g' html/*.html upload: clean doxygen check_output cd html ; ../scripts/git-update-ghpages nlohmann/json diff --git a/doc/examples/at__size_type_const.cpp b/doc/examples/at__size_type_const.cpp index c29db4fb..88d28be6 100644 --- a/doc/examples/at__size_type_const.cpp +++ b/doc/examples/at__size_type_const.cpp @@ -15,8 +15,8 @@ int main() { std::cout << array.at(5) << '\n'; } - catch (json::out_of_range) + catch (const json::out_of_range& e) { - std::cout << "out of range" << '\n'; + std::cout << e.what() << '\n'; } } diff --git a/doc/examples/at__size_type_const.link b/doc/examples/at__size_type_const.link index c5f66fe0..c703c5d9 100644 --- a/doc/examples/at__size_type_const.link +++ b/doc/examples/at__size_type_const.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/at__size_type_const.output b/doc/examples/at__size_type_const.output index d0b7a66a..e52b6cc0 100644 --- a/doc/examples/at__size_type_const.output +++ b/doc/examples/at__size_type_const.output @@ -1,2 +1,2 @@ "third" -out of range +[json.exception.out_of_range.401] array index 5 is out of range diff --git a/doc/examples/basic_json__InputIt_InputIt.cpp b/doc/examples/basic_json__InputIt_InputIt.cpp index 86a0faf8..ce07e07b 100644 --- a/doc/examples/basic_json__InputIt_InputIt.cpp +++ b/doc/examples/basic_json__InputIt_InputIt.cpp @@ -18,4 +18,14 @@ int main() std::cout << j_array_range << '\n'; std::cout << j_number_range << '\n'; std::cout << j_object_range << '\n'; + + // example for an exception + try + { + json j_invalid(j_number.begin() + 1, j_number.end()); + } + catch (json::invalid_iterator& e) + { + std::cout << e.what() << '\n'; + } } diff --git a/doc/examples/basic_json__InputIt_InputIt.link b/doc/examples/basic_json__InputIt_InputIt.link index ae46528c..d360496a 100644 --- a/doc/examples/basic_json__InputIt_InputIt.link +++ b/doc/examples/basic_json__InputIt_InputIt.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/basic_json__InputIt_InputIt.output b/doc/examples/basic_json__InputIt_InputIt.output index 36714200..bfb01778 100644 --- a/doc/examples/basic_json__InputIt_InputIt.output +++ b/doc/examples/basic_json__InputIt_InputIt.output @@ -1,3 +1,4 @@ ["bravo","charly"] 42 {"one":"eins"} +[json.exception.invalid_iterator.204] iterators out of range diff --git a/doc/examples/object.cpp b/doc/examples/object.cpp index 54f3ca3b..5b8ba589 100644 --- a/doc/examples/object.cpp +++ b/doc/examples/object.cpp @@ -8,10 +8,20 @@ int main() json j_no_init_list = json::object(); json j_empty_init_list = json::object({}); json j_list_of_pairs = json::object({ {"one", 1}, {"two", 2} }); - //json j_invalid_list = json::object({ "one", 1 }); // would throw // serialize the JSON objects std::cout << j_no_init_list << '\n'; std::cout << j_empty_init_list << '\n'; std::cout << j_list_of_pairs << '\n'; + + // example for an exception + try + { + // can only create an object from a list of pairs + json j_invalid_object = json::object({{ "one", 1, 2 }}); + } + catch (json::type_error& e) + { + std::cout << e.what() << '\n'; + } } diff --git a/doc/examples/object.link b/doc/examples/object.link index fd0f424b..d9e63e53 100644 --- a/doc/examples/object.link +++ b/doc/examples/object.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/object.output b/doc/examples/object.output index f6c27ee8..1a1d8140 100644 --- a/doc/examples/object.output +++ b/doc/examples/object.output @@ -1,3 +1,4 @@ {} {} {"one":1,"two":2} +[json.exception.type_error.301] cannot create object from initializer list diff --git a/doc/examples/operator__ValueType.cpp b/doc/examples/operator__ValueType.cpp index 9307e69b..77af700f 100644 --- a/doc/examples/operator__ValueType.cpp +++ b/doc/examples/operator__ValueType.cpp @@ -46,4 +46,14 @@ int main() { std::cout << i.first << ": " << i.second << '\n'; } + + // example for an exception + try + { + bool v1 = json_types["string"]; + } + catch (json::type_error& e) + { + std::cout << e.what() << '\n'; + } } diff --git a/doc/examples/operator__ValueType.link b/doc/examples/operator__ValueType.link index 1428a679..9a689f82 100644 --- a/doc/examples/operator__ValueType.link +++ b/doc/examples/operator__ValueType.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/operator__ValueType.output b/doc/examples/operator__ValueType.output index 5cd9cd3a..a3bd9fff 100644 --- a/doc/examples/operator__ValueType.output +++ b/doc/examples/operator__ValueType.output @@ -9,3 +9,4 @@ number: {"floating-point":17.23,"integer":42} null: null boolean: true array: [1,2,3,4,5] +[json.exception.type_error.302] type must be boolean, but is string diff --git a/src/json.hpp b/src/json.hpp index 4b2af732..9fb7690f 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -162,18 +162,18 @@ Exceptions have ids 1xx. name / id | example massage | description ------------------------------ | --------------- | ------------------------- -json.exception.[parse_error](@ref parse_error).101 | "parse error at 2: unexpected end of input; expected string literal" | This error indicates a syntax error while deserializing a JSON text. The error message describes that an unexpected token (character) was encountered, and the member @ref parse_error::byte indicates the error position. -json.exception.[parse_error](@ref parse_error).102 | "parse error at 14: missing or wrong low surrogate" | JSON uses the `\uxxxx` format to describe Unicode characters. Code points above above 0xFFFF are split into two `\uxxxx` entries ("surrogate pairs"). This error indicates that the surrogate pair is incomplete or contains an invalid code point. -json.exception.[parse_error](@ref parse_error).103 | "parse error: code points above 0x10FFFF are invalid" | Unicode supports code points up to 0x10FFFF. Code points above 0x10FFFF are invalid. -json.exception.[parse_error](@ref parse_error).104 | "parse error: JSON patch must be an array of objects" | [RFC 6902](https://tools.ietf.org/html/rfc6902) requires a JSON Patch document to be a JSON document that represents an array of objects. -json.exception.[parse_error](@ref parse_error).105 | "parse error: operation must have string member 'op'" | An operation of a JSON Patch document must contain exactly one "op" member, whose value indicates the operation to perform. Its value must be one of "add", "remove", "replace", "move", "copy", or "test"; other values are errors. -json.exception.[parse_error](@ref parse_error).106 | "parse error: array index '01' must not begin with '0'" | An array index in a JSON Pointer ([RFC 6901](https://tools.ietf.org/html/rfc6901)) may be `0` or any number wihtout a leading `0`. -json.exception.[parse_error](@ref parse_error).107 | "parse error: JSON pointer must be empty or begin with '/' - was: 'foo'" | A JSON Pointer must be a Unicode string containing a sequence of zero or more reference tokens, each prefixed by a `/` character. -json.exception.[parse_error](@ref parse_error).108 | "parse error: escape character '~' must be followed with '0' or '1'" | In a JSON Pointer, only `~0` and `~1` are valid escape sequences. -json.exception.[parse_error](@ref parse_error).109 | "parse error: array index 'one' is not a number" | A JSON Pointer array index must be a number. -json.exception.[parse_error](@ref parse_error).110 | "parse error at 1: cannot read 2 bytes from vector" | When parsing CBOR or MessagePack, the byte vector ends before the complete value has been read. -json.exception.[parse_error](@ref parse_error).111 | "parse error: bad input stream" | Parsing CBOR or MessagePack from an input stream where the `badbit` or `failbit` is set. -json.exception.[parse_error](@ref parse_error).112 | "parse error at 1: error reading CBOR; last byte: 0xf8" | Not all types of CBOR or MessagePack are supported. This exception occurs if an unsupported byte was read. +json.exception.parse_error.101 | parse error at 2: unexpected end of input; expected string literal | This error indicates a syntax error while deserializing a JSON text. The error message describes that an unexpected token (character) was encountered, and the member @a byte indicates the error position. +json.exception.parse_error.102 | parse error at 14: missing or wrong low surrogate | JSON uses the `\uxxxx` format to describe Unicode characters. Code points above above 0xFFFF are split into two `\uxxxx` entries ("surrogate pairs"). This error indicates that the surrogate pair is incomplete or contains an invalid code point. +json.exception.parse_error.103 | parse error: code points above 0x10FFFF are invalid | Unicode supports code points up to 0x10FFFF. Code points above 0x10FFFF are invalid. +json.exception.parse_error.104 | parse error: JSON patch must be an array of objects | [RFC 6902](https://tools.ietf.org/html/rfc6902) requires a JSON Patch document to be a JSON document that represents an array of objects. +json.exception.parse_error.105 | parse error: operation must have string member 'op' | An operation of a JSON Patch document must contain exactly one "op" member, whose value indicates the operation to perform. Its value must be one of "add", "remove", "replace", "move", "copy", or "test"; other values are errors. +json.exception.parse_error.106 | parse error: array index '01' must not begin with '0' | An array index in a JSON Pointer ([RFC 6901](https://tools.ietf.org/html/rfc6901)) may be `0` or any number wihtout a leading `0`. +json.exception.parse_error.107 | parse error: JSON pointer must be empty or begin with '/' - was: 'foo' | A JSON Pointer must be a Unicode string containing a sequence of zero or more reference tokens, each prefixed by a `/` character. +json.exception.parse_error.108 | parse error: escape character '~' must be followed with '0' or '1' | In a JSON Pointer, only `~0` and `~1` are valid escape sequences. +json.exception.parse_error.109 | parse error: array index 'one' is not a number | A JSON Pointer array index must be a number. +json.exception.parse_error.110 | parse error at 1: cannot read 2 bytes from vector | When parsing CBOR or MessagePack, the byte vector ends before the complete value has been read. +json.exception.parse_error.111 | parse error: bad input stream | Parsing CBOR or MessagePack from an input stream where the [`badbit` or `failbit`](http://en.cppreference.com/w/cpp/io/ios_base/iostate) is set. +json.exception.parse_error.112 | parse error at 1: error reading CBOR; last byte: 0xf8 | Not all types of CBOR or MessagePack are supported. This exception occurs if an unsupported byte was read. @since version 3.0.0 */ @@ -212,22 +212,22 @@ class parse_error : public exception Exceptions have ids 2xx. -name / id | example massage | description ------------------------------- | --------------- | ------------------------- -json.exception.[invalid_iterator](@ref invalid_iterator).201 | "iterators are not compatible" | The iterators passed to constructor @ref basic_json(InputIT first, InputIT last) are not compatible, meaning they do not belong to the same container. Therefore, the range (@a first, @a last) is invalid. -json.exception.[invalid_iterator](@ref invalid_iterator).202 | "iterator does not fit current value" | In an erase or insert function, the passed iterator @a pos does not belong to the JSON value for which the function was called. It hence does not define a valid position for the deletion/insertion. -json.exception.[invalid_iterator](@ref invalid_iterator).203 | "iterators do not fit current value" | Either iterator passed to function @ref erase(IteratorType first, IteratorType last) does not belong to the JSON value from which values shall be erased. It hence does not define a valid range to delete values from. -json.exception.[invalid_iterator](@ref invalid_iterator).204 | "iterators out of range" | When an iterator range for a primitive type (number, boolean, or string) is passed to a constructor or an erase function, this range has to be exactly (@ref begin(), @ref end()), because this is the only way the single stored value is expressed. All other ranges are invalid. -json.exception.[invalid_iterator](@ref invalid_iterator).205 | "iterator out of range" | When an iterator for a primitive type (number, boolean, or string) is passed to an erase function, the iterator has to be the @ref begin() iterator, because it is the only way to address the stored value. All other iterators are invalid. -json.exception.[invalid_iterator](@ref invalid_iterator).206 | "cannot construct with iterators from null" | The iterators passed to constructor @ref basic_json(InputIT first, InputIT last) belong to a JSON null value and hence to not define a valid range. -json.exception.[invalid_iterator](@ref invalid_iterator).207 | "cannot use key() for non-object iterators" | The key() member function can only be used on iterators belonging to a JSON object, because other types do not have a concept of a key. -json.exception.[invalid_iterator](@ref invalid_iterator).208 | "cannot use operator[] for object iterators" | The operator[] to specify a concrete offset cannot be used on iterators belonging to a JSON object, because JSON objects are unordered. -json.exception.[invalid_iterator](@ref invalid_iterator).209 | "cannot use offsets with object iterators" | The offset operators (+, -, +=, -=) cannot be used on iterators belonging to a JSON object, because JSON objects are unordered. -json.exception.[invalid_iterator](@ref invalid_iterator).210 | "iterators do not fit" | The iterator range passed to the insert function are not compatible, meaning they do not belong to the same container. Therefore, the range (@a first, @a last) is invalid. -json.exception.[invalid_iterator](@ref invalid_iterator).211 | "passed iterators may not belong to container" | The iterator range passed to the insert function must not be a subrange of the container to insert to. -json.exception.[invalid_iterator](@ref invalid_iterator).212 | "cannot compare iterators of different containers" | When two iterators are compared, they must belong to the same container. -json.exception.[invalid_iterator](@ref invalid_iterator).213 | "cannot compare order of object iterators" | The order of object iterators cannot be compated, because JSON objects are unordered. -json.exception.[invalid_iterator](@ref invalid_iterator).214 | "cannot get value" | Cannot get value for iterator: Either the iterator belongs to a null value or it is an iterator to a primitive type (number, boolean, or string), but the iterator is different to @ref begin(). +name / id | example massage | description +----------------------------------- | --------------- | ------------------------- +json.exception.invalid_iterator.201 | iterators are not compatible | The iterators passed to constructor @ref basic_json(InputIT first, InputIT last) are not compatible, meaning they do not belong to the same container. Therefore, the range (@a first, @a last) is invalid. +json.exception.invalid_iterator.202 | iterator does not fit current value | In an erase or insert function, the passed iterator @a pos does not belong to the JSON value for which the function was called. It hence does not define a valid position for the deletion/insertion. +json.exception.invalid_iterator.203 | iterators do not fit current value | Either iterator passed to function @ref erase(IteratorType first, IteratorType last) does not belong to the JSON value from which values shall be erased. It hence does not define a valid range to delete values from. +json.exception.invalid_iterator.204 | iterators out of range | When an iterator range for a primitive type (number, boolean, or string) is passed to a constructor or an erase function, this range has to be exactly (@ref begin(), @ref end()), because this is the only way the single stored value is expressed. All other ranges are invalid. +json.exception.invalid_iterator.205 | iterator out of range | When an iterator for a primitive type (number, boolean, or string) is passed to an erase function, the iterator has to be the @ref begin() iterator, because it is the only way to address the stored value. All other iterators are invalid. +json.exception.invalid_iterator.206 | cannot construct with iterators from null | The iterators passed to constructor @ref basic_json(InputIT first, InputIT last) belong to a JSON null value and hence to not define a valid range. +json.exception.invalid_iterator.207 | cannot use key() for non-object iterators | The key() member function can only be used on iterators belonging to a JSON object, because other types do not have a concept of a key. +json.exception.invalid_iterator.208 | cannot use operator[] for object iterators | The operator[] to specify a concrete offset cannot be used on iterators belonging to a JSON object, because JSON objects are unordered. +json.exception.invalid_iterator.209 | cannot use offsets with object iterators | The offset operators (+, -, +=, -=) cannot be used on iterators belonging to a JSON object, because JSON objects are unordered. +json.exception.invalid_iterator.210 | iterators do not fit | The iterator range passed to the insert function are not compatible, meaning they do not belong to the same container. Therefore, the range (@a first, @a last) is invalid. +json.exception.invalid_iterator.211 | passed iterators may not belong to container | The iterator range passed to the insert function must not be a subrange of the container to insert to. +json.exception.invalid_iterator.212 | cannot compare iterators of different containers | When two iterators are compared, they must belong to the same container. +json.exception.invalid_iterator.213 | cannot compare order of object iterators | The order of object iterators cannot be compated, because JSON objects are unordered. +json.exception.invalid_iterator.214 | cannot get value | Cannot get value for iterator: Either the iterator belongs to a null value or it is an iterator to a primitive type (number, boolean, or string), but the iterator is different to @ref begin(). @since version 3.0.0 */ @@ -244,22 +244,22 @@ class invalid_iterator : public exception Exceptions have ids 3xx. -name / id | example massage | description ------------------------------- | --------------- | ------------------------- -json.exception.[type_error](@ref type_error).301 | "cannot create object from initializer list" | To create an object from an initializer list, the initializer list must consist only of a list of pairs whose first element is a string. When this constraint is violated, an array is created instead. -json.exception.[type_error](@ref type_error).302 | "type must be object, but is array" | During implicit or explicit value conversion, the JSON type must be compatible to the target type. For instance, a JSON string can only be converted into string types, but not into numbers or boolean types. -json.exception.[type_error](@ref type_error).303 | "incompatible ReferenceType for get_ref, actual type is object" | To retrieve a reference to a value stored in a @ref basic_json object with @ref get_ref, the type of the reference must match the value type. For instance, for a JSON array, the @a ReferenceType must be @ref array_t&. -json.exception.[type_error](@ref type_error).304 | "cannot use at() with string" | The @ref at() member functions can only be executed for certain JSON types. -json.exception.[type_error](@ref type_error).305 | "cannot use operator[] with string" | The @ref operator[] member functions can only be executed for certain JSON types. -json.exception.[type_error](@ref type_error).306 | "cannot use value() with string" | The @ref value() member functions can only be executed for certain JSON types. -json.exception.[type_error](@ref type_error).307 | "cannot use erase() with string" | The @ref erase() member functions can only be executed for certain JSON types. -json.exception.[type_error](@ref type_error).308 | "cannot use push_back() with string" | The @ref push_back() and @ref operator+= member functions can only be executed for certain JSON types. -json.exception.[type_error](@ref type_error).309 | "cannot use insert() with" | The @ref insert() member functions can only be executed for certain JSON types. -json.exception.[type_error](@ref type_error).310 | "cannot use swap() with number" | The @ref swap() member functions can only be executed for certain JSON types. -json.exception.[type_error](@ref type_error).311 | "cannot use emplace_back() with string" | The @ref emplace_back() member function can only be executed for certain JSON types. -json.exception.[type_error](@ref type_error).313 | "invalid value to unflatten" | The @ref unflatten function converts an object whose keys are JSON Pointers back into an arbitrary nested JSON value. The JSON Pointers must not overlap, because then the resulting value would not be well defined. -json.exception.[type_error](@ref type_error).314 | "only objects can be unflattened" | The @ref unflatten function only works for an object whose keys are JSON Pointers. -json.exception.[type_error](@ref type_error).315 | "values in object must be primitive" | The @ref unflatten function only works for an object whose keys are JSON Pointers and whose values are primitive. +name / id | example massage | description +----------------------------- | --------------- | ------------------------- +json.exception.type_error.301 | cannot create object from initializer list | To create an object from an initializer list, the initializer list must consist only of a list of pairs whose first element is a string. When this constraint is violated, an array is created instead. +json.exception.type_error.302 | type must be object, but is array | During implicit or explicit value conversion, the JSON type must be compatible to the target type. For instance, a JSON string can only be converted into string types, but not into numbers or boolean types. +json.exception.type_error.303 | incompatible ReferenceType for get_ref, actual type is object | To retrieve a reference to a value stored in a @ref basic_json object with @ref get_ref, the type of the reference must match the value type. For instance, for a JSON array, the @a ReferenceType must be @ref array_t&. +json.exception.type_error.304 | cannot use at() with string | The @ref at() member functions can only be executed for certain JSON types. +json.exception.type_error.305 | cannot use operator[] with string | The @ref operator[] member functions can only be executed for certain JSON types. +json.exception.type_error.306 | cannot use value() with string | The @ref value() member functions can only be executed for certain JSON types. +json.exception.type_error.307 | cannot use erase() with string | The @ref erase() member functions can only be executed for certain JSON types. +json.exception.type_error.308 | cannot use push_back() with string | The @ref push_back() and @ref operator+= member functions can only be executed for certain JSON types. +json.exception.type_error.309 | cannot use insert() with | The @ref insert() member functions can only be executed for certain JSON types. +json.exception.type_error.310 | cannot use swap() with number | The @ref swap() member functions can only be executed for certain JSON types. +json.exception.type_error.311 | cannot use emplace_back() with string | The @ref emplace_back() member function can only be executed for certain JSON types. +json.exception.type_error.313 | invalid value to unflatten | The @ref unflatten function converts an object whose keys are JSON Pointers back into an arbitrary nested JSON value. The JSON Pointers must not overlap, because then the resulting value would not be well defined. +json.exception.type_error.314 | only objects can be unflattened | The @ref unflatten function only works for an object whose keys are JSON Pointers. +json.exception.type_error.315 | values in object must be primitive | The @ref unflatten function only works for an object whose keys are JSON Pointers and whose values are primitive. @since version 3.0.0 */ @@ -276,13 +276,13 @@ class type_error : public exception Exceptions have ids 4xx. -name / id | example massage | description ------------------------------- | --------------- | ------------------------- -json.exception.[out_of_range](@ref out_of_range).401 | "array index 3 is out of range" | The provided array index @a i is larger than @a size-1. -json.exception.[out_of_range](@ref out_of_range).402 | "array index '-' (3) is out of range" | The special array index `-` in a JSON Pointer never describes a valid element of the array, but the index past the end. -json.exception.[out_of_range](@ref out_of_range).403 | "key 'foo' not found" | The provided key was not found in the JSON object. -json.exception.[out_of_range](@ref out_of_range).404 | "unresolved reference token 'foo'" | A reference token in a JSON Pointer could not be resolved. -json.exception.[out_of_range](@ref out_of_range).405 | "JSON pointer has no parent" | The JSON Patch operations 'remove' and 'add' can not be applied to the root element of the JSON value. +name / id | example massage | description +------------------------------- | --------------- | ------------------------- +json.exception.out_of_range.401 | array index 3 is out of range | The provided array index @a i is larger than @a size-1. +json.exception.out_of_range.402 | array index '-' (3) is out of range | The special array index `-` in a JSON Pointer never describes a valid element of the array, but the index past the end. That is, it can only be used to add elements at this position, but not to read it. +json.exception.out_of_range.403 | key 'foo' not found | The provided key was not found in the JSON object. +json.exception.out_of_range.404 | unresolved reference token 'foo' | A reference token in a JSON Pointer could not be resolved. +json.exception.out_of_range.405 | JSON pointer has no parent | The JSON Patch operations 'remove' and 'add' can not be applied to the root element of the JSON value. @since version 3.0.0 */ @@ -301,8 +301,7 @@ Exceptions have ids 5xx. name / id | example massage | description ------------------------------ | --------------- | ------------------------- -json.exception.[other_error](@ref other_error).501 | "unsuccessful: {"op":"test","path":"/baz", - "value":"bar"}" | A JSON Patch operation 'test' failed. +json.exception.other_error.501 | unsuccessful: {"op":"test","path":"/baz", "value":"bar"} | A JSON Patch operation 'test' failed. The unsuccessful operation is also printed. @since version 3.0.0 */ @@ -2122,9 +2121,6 @@ class basic_json @complexity Constant. - @throw std::bad_alloc if allocation for object, array, or string value - fails - @liveexample{The following code shows the constructor for different @ref value_t values,basic_json__value_t} @@ -2281,8 +2277,10 @@ class basic_json @throw type_error.301 if @a type_deduction is `false`, @a manual_type is `value_t::object`, but @a init contains an element which is not a pair - whose first element is a string; example: `"cannot create object from - initializer list"` + whose first element is a string. In this case, the constructor could not + create an object. If @a type_deduction would have be `true`, an array + would have been created. See @ref object(std::initializer_list) + for an example. @complexity Linear in the size of the initializer list @a init. @@ -2396,16 +2394,17 @@ class basic_json related function @ref array(std::initializer_list), there are no cases which can only be expressed by this function. That is, any initializer list @a init can also be passed to the initializer list - constructor @ref basic_json(std::initializer_list, bool, - value_t). + constructor @ref basic_json(std::initializer_list, bool, value_t). @param[in] init initializer list to create an object from (optional) @return JSON object value - @throw type_error.301 if @a init is not a pair whose first elements are - strings; thrown by - @ref basic_json(std::initializer_list, bool, value_t) + @throw type_error.301 if @a init is not a list of pairs whose first + elements are strings. In this case, no object can be created. When such a + value is passed to @ref basic_json(std::initializer_list, bool, value_t), + an array would have been created from the passed initializer list @a init. + See example below. @complexity Linear in the size of @a init. @@ -2457,10 +2456,10 @@ class basic_json The semantics depends on the different types a JSON value can have: - In case of primitive types (number, boolean, or string), @a first must be `begin()` and @a last must be `end()`. In this case, the value is - copied. Otherwise, std::out_of_range is thrown. + copied. Otherwise, invalid_iterator.204 is thrown. - In case of structured types (array, object), the constructor behaves as similar versions for `std::vector`. - - In case of a null type, std::domain_error is thrown. + - In case of a null type, invalid_iterator.206 is thrown. @tparam InputIT an input iterator type (@ref iterator or @ref const_iterator) @@ -2471,15 +2470,19 @@ class basic_json @pre Iterators @a first and @a last must be initialized. **This precondition is enforced with an assertion.** - @throw invalid_iterator.201 if iterators are not compatible; that is, do - not belong to the same JSON value; example: `"iterators are not - compatible"` - @throw invalid_iterator.204 if iterators are for a primitive type (number, - boolean, or string) where an out of range error can be detected easily; - example: `"iterators out of range"` - @throw std::bad_alloc if allocation for object, array, or string fails - @throw invalid_iterator.206 if called with a null value; example: `"cannot - construct with iterators from null"` + @pre Range `[first, last)` is valid. Usually, this precondition cannot be + checked efficiently. Only certain edge cases are detected; see the + description of the exceptions below. + + @throw invalid_iterator.201 if iterators @a first and @a last are not + compatible (i.e., do not belong to the same JSON value). In this case, + the range `[first, last)` is undefined. + @throw invalid_iterator.204 if iterators @a first and @a last belong to a + primitive type (number, boolean, or string), but @a first does not point + to the first element any more. In this case, the range `[first, last)` is + undefined. See example code below. + @throw invalid_iterator.206 if iterators @a first and @a last belong to a + null value. In this case, the range `[first, last)` is undefined. @complexity Linear in distance between @a first and @a last. @@ -2603,8 +2606,6 @@ class basic_json - The complexity is linear. - As postcondition, it holds: `other == basic_json(other)`. - @throw std::bad_alloc if allocation for object, array, or string fails. - @liveexample{The following code shows an example for the copy constructor.,basic_json__basic_json} @@ -3607,10 +3608,10 @@ class basic_json @return reference to the internally stored JSON value if the requested reference type @a ReferenceType fits to the JSON value; throws - std::domain_error otherwise + type_error.303 otherwise @throw type_error.303 in case passed type @a ReferenceType is incompatible - with the stored JSON value + with the stored JSON value; see example below @complexity Constant. @@ -3654,7 +3655,8 @@ class basic_json @return copy of the JSON value, converted to type @a ValueType @throw type_error.302 in case passed type @a ValueType is incompatible - to JSON, thrown by @ref get() const + to the JSON value type (e.g., the JSON value is of type boolean, but a + string is requested); see example below @complexity Linear in the size of the JSON value. @@ -3701,10 +3703,10 @@ class basic_json @return reference to the element at index @a idx - @throw type_error.304 if the JSON value is not an array; example: `"cannot - use at() with string"` + @throw type_error.304 if the JSON value is not an array; in this case, + calling `at` with an index makes no sense. @throw out_of_range.401 if the index @a idx is out of range of the array; - that is, `idx >= size()`; example: `"array index 7 is out of range"` + that is, `idx >= size()`; see example below. @complexity Constant. @@ -3744,10 +3746,10 @@ class basic_json @return const reference to the element at index @a idx - @throw type_error.304 if the JSON value is not an array; example: `"cannot - use at() with string"` + @throw type_error.304 if the JSON value is not an array; in this case, + calling `at` with an index makes no sense. @throw out_of_range.401 if the index @a idx is out of range of the array; - that is, `idx >= size()`; example: `"array index 7 is out of range"` + that is, `idx >= size()`; see example below. @complexity Constant. @@ -3787,10 +3789,10 @@ class basic_json @return reference to the element at key @a key - @throw type_error.304 if the JSON value is not an object; example: - `"cannot use at() with boolean"` + @throw type_error.304 if the JSON value is not an object; in this case, + calling `at` with a key makes no sense. @throw out_of_range.403 if the key @a key is is not stored in the object; - that is, `find(key) == end()`; example: `"key "the fast" not found"` + that is, `find(key) == end()`; see example below. @complexity Logarithmic in the size of the container. @@ -3834,10 +3836,10 @@ class basic_json @return const reference to the element at key @a key - @throw type_error.304 if the JSON value is not an object; example: - `"cannot use at() with boolean"` + @throw type_error.304 if the JSON value is not an object; in this case, + calling `at` with a key makes no sense. @throw out_of_range.403 if the key @a key is is not stored in the object; - that is, `find(key) == end()`; example: `"key "the fast" not found"` + that is, `find(key) == end()`; see example below. @complexity Logarithmic in the size of the container. @@ -3884,8 +3886,8 @@ class basic_json @return reference to the element at index @a idx - @throw type_error.305 if JSON is not an array or null; example: `"cannot - use operator[] with string" + @throw type_error.305 if the JSON value is not an array or null; in that + cases, using the [] operator with an index makes no sense. @complexity Constant if @a idx is in the range of the array. Otherwise linear in `idx - size()`. @@ -3932,8 +3934,8 @@ class basic_json @return const reference to the element at index @a idx - @throw type_error.305 if JSON is not an array; example: `"cannot use - operator[] with null"` + @throw type_error.305 if the JSON value is not an array; in that cases, + using the [] operator with an index makes no sense. @complexity Constant. @@ -3966,8 +3968,8 @@ class basic_json @return reference to the element at key @a key - @throw type_error.305 if JSON is not an object or null; example: `"cannot - use operator[] with string"` + @throw type_error.305 if the JSON value is not an object or null; in that + cases, using the [] operator with a key makes no sense. @complexity Logarithmic in the size of the container. @@ -4015,8 +4017,8 @@ class basic_json @pre The element with key @a key must exist. **This precondition is enforced with an assertion.** - @throw type_error.305 if JSON is not an object; example: `"cannot use - operator[] with null"` + @throw type_error.305 if the JSON value is not an object; in that cases, + using the [] operator with a key makes no sense. @complexity Logarithmic in the size of the container. @@ -4054,8 +4056,8 @@ class basic_json @return reference to the element at key @a key - @throw type_error.305 if JSON is not an object or null; example: `"cannot - use operator[] with string"` + @throw type_error.305 if the JSON value is not an object or null; in that + cases, using the [] operator with a key makes no sense. @complexity Logarithmic in the size of the container. @@ -4089,8 +4091,8 @@ class basic_json @return const reference to the element at key @a key - @throw type_error.305 if JSON is not an object; example: `"cannot use - operator[] with null"` + @throw type_error.305 if the JSON value is not an object; in that cases, + using the [] operator with a key makes no sense. @complexity Logarithmic in the size of the container. @@ -4122,8 +4124,8 @@ class basic_json @return reference to the element at key @a key - @throw type_error.305 if JSON is not an object or null; example: `"cannot - use operator[] with string"` + @throw type_error.305 if the JSON value is not an object or null; in that + cases, using the [] operator with a key makes no sense. @complexity Logarithmic in the size of the container. @@ -4172,8 +4174,8 @@ class basic_json @pre The element with key @a key must exist. **This precondition is enforced with an assertion.** - @throw type_error.305 if JSON is not an object; example: `"cannot use - operator[] with null"` + @throw type_error.305 if the JSON value is not an object; in that cases, + using the [] operator with a key makes no sense. @complexity Logarithmic in the size of the container. @@ -4232,8 +4234,8 @@ class basic_json @return copy of the element at key @a key or @a default_value if @a key is not found - @throw type_error.306 if JSON is not an object; example: `"cannot use - value() with null"` + @throw type_error.306 if the JSON value is not an objec; in that cases, + using `value()` with a key makes no sense. @complexity Logarithmic in the size of the container. @@ -4307,8 +4309,8 @@ class basic_json @return copy of the element at key @a key or @a default_value if @a key is not found - @throw type_error.306 if JSON is not an object; example: `"cannot use - value() with null"` + @throw type_error.306 if the JSON value is not an objec; in that cases, + using `value()` with a key makes no sense. @complexity Logarithmic in the size of the container. diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index d2f6346f..22e12841 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -162,18 +162,18 @@ Exceptions have ids 1xx. name / id | example massage | description ------------------------------ | --------------- | ------------------------- -json.exception.[parse_error](@ref parse_error).101 | "parse error at 2: unexpected end of input; expected string literal" | This error indicates a syntax error while deserializing a JSON text. The error message describes that an unexpected token (character) was encountered, and the member @ref parse_error::byte indicates the error position. -json.exception.[parse_error](@ref parse_error).102 | "parse error at 14: missing or wrong low surrogate" | JSON uses the `\uxxxx` format to describe Unicode characters. Code points above above 0xFFFF are split into two `\uxxxx` entries ("surrogate pairs"). This error indicates that the surrogate pair is incomplete or contains an invalid code point. -json.exception.[parse_error](@ref parse_error).103 | "parse error: code points above 0x10FFFF are invalid" | Unicode supports code points up to 0x10FFFF. Code points above 0x10FFFF are invalid. -json.exception.[parse_error](@ref parse_error).104 | "parse error: JSON patch must be an array of objects" | [RFC 6902](https://tools.ietf.org/html/rfc6902) requires a JSON Patch document to be a JSON document that represents an array of objects. -json.exception.[parse_error](@ref parse_error).105 | "parse error: operation must have string member 'op'" | An operation of a JSON Patch document must contain exactly one "op" member, whose value indicates the operation to perform. Its value must be one of "add", "remove", "replace", "move", "copy", or "test"; other values are errors. -json.exception.[parse_error](@ref parse_error).106 | "parse error: array index '01' must not begin with '0'" | An array index in a JSON Pointer ([RFC 6901](https://tools.ietf.org/html/rfc6901)) may be `0` or any number wihtout a leading `0`. -json.exception.[parse_error](@ref parse_error).107 | "parse error: JSON pointer must be empty or begin with '/' - was: 'foo'" | A JSON Pointer must be a Unicode string containing a sequence of zero or more reference tokens, each prefixed by a `/` character. -json.exception.[parse_error](@ref parse_error).108 | "parse error: escape character '~' must be followed with '0' or '1'" | In a JSON Pointer, only `~0` and `~1` are valid escape sequences. -json.exception.[parse_error](@ref parse_error).109 | "parse error: array index 'one' is not a number" | A JSON Pointer array index must be a number. -json.exception.[parse_error](@ref parse_error).110 | "parse error at 1: cannot read 2 bytes from vector" | When parsing CBOR or MessagePack, the byte vector ends before the complete value has been read. -json.exception.[parse_error](@ref parse_error).111 | "parse error: bad input stream" | Parsing CBOR or MessagePack from an input stream where the `badbit` or `failbit` is set. -json.exception.[parse_error](@ref parse_error).112 | "parse error at 1: error reading CBOR; last byte: 0xf8" | Not all types of CBOR or MessagePack are supported. This exception occurs if an unsupported byte was read. +json.exception.parse_error.101 | parse error at 2: unexpected end of input; expected string literal | This error indicates a syntax error while deserializing a JSON text. The error message describes that an unexpected token (character) was encountered, and the member @a byte indicates the error position. +json.exception.parse_error.102 | parse error at 14: missing or wrong low surrogate | JSON uses the `\uxxxx` format to describe Unicode characters. Code points above above 0xFFFF are split into two `\uxxxx` entries ("surrogate pairs"). This error indicates that the surrogate pair is incomplete or contains an invalid code point. +json.exception.parse_error.103 | parse error: code points above 0x10FFFF are invalid | Unicode supports code points up to 0x10FFFF. Code points above 0x10FFFF are invalid. +json.exception.parse_error.104 | parse error: JSON patch must be an array of objects | [RFC 6902](https://tools.ietf.org/html/rfc6902) requires a JSON Patch document to be a JSON document that represents an array of objects. +json.exception.parse_error.105 | parse error: operation must have string member 'op' | An operation of a JSON Patch document must contain exactly one "op" member, whose value indicates the operation to perform. Its value must be one of "add", "remove", "replace", "move", "copy", or "test"; other values are errors. +json.exception.parse_error.106 | parse error: array index '01' must not begin with '0' | An array index in a JSON Pointer ([RFC 6901](https://tools.ietf.org/html/rfc6901)) may be `0` or any number wihtout a leading `0`. +json.exception.parse_error.107 | parse error: JSON pointer must be empty or begin with '/' - was: 'foo' | A JSON Pointer must be a Unicode string containing a sequence of zero or more reference tokens, each prefixed by a `/` character. +json.exception.parse_error.108 | parse error: escape character '~' must be followed with '0' or '1' | In a JSON Pointer, only `~0` and `~1` are valid escape sequences. +json.exception.parse_error.109 | parse error: array index 'one' is not a number | A JSON Pointer array index must be a number. +json.exception.parse_error.110 | parse error at 1: cannot read 2 bytes from vector | When parsing CBOR or MessagePack, the byte vector ends before the complete value has been read. +json.exception.parse_error.111 | parse error: bad input stream | Parsing CBOR or MessagePack from an input stream where the [`badbit` or `failbit`](http://en.cppreference.com/w/cpp/io/ios_base/iostate) is set. +json.exception.parse_error.112 | parse error at 1: error reading CBOR; last byte: 0xf8 | Not all types of CBOR or MessagePack are supported. This exception occurs if an unsupported byte was read. @since version 3.0.0 */ @@ -212,22 +212,22 @@ class parse_error : public exception Exceptions have ids 2xx. -name / id | example massage | description ------------------------------- | --------------- | ------------------------- -json.exception.[invalid_iterator](@ref invalid_iterator).201 | "iterators are not compatible" | The iterators passed to constructor @ref basic_json(InputIT first, InputIT last) are not compatible, meaning they do not belong to the same container. Therefore, the range (@a first, @a last) is invalid. -json.exception.[invalid_iterator](@ref invalid_iterator).202 | "iterator does not fit current value" | In an erase or insert function, the passed iterator @a pos does not belong to the JSON value for which the function was called. It hence does not define a valid position for the deletion/insertion. -json.exception.[invalid_iterator](@ref invalid_iterator).203 | "iterators do not fit current value" | Either iterator passed to function @ref erase(IteratorType first, IteratorType last) does not belong to the JSON value from which values shall be erased. It hence does not define a valid range to delete values from. -json.exception.[invalid_iterator](@ref invalid_iterator).204 | "iterators out of range" | When an iterator range for a primitive type (number, boolean, or string) is passed to a constructor or an erase function, this range has to be exactly (@ref begin(), @ref end()), because this is the only way the single stored value is expressed. All other ranges are invalid. -json.exception.[invalid_iterator](@ref invalid_iterator).205 | "iterator out of range" | When an iterator for a primitive type (number, boolean, or string) is passed to an erase function, the iterator has to be the @ref begin() iterator, because it is the only way to address the stored value. All other iterators are invalid. -json.exception.[invalid_iterator](@ref invalid_iterator).206 | "cannot construct with iterators from null" | The iterators passed to constructor @ref basic_json(InputIT first, InputIT last) belong to a JSON null value and hence to not define a valid range. -json.exception.[invalid_iterator](@ref invalid_iterator).207 | "cannot use key() for non-object iterators" | The key() member function can only be used on iterators belonging to a JSON object, because other types do not have a concept of a key. -json.exception.[invalid_iterator](@ref invalid_iterator).208 | "cannot use operator[] for object iterators" | The operator[] to specify a concrete offset cannot be used on iterators belonging to a JSON object, because JSON objects are unordered. -json.exception.[invalid_iterator](@ref invalid_iterator).209 | "cannot use offsets with object iterators" | The offset operators (+, -, +=, -=) cannot be used on iterators belonging to a JSON object, because JSON objects are unordered. -json.exception.[invalid_iterator](@ref invalid_iterator).210 | "iterators do not fit" | The iterator range passed to the insert function are not compatible, meaning they do not belong to the same container. Therefore, the range (@a first, @a last) is invalid. -json.exception.[invalid_iterator](@ref invalid_iterator).211 | "passed iterators may not belong to container" | The iterator range passed to the insert function must not be a subrange of the container to insert to. -json.exception.[invalid_iterator](@ref invalid_iterator).212 | "cannot compare iterators of different containers" | When two iterators are compared, they must belong to the same container. -json.exception.[invalid_iterator](@ref invalid_iterator).213 | "cannot compare order of object iterators" | The order of object iterators cannot be compated, because JSON objects are unordered. -json.exception.[invalid_iterator](@ref invalid_iterator).214 | "cannot get value" | Cannot get value for iterator: Either the iterator belongs to a null value or it is an iterator to a primitive type (number, boolean, or string), but the iterator is different to @ref begin(). +name / id | example massage | description +----------------------------------- | --------------- | ------------------------- +json.exception.invalid_iterator.201 | iterators are not compatible | The iterators passed to constructor @ref basic_json(InputIT first, InputIT last) are not compatible, meaning they do not belong to the same container. Therefore, the range (@a first, @a last) is invalid. +json.exception.invalid_iterator.202 | iterator does not fit current value | In an erase or insert function, the passed iterator @a pos does not belong to the JSON value for which the function was called. It hence does not define a valid position for the deletion/insertion. +json.exception.invalid_iterator.203 | iterators do not fit current value | Either iterator passed to function @ref erase(IteratorType first, IteratorType last) does not belong to the JSON value from which values shall be erased. It hence does not define a valid range to delete values from. +json.exception.invalid_iterator.204 | iterators out of range | When an iterator range for a primitive type (number, boolean, or string) is passed to a constructor or an erase function, this range has to be exactly (@ref begin(), @ref end()), because this is the only way the single stored value is expressed. All other ranges are invalid. +json.exception.invalid_iterator.205 | iterator out of range | When an iterator for a primitive type (number, boolean, or string) is passed to an erase function, the iterator has to be the @ref begin() iterator, because it is the only way to address the stored value. All other iterators are invalid. +json.exception.invalid_iterator.206 | cannot construct with iterators from null | The iterators passed to constructor @ref basic_json(InputIT first, InputIT last) belong to a JSON null value and hence to not define a valid range. +json.exception.invalid_iterator.207 | cannot use key() for non-object iterators | The key() member function can only be used on iterators belonging to a JSON object, because other types do not have a concept of a key. +json.exception.invalid_iterator.208 | cannot use operator[] for object iterators | The operator[] to specify a concrete offset cannot be used on iterators belonging to a JSON object, because JSON objects are unordered. +json.exception.invalid_iterator.209 | cannot use offsets with object iterators | The offset operators (+, -, +=, -=) cannot be used on iterators belonging to a JSON object, because JSON objects are unordered. +json.exception.invalid_iterator.210 | iterators do not fit | The iterator range passed to the insert function are not compatible, meaning they do not belong to the same container. Therefore, the range (@a first, @a last) is invalid. +json.exception.invalid_iterator.211 | passed iterators may not belong to container | The iterator range passed to the insert function must not be a subrange of the container to insert to. +json.exception.invalid_iterator.212 | cannot compare iterators of different containers | When two iterators are compared, they must belong to the same container. +json.exception.invalid_iterator.213 | cannot compare order of object iterators | The order of object iterators cannot be compated, because JSON objects are unordered. +json.exception.invalid_iterator.214 | cannot get value | Cannot get value for iterator: Either the iterator belongs to a null value or it is an iterator to a primitive type (number, boolean, or string), but the iterator is different to @ref begin(). @since version 3.0.0 */ @@ -244,22 +244,22 @@ class invalid_iterator : public exception Exceptions have ids 3xx. -name / id | example massage | description ------------------------------- | --------------- | ------------------------- -json.exception.[type_error](@ref type_error).301 | "cannot create object from initializer list" | To create an object from an initializer list, the initializer list must consist only of a list of pairs whose first element is a string. When this constraint is violated, an array is created instead. -json.exception.[type_error](@ref type_error).302 | "type must be object, but is array" | During implicit or explicit value conversion, the JSON type must be compatible to the target type. For instance, a JSON string can only be converted into string types, but not into numbers or boolean types. -json.exception.[type_error](@ref type_error).303 | "incompatible ReferenceType for get_ref, actual type is object" | To retrieve a reference to a value stored in a @ref basic_json object with @ref get_ref, the type of the reference must match the value type. For instance, for a JSON array, the @a ReferenceType must be @ref array_t&. -json.exception.[type_error](@ref type_error).304 | "cannot use at() with string" | The @ref at() member functions can only be executed for certain JSON types. -json.exception.[type_error](@ref type_error).305 | "cannot use operator[] with string" | The @ref operator[] member functions can only be executed for certain JSON types. -json.exception.[type_error](@ref type_error).306 | "cannot use value() with string" | The @ref value() member functions can only be executed for certain JSON types. -json.exception.[type_error](@ref type_error).307 | "cannot use erase() with string" | The @ref erase() member functions can only be executed for certain JSON types. -json.exception.[type_error](@ref type_error).308 | "cannot use push_back() with string" | The @ref push_back() and @ref operator+= member functions can only be executed for certain JSON types. -json.exception.[type_error](@ref type_error).309 | "cannot use insert() with" | The @ref insert() member functions can only be executed for certain JSON types. -json.exception.[type_error](@ref type_error).310 | "cannot use swap() with number" | The @ref swap() member functions can only be executed for certain JSON types. -json.exception.[type_error](@ref type_error).311 | "cannot use emplace_back() with string" | The @ref emplace_back() member function can only be executed for certain JSON types. -json.exception.[type_error](@ref type_error).313 | "invalid value to unflatten" | The @ref unflatten function converts an object whose keys are JSON Pointers back into an arbitrary nested JSON value. The JSON Pointers must not overlap, because then the resulting value would not be well defined. -json.exception.[type_error](@ref type_error).314 | "only objects can be unflattened" | The @ref unflatten function only works for an object whose keys are JSON Pointers. -json.exception.[type_error](@ref type_error).315 | "values in object must be primitive" | The @ref unflatten function only works for an object whose keys are JSON Pointers and whose values are primitive. +name / id | example massage | description +----------------------------- | --------------- | ------------------------- +json.exception.type_error.301 | cannot create object from initializer list | To create an object from an initializer list, the initializer list must consist only of a list of pairs whose first element is a string. When this constraint is violated, an array is created instead. +json.exception.type_error.302 | type must be object, but is array | During implicit or explicit value conversion, the JSON type must be compatible to the target type. For instance, a JSON string can only be converted into string types, but not into numbers or boolean types. +json.exception.type_error.303 | incompatible ReferenceType for get_ref, actual type is object | To retrieve a reference to a value stored in a @ref basic_json object with @ref get_ref, the type of the reference must match the value type. For instance, for a JSON array, the @a ReferenceType must be @ref array_t&. +json.exception.type_error.304 | cannot use at() with string | The @ref at() member functions can only be executed for certain JSON types. +json.exception.type_error.305 | cannot use operator[] with string | The @ref operator[] member functions can only be executed for certain JSON types. +json.exception.type_error.306 | cannot use value() with string | The @ref value() member functions can only be executed for certain JSON types. +json.exception.type_error.307 | cannot use erase() with string | The @ref erase() member functions can only be executed for certain JSON types. +json.exception.type_error.308 | cannot use push_back() with string | The @ref push_back() and @ref operator+= member functions can only be executed for certain JSON types. +json.exception.type_error.309 | cannot use insert() with | The @ref insert() member functions can only be executed for certain JSON types. +json.exception.type_error.310 | cannot use swap() with number | The @ref swap() member functions can only be executed for certain JSON types. +json.exception.type_error.311 | cannot use emplace_back() with string | The @ref emplace_back() member function can only be executed for certain JSON types. +json.exception.type_error.313 | invalid value to unflatten | The @ref unflatten function converts an object whose keys are JSON Pointers back into an arbitrary nested JSON value. The JSON Pointers must not overlap, because then the resulting value would not be well defined. +json.exception.type_error.314 | only objects can be unflattened | The @ref unflatten function only works for an object whose keys are JSON Pointers. +json.exception.type_error.315 | values in object must be primitive | The @ref unflatten function only works for an object whose keys are JSON Pointers and whose values are primitive. @since version 3.0.0 */ @@ -276,13 +276,13 @@ class type_error : public exception Exceptions have ids 4xx. -name / id | example massage | description ------------------------------- | --------------- | ------------------------- -json.exception.[out_of_range](@ref out_of_range).401 | "array index 3 is out of range" | The provided array index @a i is larger than @a size-1. -json.exception.[out_of_range](@ref out_of_range).402 | "array index '-' (3) is out of range" | The special array index `-` in a JSON Pointer never describes a valid element of the array, but the index past the end. -json.exception.[out_of_range](@ref out_of_range).403 | "key 'foo' not found" | The provided key was not found in the JSON object. -json.exception.[out_of_range](@ref out_of_range).404 | "unresolved reference token 'foo'" | A reference token in a JSON Pointer could not be resolved. -json.exception.[out_of_range](@ref out_of_range).405 | "JSON pointer has no parent" | The JSON Patch operations 'remove' and 'add' can not be applied to the root element of the JSON value. +name / id | example massage | description +------------------------------- | --------------- | ------------------------- +json.exception.out_of_range.401 | array index 3 is out of range | The provided array index @a i is larger than @a size-1. +json.exception.out_of_range.402 | array index '-' (3) is out of range | The special array index `-` in a JSON Pointer never describes a valid element of the array, but the index past the end. That is, it can only be used to add elements at this position, but not to read it. +json.exception.out_of_range.403 | key 'foo' not found | The provided key was not found in the JSON object. +json.exception.out_of_range.404 | unresolved reference token 'foo' | A reference token in a JSON Pointer could not be resolved. +json.exception.out_of_range.405 | JSON pointer has no parent | The JSON Patch operations 'remove' and 'add' can not be applied to the root element of the JSON value. @since version 3.0.0 */ @@ -301,8 +301,7 @@ Exceptions have ids 5xx. name / id | example massage | description ------------------------------ | --------------- | ------------------------- -json.exception.[other_error](@ref other_error).501 | "unsuccessful: {"op":"test","path":"/baz", - "value":"bar"}" | A JSON Patch operation 'test' failed. +json.exception.other_error.501 | unsuccessful: {"op":"test","path":"/baz", "value":"bar"} | A JSON Patch operation 'test' failed. The unsuccessful operation is also printed. @since version 3.0.0 */ @@ -2122,9 +2121,6 @@ class basic_json @complexity Constant. - @throw std::bad_alloc if allocation for object, array, or string value - fails - @liveexample{The following code shows the constructor for different @ref value_t values,basic_json__value_t} @@ -2281,8 +2277,10 @@ class basic_json @throw type_error.301 if @a type_deduction is `false`, @a manual_type is `value_t::object`, but @a init contains an element which is not a pair - whose first element is a string; example: `"cannot create object from - initializer list"` + whose first element is a string. In this case, the constructor could not + create an object. If @a type_deduction would have be `true`, an array + would have been created. See @ref object(std::initializer_list) + for an example. @complexity Linear in the size of the initializer list @a init. @@ -2396,16 +2394,17 @@ class basic_json related function @ref array(std::initializer_list), there are no cases which can only be expressed by this function. That is, any initializer list @a init can also be passed to the initializer list - constructor @ref basic_json(std::initializer_list, bool, - value_t). + constructor @ref basic_json(std::initializer_list, bool, value_t). @param[in] init initializer list to create an object from (optional) @return JSON object value - @throw type_error.301 if @a init is not a pair whose first elements are - strings; thrown by - @ref basic_json(std::initializer_list, bool, value_t) + @throw type_error.301 if @a init is not a list of pairs whose first + elements are strings. In this case, no object can be created. When such a + value is passed to @ref basic_json(std::initializer_list, bool, value_t), + an array would have been created from the passed initializer list @a init. + See example below. @complexity Linear in the size of @a init. @@ -2457,10 +2456,10 @@ class basic_json The semantics depends on the different types a JSON value can have: - In case of primitive types (number, boolean, or string), @a first must be `begin()` and @a last must be `end()`. In this case, the value is - copied. Otherwise, std::out_of_range is thrown. + copied. Otherwise, invalid_iterator.204 is thrown. - In case of structured types (array, object), the constructor behaves as similar versions for `std::vector`. - - In case of a null type, std::domain_error is thrown. + - In case of a null type, invalid_iterator.206 is thrown. @tparam InputIT an input iterator type (@ref iterator or @ref const_iterator) @@ -2471,15 +2470,19 @@ class basic_json @pre Iterators @a first and @a last must be initialized. **This precondition is enforced with an assertion.** - @throw invalid_iterator.201 if iterators are not compatible; that is, do - not belong to the same JSON value; example: `"iterators are not - compatible"` - @throw invalid_iterator.204 if iterators are for a primitive type (number, - boolean, or string) where an out of range error can be detected easily; - example: `"iterators out of range"` - @throw std::bad_alloc if allocation for object, array, or string fails - @throw invalid_iterator.206 if called with a null value; example: `"cannot - construct with iterators from null"` + @pre Range `[first, last)` is valid. Usually, this precondition cannot be + checked efficiently. Only certain edge cases are detected; see the + description of the exceptions below. + + @throw invalid_iterator.201 if iterators @a first and @a last are not + compatible (i.e., do not belong to the same JSON value). In this case, + the range `[first, last)` is undefined. + @throw invalid_iterator.204 if iterators @a first and @a last belong to a + primitive type (number, boolean, or string), but @a first does not point + to the first element any more. In this case, the range `[first, last)` is + undefined. See example code below. + @throw invalid_iterator.206 if iterators @a first and @a last belong to a + null value. In this case, the range `[first, last)` is undefined. @complexity Linear in distance between @a first and @a last. @@ -2603,8 +2606,6 @@ class basic_json - The complexity is linear. - As postcondition, it holds: `other == basic_json(other)`. - @throw std::bad_alloc if allocation for object, array, or string fails. - @liveexample{The following code shows an example for the copy constructor.,basic_json__basic_json} @@ -3607,10 +3608,10 @@ class basic_json @return reference to the internally stored JSON value if the requested reference type @a ReferenceType fits to the JSON value; throws - std::domain_error otherwise + type_error.303 otherwise @throw type_error.303 in case passed type @a ReferenceType is incompatible - with the stored JSON value + with the stored JSON value; see example below @complexity Constant. @@ -3654,7 +3655,8 @@ class basic_json @return copy of the JSON value, converted to type @a ValueType @throw type_error.302 in case passed type @a ValueType is incompatible - to JSON, thrown by @ref get() const + to the JSON value type (e.g., the JSON value is of type boolean, but a + string is requested); see example below @complexity Linear in the size of the JSON value. @@ -3701,10 +3703,10 @@ class basic_json @return reference to the element at index @a idx - @throw type_error.304 if the JSON value is not an array; example: `"cannot - use at() with string"` + @throw type_error.304 if the JSON value is not an array; in this case, + calling `at` with an index makes no sense. @throw out_of_range.401 if the index @a idx is out of range of the array; - that is, `idx >= size()`; example: `"array index 7 is out of range"` + that is, `idx >= size()`; see example below. @complexity Constant. @@ -3744,10 +3746,10 @@ class basic_json @return const reference to the element at index @a idx - @throw type_error.304 if the JSON value is not an array; example: `"cannot - use at() with string"` + @throw type_error.304 if the JSON value is not an array; in this case, + calling `at` with an index makes no sense. @throw out_of_range.401 if the index @a idx is out of range of the array; - that is, `idx >= size()`; example: `"array index 7 is out of range"` + that is, `idx >= size()`; see example below. @complexity Constant. @@ -3787,10 +3789,10 @@ class basic_json @return reference to the element at key @a key - @throw type_error.304 if the JSON value is not an object; example: - `"cannot use at() with boolean"` + @throw type_error.304 if the JSON value is not an object; in this case, + calling `at` with a key makes no sense. @throw out_of_range.403 if the key @a key is is not stored in the object; - that is, `find(key) == end()`; example: `"key "the fast" not found"` + that is, `find(key) == end()`; see example below. @complexity Logarithmic in the size of the container. @@ -3834,10 +3836,10 @@ class basic_json @return const reference to the element at key @a key - @throw type_error.304 if the JSON value is not an object; example: - `"cannot use at() with boolean"` + @throw type_error.304 if the JSON value is not an object; in this case, + calling `at` with a key makes no sense. @throw out_of_range.403 if the key @a key is is not stored in the object; - that is, `find(key) == end()`; example: `"key "the fast" not found"` + that is, `find(key) == end()`; see example below. @complexity Logarithmic in the size of the container. @@ -3884,8 +3886,8 @@ class basic_json @return reference to the element at index @a idx - @throw type_error.305 if JSON is not an array or null; example: `"cannot - use operator[] with string" + @throw type_error.305 if the JSON value is not an array or null; in that + cases, using the [] operator with an index makes no sense. @complexity Constant if @a idx is in the range of the array. Otherwise linear in `idx - size()`. @@ -3932,8 +3934,8 @@ class basic_json @return const reference to the element at index @a idx - @throw type_error.305 if JSON is not an array; example: `"cannot use - operator[] with null"` + @throw type_error.305 if the JSON value is not an array; in that cases, + using the [] operator with an index makes no sense. @complexity Constant. @@ -3966,8 +3968,8 @@ class basic_json @return reference to the element at key @a key - @throw type_error.305 if JSON is not an object or null; example: `"cannot - use operator[] with string"` + @throw type_error.305 if the JSON value is not an object or null; in that + cases, using the [] operator with a key makes no sense. @complexity Logarithmic in the size of the container. @@ -4015,8 +4017,8 @@ class basic_json @pre The element with key @a key must exist. **This precondition is enforced with an assertion.** - @throw type_error.305 if JSON is not an object; example: `"cannot use - operator[] with null"` + @throw type_error.305 if the JSON value is not an object; in that cases, + using the [] operator with a key makes no sense. @complexity Logarithmic in the size of the container. @@ -4054,8 +4056,8 @@ class basic_json @return reference to the element at key @a key - @throw type_error.305 if JSON is not an object or null; example: `"cannot - use operator[] with string"` + @throw type_error.305 if the JSON value is not an object or null; in that + cases, using the [] operator with a key makes no sense. @complexity Logarithmic in the size of the container. @@ -4089,8 +4091,8 @@ class basic_json @return const reference to the element at key @a key - @throw type_error.305 if JSON is not an object; example: `"cannot use - operator[] with null"` + @throw type_error.305 if the JSON value is not an object; in that cases, + using the [] operator with a key makes no sense. @complexity Logarithmic in the size of the container. @@ -4122,8 +4124,8 @@ class basic_json @return reference to the element at key @a key - @throw type_error.305 if JSON is not an object or null; example: `"cannot - use operator[] with string"` + @throw type_error.305 if the JSON value is not an object or null; in that + cases, using the [] operator with a key makes no sense. @complexity Logarithmic in the size of the container. @@ -4172,8 +4174,8 @@ class basic_json @pre The element with key @a key must exist. **This precondition is enforced with an assertion.** - @throw type_error.305 if JSON is not an object; example: `"cannot use - operator[] with null"` + @throw type_error.305 if the JSON value is not an object; in that cases, + using the [] operator with a key makes no sense. @complexity Logarithmic in the size of the container. @@ -4232,8 +4234,8 @@ class basic_json @return copy of the element at key @a key or @a default_value if @a key is not found - @throw type_error.306 if JSON is not an object; example: `"cannot use - value() with null"` + @throw type_error.306 if the JSON value is not an objec; in that cases, + using `value()` with a key makes no sense. @complexity Logarithmic in the size of the container. @@ -4307,8 +4309,8 @@ class basic_json @return copy of the element at key @a key or @a default_value if @a key is not found - @throw type_error.306 if JSON is not an object; example: `"cannot use - value() with null"` + @throw type_error.306 if the JSON value is not an objec; in that cases, + using `value()` with a key makes no sense. @complexity Logarithmic in the size of the container. From 0f6b8aa7181aabb51ef71c80b64e12016e7667ae Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Wed, 8 Mar 2017 23:30:38 +0100 Subject: [PATCH 056/530] :memo: more documentation for the new exceptions --- doc/examples/at_json_pointer.cpp | 10 +++++++ doc/examples/at_json_pointer.link | 2 +- doc/examples/at_json_pointer.output | 1 + doc/examples/at_json_pointer_const.cpp | 10 +++++++ doc/examples/at_json_pointer_const.link | 2 +- doc/examples/at_json_pointer_const.output | 1 + doc/examples/back.cpp | 13 +++++++-- doc/examples/back.link | 2 +- doc/examples/back.output | 1 + src/json.hpp | 35 +++++++++++++++++------ src/json.hpp.re2c | 35 +++++++++++++++++------ 11 files changed, 89 insertions(+), 23 deletions(-) diff --git a/doc/examples/at_json_pointer.cpp b/doc/examples/at_json_pointer.cpp index 0665e608..6d1617e6 100644 --- a/doc/examples/at_json_pointer.cpp +++ b/doc/examples/at_json_pointer.cpp @@ -32,4 +32,14 @@ int main() j.at("/array/1"_json_pointer) = 21; // output the changed array std::cout << j["array"] << '\n'; + + // try to use an invalid JSON pointer + try + { + auto ref = j.at("/number/foo"_json_pointer); + } + catch (json::out_of_range& e) + { + std::cout << e.what() << '\n'; + } } diff --git a/doc/examples/at_json_pointer.link b/doc/examples/at_json_pointer.link index 7a7efa26..45402303 100644 --- a/doc/examples/at_json_pointer.link +++ b/doc/examples/at_json_pointer.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/at_json_pointer.output b/doc/examples/at_json_pointer.output index 11913c72..a45737ad 100644 --- a/doc/examples/at_json_pointer.output +++ b/doc/examples/at_json_pointer.output @@ -4,3 +4,4 @@ 2 "bar" [1,21] +[json.exception.out_of_range.404] unresolved reference token 'foo' diff --git a/doc/examples/at_json_pointer_const.cpp b/doc/examples/at_json_pointer_const.cpp index e3cfc515..dab1b39c 100644 --- a/doc/examples/at_json_pointer_const.cpp +++ b/doc/examples/at_json_pointer_const.cpp @@ -20,4 +20,14 @@ int main() std::cout << j.at("/array"_json_pointer) << '\n'; // output element with JSON pointer "/array/1" std::cout << j.at("/array/1"_json_pointer) << '\n'; + + // try to use an invalid JSON pointer + try + { + auto ref = j.at("/number/foo"_json_pointer); + } + catch (json::out_of_range& e) + { + std::cout << e.what() << '\n'; + } } diff --git a/doc/examples/at_json_pointer_const.link b/doc/examples/at_json_pointer_const.link index 9057e0b2..70e7cf86 100644 --- a/doc/examples/at_json_pointer_const.link +++ b/doc/examples/at_json_pointer_const.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/at_json_pointer_const.output b/doc/examples/at_json_pointer_const.output index 7b9306bb..21712e86 100644 --- a/doc/examples/at_json_pointer_const.output +++ b/doc/examples/at_json_pointer_const.output @@ -2,3 +2,4 @@ "foo" [1,2] 2 +[json.exception.out_of_range.404] unresolved reference token 'foo' diff --git a/doc/examples/back.cpp b/doc/examples/back.cpp index 70516e58..45f9483c 100644 --- a/doc/examples/back.cpp +++ b/doc/examples/back.cpp @@ -5,7 +5,6 @@ using json = nlohmann::json; int main() { // create JSON values - json j_null; json j_boolean = true; json j_number_integer = 17; json j_number_float = 23.42; @@ -16,7 +15,6 @@ int main() json j_string = "Hello, world"; // call back() - //std::cout << j_null.back() << '\n'; // would throw std::cout << j_boolean.back() << '\n'; std::cout << j_number_integer.back() << '\n'; std::cout << j_number_float.back() << '\n'; @@ -25,4 +23,15 @@ int main() std::cout << j_array.back() << '\n'; //std::cout << j_array_empty.back() << '\n'; // undefined behavior std::cout << j_string.back() << '\n'; + + // back() called on a null value + try + { + json j_null; + j_null.back(); + } + catch (json::invalid_iterator& e) + { + std::cout << e.what() << '\n'; + } } diff --git a/doc/examples/back.link b/doc/examples/back.link index 0b009780..15b1102b 100644 --- a/doc/examples/back.link +++ b/doc/examples/back.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/back.output b/doc/examples/back.output index 159ba0fc..2990dbf0 100644 --- a/doc/examples/back.output +++ b/doc/examples/back.output @@ -4,3 +4,4 @@ true 2 16 "Hello, world" +[json.exception.invalid_iterator.214] cannot get value diff --git a/src/json.hpp b/src/json.hpp index 9fb7690f..5a3e6c3a 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -4411,7 +4411,8 @@ class basic_json assertions**). @post The JSON value remains unchanged. - @throw invalid_iterator.214 when called on `null` value. + @throw invalid_iterator.214 when called on a `null` value. See example + below. @liveexample{The following code shows an example for `back()`.,back} @@ -12769,10 +12770,18 @@ basic_json_parser_74: @complexity Constant. - @throw parse_error.106 if an array index begins with '0' - @throw parse_error.109 if an array index was not a number - @throw out_of_range.402 if the array index '-' is used - @throw out_of_range.404 if the JSON pointer can not be resolved + @throw parse_error.106 if an array index in the passed JSON pointer @a ptr + begins with '0' + + @throw parse_error.109 if an array index in the passed JSON pointer @a ptr + is not a number + + @throw out_of_range.402 if the array index `-` is used in the passed JSON + pointer @a ptr. As `at` provides checked access (and no elements are + implicitly inserted), the index `-` is always invalid. + + @throw out_of_range.404 if the JSON pointer @a ptr can not be resolved; + see example below. @liveexample{The behavior is shown in the example.,at_json_pointer} @@ -12795,10 +12804,18 @@ basic_json_parser_74: @complexity Constant. - @throw parse_error.106 if an array index begins with '0' - @throw parse_error.109 if an array index was not a number - @throw out_of_range.402 if the array index '-' is used - @throw out_of_range.404 if the JSON pointer can not be resolved + @throw parse_error.106 if an array index in the passed JSON pointer @a ptr + begins with '0' + + @throw parse_error.109 if an array index in the passed JSON pointer @a ptr + is not a number + + @throw out_of_range.402 if the array index `-` is used in the passed JSON + pointer @a ptr. As `at` provides checked access (and no elements are + implicitly inserted), the index `-` is always invalid. + + @throw out_of_range.404 if the JSON pointer @a ptr can not be resolved; + see example below. @liveexample{The behavior is shown in the example.,at_json_pointer_const} diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 22e12841..289e3f10 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -4411,7 +4411,8 @@ class basic_json assertions**). @post The JSON value remains unchanged. - @throw invalid_iterator.214 when called on `null` value. + @throw invalid_iterator.214 when called on a `null` value. See example + below. @liveexample{The following code shows an example for `back()`.,back} @@ -11802,10 +11803,18 @@ class basic_json @complexity Constant. - @throw parse_error.106 if an array index begins with '0' - @throw parse_error.109 if an array index was not a number - @throw out_of_range.402 if the array index '-' is used - @throw out_of_range.404 if the JSON pointer can not be resolved + @throw parse_error.106 if an array index in the passed JSON pointer @a ptr + begins with '0' + + @throw parse_error.109 if an array index in the passed JSON pointer @a ptr + is not a number + + @throw out_of_range.402 if the array index `-` is used in the passed JSON + pointer @a ptr. As `at` provides checked access (and no elements are + implicitly inserted), the index `-` is always invalid. + + @throw out_of_range.404 if the JSON pointer @a ptr can not be resolved; + see example below. @liveexample{The behavior is shown in the example.,at_json_pointer} @@ -11828,10 +11837,18 @@ class basic_json @complexity Constant. - @throw parse_error.106 if an array index begins with '0' - @throw parse_error.109 if an array index was not a number - @throw out_of_range.402 if the array index '-' is used - @throw out_of_range.404 if the JSON pointer can not be resolved + @throw parse_error.106 if an array index in the passed JSON pointer @a ptr + begins with '0' + + @throw parse_error.109 if an array index in the passed JSON pointer @a ptr + is not a number + + @throw out_of_range.402 if the array index `-` is used in the passed JSON + pointer @a ptr. As `at` provides checked access (and no elements are + implicitly inserted), the index `-` is always invalid. + + @throw out_of_range.404 if the JSON pointer @a ptr can not be resolved; + see example below. @liveexample{The behavior is shown in the example.,at_json_pointer_const} From 758c4addc1c6a010a3e9f3730419f9dfae9cd769 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Thu, 9 Mar 2017 18:20:26 +0100 Subject: [PATCH 057/530] :ambulance: fix for #492 The original test case relied on an invalidated iterator. This error did not occur before, but only with GCC with -D_GLIBCXX_DEBUG. This commit fixes the test case. The library is unaffected by this change. --- test/src/unit-modifiers.cpp | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/test/src/unit-modifiers.cpp b/test/src/unit-modifiers.cpp index 0edc9a12..2b5fdd71 100644 --- a/test/src/unit-modifiers.cpp +++ b/test/src/unit-modifiers.cpp @@ -583,10 +583,11 @@ TEST_CASE("modifiers") SECTION("insert nothing (count = 0)") { - auto pos = j_array.end(); auto it = j_array.insert(j_array.end(), 0, 5); CHECK(j_array.size() == 4); - CHECK(it == pos); + // the returned iterator points to the first inserted element; + // there were 4 elements, so it should point to the 5th + CHECK(it == j_array.begin() + 4); CHECK(j_array == json({1, 2, 3, 4})); } } From 9cfb777b654fbd96535e96069bc36f7bf92f476a Mon Sep 17 00:00:00 2001 From: Ted Lyngmo Date: Sat, 11 Mar 2017 13:56:38 +0100 Subject: [PATCH 058/530] Fix -Weffc++ warnings (GNU 6.3.1) --- src/json.hpp | 1156 +++++++++++------------------------------ src/json.hpp.re2c | 15 + test/src/unit-udt.cpp | 13 + 3 files changed, 340 insertions(+), 844 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index ff0ce5ad..f2e02628 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -6168,6 +6168,21 @@ class basic_json thousands_sep(!loc->thousands_sep ? '\0' : loc->thousands_sep[0]), decimal_point(!loc->decimal_point ? '\0' : loc->decimal_point[0]) {} + serializer(const serializer& cpy) + : o(cpy.o), + thousands_sep(cpy.thousands_sep), + decimal_point(cpy.decimal_point), + loc(cpy.loc), + indent_string(cpy.indent_string) + {} + serializer& operator=(const serializer& rhs) { + o = rhs.o; + thousands_sep = rhs.thousands_sep; + decimal_point = rhs.decimal_point; + loc = rhs.loc; + indent_string = rhs.indent_string; + return *this; + } /*! @brief internal implementation of the serialization function @@ -9859,12 +9874,11 @@ class basic_json { lexer_char_t yych; unsigned int yyaccept = 0; - static const unsigned char yybm[] = - { - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 32, 32, 0, 0, 32, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, + static const unsigned char yybm[] = { + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 32, 32, 0, 0, 32, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, 160, 128, 0, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 192, 192, 192, 192, 192, 192, 192, 192, @@ -9877,1012 +9891,466 @@ class basic_json 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, }; - if ((m_limit - m_cursor) < 5) - { - fill_line_buffer(5); // LCOV_EXCL_LINE - } + if ((m_limit - m_cursor) < 5) fill_line_buffer(5); // LCOV_EXCL_LINE yych = *m_cursor; - if (yybm[0 + yych] & 32) - { + if (yybm[0+yych] & 32) { goto basic_json_parser_6; } - if (yych <= '[') - { - if (yych <= '-') - { - if (yych <= '"') - { - if (yych <= 0x00) - { - goto basic_json_parser_2; - } - if (yych <= '!') - { - goto basic_json_parser_4; - } + if (yych <= '[') { + if (yych <= '-') { + if (yych <= '"') { + if (yych <= 0x00) goto basic_json_parser_2; + if (yych <= '!') goto basic_json_parser_4; goto basic_json_parser_9; - } - else - { - if (yych <= '+') - { - goto basic_json_parser_4; - } - if (yych <= ',') - { - goto basic_json_parser_10; - } + } else { + if (yych <= '+') goto basic_json_parser_4; + if (yych <= ',') goto basic_json_parser_10; goto basic_json_parser_12; } - } - else - { - if (yych <= '9') - { - if (yych <= '/') - { - goto basic_json_parser_4; - } - if (yych <= '0') - { - goto basic_json_parser_13; - } + } else { + if (yych <= '9') { + if (yych <= '/') goto basic_json_parser_4; + if (yych <= '0') goto basic_json_parser_13; goto basic_json_parser_15; - } - else - { - if (yych <= ':') - { - goto basic_json_parser_17; - } - if (yych <= 'Z') - { - goto basic_json_parser_4; - } + } else { + if (yych <= ':') goto basic_json_parser_17; + if (yych <= 'Z') goto basic_json_parser_4; goto basic_json_parser_19; } } - } - else - { - if (yych <= 'n') - { - if (yych <= 'e') - { - if (yych == ']') - { - goto basic_json_parser_21; - } + } else { + if (yych <= 'n') { + if (yych <= 'e') { + if (yych == ']') goto basic_json_parser_21; goto basic_json_parser_4; - } - else - { - if (yych <= 'f') - { - goto basic_json_parser_23; - } - if (yych <= 'm') - { - goto basic_json_parser_4; - } + } else { + if (yych <= 'f') goto basic_json_parser_23; + if (yych <= 'm') goto basic_json_parser_4; goto basic_json_parser_24; } - } - else - { - if (yych <= 'z') - { - if (yych == 't') - { - goto basic_json_parser_25; - } + } else { + if (yych <= 'z') { + if (yych == 't') goto basic_json_parser_25; goto basic_json_parser_4; - } - else - { - if (yych <= '{') - { - goto basic_json_parser_26; - } - if (yych == '}') - { - goto basic_json_parser_28; - } + } else { + if (yych <= '{') goto basic_json_parser_26; + if (yych == '}') goto basic_json_parser_28; goto basic_json_parser_4; } } } -basic_json_parser_2: + basic_json_parser_2: ++m_cursor; - { - last_token_type = token_type::end_of_input; - break; - } -basic_json_parser_4: + { last_token_type = token_type::end_of_input; break; } + basic_json_parser_4: ++m_cursor; -basic_json_parser_5: - { - last_token_type = token_type::parse_error; - break; - } -basic_json_parser_6: + basic_json_parser_5: + { last_token_type = token_type::parse_error; break; } + basic_json_parser_6: ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } + if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE yych = *m_cursor; - if (yybm[0 + yych] & 32) - { + if (yybm[0+yych] & 32) { goto basic_json_parser_6; } - { - continue; - } -basic_json_parser_9: + { continue; } + basic_json_parser_9: yyaccept = 0; yych = *(m_marker = ++m_cursor); - if (yych <= 0x1F) - { - goto basic_json_parser_5; - } - if (yych <= 0x7F) - { - goto basic_json_parser_31; - } - if (yych <= 0xC1) - { - goto basic_json_parser_5; - } - if (yych <= 0xF4) - { - goto basic_json_parser_31; - } + if (yych <= 0x1F) goto basic_json_parser_5; + if (yych <= 0x7F) goto basic_json_parser_31; + if (yych <= 0xC1) goto basic_json_parser_5; + if (yych <= 0xF4) goto basic_json_parser_31; goto basic_json_parser_5; -basic_json_parser_10: + basic_json_parser_10: ++m_cursor; - { - last_token_type = token_type::value_separator; - break; - } -basic_json_parser_12: + { last_token_type = token_type::value_separator; break; } + basic_json_parser_12: yych = *++m_cursor; - if (yych <= '/') - { - goto basic_json_parser_5; - } - if (yych <= '0') - { - goto basic_json_parser_43; - } - if (yych <= '9') - { - goto basic_json_parser_45; - } + if (yych <= '/') goto basic_json_parser_5; + if (yych <= '0') goto basic_json_parser_43; + if (yych <= '9') goto basic_json_parser_45; goto basic_json_parser_5; -basic_json_parser_13: + basic_json_parser_13: yyaccept = 1; yych = *(m_marker = ++m_cursor); - if (yych <= '9') - { - if (yych == '.') - { - goto basic_json_parser_47; - } - if (yych >= '0') - { - goto basic_json_parser_48; + if (yych <= '9') { + if (yych == '.') goto basic_json_parser_47; + if (yych >= '0') goto basic_json_parser_48; + } else { + if (yych <= 'E') { + if (yych >= 'E') goto basic_json_parser_51; + } else { + if (yych == 'e') goto basic_json_parser_51; } } - else - { - if (yych <= 'E') - { - if (yych >= 'E') - { - goto basic_json_parser_51; - } - } - else - { - if (yych == 'e') - { - goto basic_json_parser_51; - } - } - } -basic_json_parser_14: - { - last_token_type = token_type::value_unsigned; - break; - } -basic_json_parser_15: + basic_json_parser_14: + { last_token_type = token_type::value_unsigned; break; } + basic_json_parser_15: yyaccept = 1; m_marker = ++m_cursor; - if ((m_limit - m_cursor) < 3) - { - fill_line_buffer(3); // LCOV_EXCL_LINE - } + if ((m_limit - m_cursor) < 3) fill_line_buffer(3); // LCOV_EXCL_LINE yych = *m_cursor; - if (yybm[0 + yych] & 64) - { + if (yybm[0+yych] & 64) { goto basic_json_parser_15; } - if (yych <= 'D') - { - if (yych == '.') - { - goto basic_json_parser_47; - } + if (yych <= 'D') { + if (yych == '.') goto basic_json_parser_47; + goto basic_json_parser_14; + } else { + if (yych <= 'E') goto basic_json_parser_51; + if (yych == 'e') goto basic_json_parser_51; goto basic_json_parser_14; } - else - { - if (yych <= 'E') - { - goto basic_json_parser_51; - } - if (yych == 'e') - { - goto basic_json_parser_51; - } - goto basic_json_parser_14; - } -basic_json_parser_17: + basic_json_parser_17: ++m_cursor; - { - last_token_type = token_type::name_separator; - break; - } -basic_json_parser_19: + { last_token_type = token_type::name_separator; break; } + basic_json_parser_19: ++m_cursor; - { - last_token_type = token_type::begin_array; - break; - } -basic_json_parser_21: + { last_token_type = token_type::begin_array; break; } + basic_json_parser_21: ++m_cursor; - { - last_token_type = token_type::end_array; - break; - } -basic_json_parser_23: + { last_token_type = token_type::end_array; break; } + basic_json_parser_23: yyaccept = 0; yych = *(m_marker = ++m_cursor); - if (yych == 'a') - { - goto basic_json_parser_52; - } + if (yych == 'a') goto basic_json_parser_52; goto basic_json_parser_5; -basic_json_parser_24: + basic_json_parser_24: yyaccept = 0; yych = *(m_marker = ++m_cursor); - if (yych == 'u') - { - goto basic_json_parser_53; - } + if (yych == 'u') goto basic_json_parser_53; goto basic_json_parser_5; -basic_json_parser_25: + basic_json_parser_25: yyaccept = 0; yych = *(m_marker = ++m_cursor); - if (yych == 'r') - { - goto basic_json_parser_54; - } + if (yych == 'r') goto basic_json_parser_54; goto basic_json_parser_5; -basic_json_parser_26: + basic_json_parser_26: ++m_cursor; - { - last_token_type = token_type::begin_object; - break; - } -basic_json_parser_28: + { last_token_type = token_type::begin_object; break; } + basic_json_parser_28: ++m_cursor; - { - last_token_type = token_type::end_object; - break; - } -basic_json_parser_30: + { last_token_type = token_type::end_object; break; } + basic_json_parser_30: ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } + if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE yych = *m_cursor; -basic_json_parser_31: - if (yybm[0 + yych] & 128) - { + basic_json_parser_31: + if (yybm[0+yych] & 128) { goto basic_json_parser_30; } - if (yych <= 0xE0) - { - if (yych <= '\\') - { - if (yych <= 0x1F) - { - goto basic_json_parser_32; - } - if (yych <= '"') - { - goto basic_json_parser_33; - } + if (yych <= 0xE0) { + if (yych <= '\\') { + if (yych <= 0x1F) goto basic_json_parser_32; + if (yych <= '"') goto basic_json_parser_33; goto basic_json_parser_35; - } - else - { - if (yych <= 0xC1) - { - goto basic_json_parser_32; - } - if (yych <= 0xDF) - { - goto basic_json_parser_36; - } + } else { + if (yych <= 0xC1) goto basic_json_parser_32; + if (yych <= 0xDF) goto basic_json_parser_36; goto basic_json_parser_37; } - } - else - { - if (yych <= 0xEF) - { - if (yych == 0xED) - { - goto basic_json_parser_39; - } + } else { + if (yych <= 0xEF) { + if (yych == 0xED) goto basic_json_parser_39; goto basic_json_parser_38; - } - else - { - if (yych <= 0xF0) - { - goto basic_json_parser_40; - } - if (yych <= 0xF3) - { - goto basic_json_parser_41; - } - if (yych <= 0xF4) - { - goto basic_json_parser_42; - } + } else { + if (yych <= 0xF0) goto basic_json_parser_40; + if (yych <= 0xF3) goto basic_json_parser_41; + if (yych <= 0xF4) goto basic_json_parser_42; } } -basic_json_parser_32: + basic_json_parser_32: m_cursor = m_marker; - if (yyaccept <= 1) - { - if (yyaccept == 0) - { + if (yyaccept <= 1) { + if (yyaccept == 0) { goto basic_json_parser_5; - } - else - { + } else { goto basic_json_parser_14; } - } - else - { - if (yyaccept == 2) - { + } else { + if (yyaccept == 2) { goto basic_json_parser_44; - } - else - { + } else { goto basic_json_parser_58; } } -basic_json_parser_33: + basic_json_parser_33: ++m_cursor; - { - last_token_type = token_type::value_string; - break; - } -basic_json_parser_35: + { last_token_type = token_type::value_string; break; } + basic_json_parser_35: ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } + if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE yych = *m_cursor; - if (yych <= 'e') - { - if (yych <= '/') - { - if (yych == '"') - { - goto basic_json_parser_30; - } - if (yych <= '.') - { - goto basic_json_parser_32; - } + if (yych <= 'e') { + if (yych <= '/') { + if (yych == '"') goto basic_json_parser_30; + if (yych <= '.') goto basic_json_parser_32; goto basic_json_parser_30; - } - else - { - if (yych <= '\\') - { - if (yych <= '[') - { - goto basic_json_parser_32; - } + } else { + if (yych <= '\\') { + if (yych <= '[') goto basic_json_parser_32; goto basic_json_parser_30; - } - else - { - if (yych == 'b') - { - goto basic_json_parser_30; - } + } else { + if (yych == 'b') goto basic_json_parser_30; goto basic_json_parser_32; } } - } - else - { - if (yych <= 'q') - { - if (yych <= 'f') - { - goto basic_json_parser_30; - } - if (yych == 'n') - { - goto basic_json_parser_30; - } + } else { + if (yych <= 'q') { + if (yych <= 'f') goto basic_json_parser_30; + if (yych == 'n') goto basic_json_parser_30; goto basic_json_parser_32; - } - else - { - if (yych <= 's') - { - if (yych <= 'r') - { - goto basic_json_parser_30; - } + } else { + if (yych <= 's') { + if (yych <= 'r') goto basic_json_parser_30; goto basic_json_parser_32; - } - else - { - if (yych <= 't') - { - goto basic_json_parser_30; - } - if (yych <= 'u') - { - goto basic_json_parser_55; - } + } else { + if (yych <= 't') goto basic_json_parser_30; + if (yych <= 'u') goto basic_json_parser_55; goto basic_json_parser_32; } } } -basic_json_parser_36: + basic_json_parser_36: ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } + if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE yych = *m_cursor; - if (yych <= 0x7F) - { - goto basic_json_parser_32; - } - if (yych <= 0xBF) - { - goto basic_json_parser_30; - } + if (yych <= 0x7F) goto basic_json_parser_32; + if (yych <= 0xBF) goto basic_json_parser_30; goto basic_json_parser_32; -basic_json_parser_37: + basic_json_parser_37: ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } + if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE yych = *m_cursor; - if (yych <= 0x9F) - { - goto basic_json_parser_32; - } - if (yych <= 0xBF) - { - goto basic_json_parser_36; - } + if (yych <= 0x9F) goto basic_json_parser_32; + if (yych <= 0xBF) goto basic_json_parser_36; goto basic_json_parser_32; -basic_json_parser_38: + basic_json_parser_38: ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } + if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE yych = *m_cursor; - if (yych <= 0x7F) - { - goto basic_json_parser_32; - } - if (yych <= 0xBF) - { - goto basic_json_parser_36; - } + if (yych <= 0x7F) goto basic_json_parser_32; + if (yych <= 0xBF) goto basic_json_parser_36; goto basic_json_parser_32; -basic_json_parser_39: + basic_json_parser_39: ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } + if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE yych = *m_cursor; - if (yych <= 0x7F) - { - goto basic_json_parser_32; - } - if (yych <= 0x9F) - { - goto basic_json_parser_36; - } + if (yych <= 0x7F) goto basic_json_parser_32; + if (yych <= 0x9F) goto basic_json_parser_36; goto basic_json_parser_32; -basic_json_parser_40: + basic_json_parser_40: ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } + if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE yych = *m_cursor; - if (yych <= 0x8F) - { - goto basic_json_parser_32; - } - if (yych <= 0xBF) - { - goto basic_json_parser_38; - } + if (yych <= 0x8F) goto basic_json_parser_32; + if (yych <= 0xBF) goto basic_json_parser_38; goto basic_json_parser_32; -basic_json_parser_41: + basic_json_parser_41: ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } + if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE yych = *m_cursor; - if (yych <= 0x7F) - { - goto basic_json_parser_32; - } - if (yych <= 0xBF) - { - goto basic_json_parser_38; - } + if (yych <= 0x7F) goto basic_json_parser_32; + if (yych <= 0xBF) goto basic_json_parser_38; goto basic_json_parser_32; -basic_json_parser_42: + basic_json_parser_42: ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } + if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE yych = *m_cursor; - if (yych <= 0x7F) - { - goto basic_json_parser_32; - } - if (yych <= 0x8F) - { - goto basic_json_parser_38; - } + if (yych <= 0x7F) goto basic_json_parser_32; + if (yych <= 0x8F) goto basic_json_parser_38; goto basic_json_parser_32; -basic_json_parser_43: + basic_json_parser_43: yyaccept = 2; yych = *(m_marker = ++m_cursor); - if (yych <= '9') - { - if (yych == '.') - { - goto basic_json_parser_47; - } - if (yych >= '0') - { - goto basic_json_parser_48; + if (yych <= '9') { + if (yych == '.') goto basic_json_parser_47; + if (yych >= '0') goto basic_json_parser_48; + } else { + if (yych <= 'E') { + if (yych >= 'E') goto basic_json_parser_51; + } else { + if (yych == 'e') goto basic_json_parser_51; } } - else - { - if (yych <= 'E') - { - if (yych >= 'E') - { - goto basic_json_parser_51; - } - } - else - { - if (yych == 'e') - { - goto basic_json_parser_51; - } - } - } -basic_json_parser_44: - { - last_token_type = token_type::value_integer; - break; - } -basic_json_parser_45: + basic_json_parser_44: + { last_token_type = token_type::value_integer; break; } + basic_json_parser_45: yyaccept = 2; m_marker = ++m_cursor; - if ((m_limit - m_cursor) < 3) - { - fill_line_buffer(3); // LCOV_EXCL_LINE - } + if ((m_limit - m_cursor) < 3) fill_line_buffer(3); // LCOV_EXCL_LINE yych = *m_cursor; - if (yych <= '9') - { - if (yych == '.') - { - goto basic_json_parser_47; - } - if (yych <= '/') - { - goto basic_json_parser_44; - } + if (yych <= '9') { + if (yych == '.') goto basic_json_parser_47; + if (yych <= '/') goto basic_json_parser_44; goto basic_json_parser_45; - } - else - { - if (yych <= 'E') - { - if (yych <= 'D') - { - goto basic_json_parser_44; - } + } else { + if (yych <= 'E') { + if (yych <= 'D') goto basic_json_parser_44; goto basic_json_parser_51; - } - else - { - if (yych == 'e') - { - goto basic_json_parser_51; - } + } else { + if (yych == 'e') goto basic_json_parser_51; goto basic_json_parser_44; } } -basic_json_parser_47: + basic_json_parser_47: yych = *++m_cursor; - if (yych <= '/') - { - goto basic_json_parser_32; - } - if (yych <= '9') - { - goto basic_json_parser_56; - } + if (yych <= '/') goto basic_json_parser_32; + if (yych <= '9') goto basic_json_parser_56; goto basic_json_parser_32; -basic_json_parser_48: + basic_json_parser_48: ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } + if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE yych = *m_cursor; - if (yych <= '/') - { - goto basic_json_parser_50; - } - if (yych <= '9') - { - goto basic_json_parser_48; - } -basic_json_parser_50: - { - last_token_type = token_type::parse_error; - break; - } -basic_json_parser_51: + if (yych <= '/') goto basic_json_parser_50; + if (yych <= '9') goto basic_json_parser_48; + basic_json_parser_50: + { last_token_type = token_type::parse_error; break; } + basic_json_parser_51: yych = *++m_cursor; - if (yych <= ',') - { - if (yych == '+') - { - goto basic_json_parser_59; - } + if (yych <= ',') { + if (yych == '+') goto basic_json_parser_59; + goto basic_json_parser_32; + } else { + if (yych <= '-') goto basic_json_parser_59; + if (yych <= '/') goto basic_json_parser_32; + if (yych <= '9') goto basic_json_parser_60; goto basic_json_parser_32; } - else - { - if (yych <= '-') - { - goto basic_json_parser_59; - } - if (yych <= '/') - { - goto basic_json_parser_32; - } - if (yych <= '9') - { - goto basic_json_parser_60; - } - goto basic_json_parser_32; - } -basic_json_parser_52: + basic_json_parser_52: yych = *++m_cursor; - if (yych == 'l') - { - goto basic_json_parser_62; - } + if (yych == 'l') goto basic_json_parser_62; goto basic_json_parser_32; -basic_json_parser_53: + basic_json_parser_53: yych = *++m_cursor; - if (yych == 'l') - { - goto basic_json_parser_63; - } + if (yych == 'l') goto basic_json_parser_63; goto basic_json_parser_32; -basic_json_parser_54: + basic_json_parser_54: yych = *++m_cursor; - if (yych == 'u') - { - goto basic_json_parser_64; - } + if (yych == 'u') goto basic_json_parser_64; goto basic_json_parser_32; -basic_json_parser_55: + basic_json_parser_55: ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } + if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE yych = *m_cursor; - if (yych <= '@') - { - if (yych <= '/') - { - goto basic_json_parser_32; - } - if (yych <= '9') - { - goto basic_json_parser_65; - } + if (yych <= '@') { + if (yych <= '/') goto basic_json_parser_32; + if (yych <= '9') goto basic_json_parser_65; + goto basic_json_parser_32; + } else { + if (yych <= 'F') goto basic_json_parser_65; + if (yych <= '`') goto basic_json_parser_32; + if (yych <= 'f') goto basic_json_parser_65; goto basic_json_parser_32; } - else - { - if (yych <= 'F') - { - goto basic_json_parser_65; - } - if (yych <= '`') - { - goto basic_json_parser_32; - } - if (yych <= 'f') - { - goto basic_json_parser_65; - } - goto basic_json_parser_32; - } -basic_json_parser_56: + basic_json_parser_56: yyaccept = 3; m_marker = ++m_cursor; - if ((m_limit - m_cursor) < 3) - { - fill_line_buffer(3); // LCOV_EXCL_LINE - } + if ((m_limit - m_cursor) < 3) fill_line_buffer(3); // LCOV_EXCL_LINE yych = *m_cursor; - if (yych <= 'D') - { - if (yych <= '/') - { - goto basic_json_parser_58; - } - if (yych <= '9') - { - goto basic_json_parser_56; - } + if (yych <= 'D') { + if (yych <= '/') goto basic_json_parser_58; + if (yych <= '9') goto basic_json_parser_56; + } else { + if (yych <= 'E') goto basic_json_parser_51; + if (yych == 'e') goto basic_json_parser_51; } - else - { - if (yych <= 'E') - { - goto basic_json_parser_51; - } - if (yych == 'e') - { - goto basic_json_parser_51; - } - } -basic_json_parser_58: - { - last_token_type = token_type::value_float; - break; - } -basic_json_parser_59: + basic_json_parser_58: + { last_token_type = token_type::value_float; break; } + basic_json_parser_59: yych = *++m_cursor; - if (yych <= '/') - { - goto basic_json_parser_32; - } - if (yych >= ':') - { - goto basic_json_parser_32; - } -basic_json_parser_60: + if (yych <= '/') goto basic_json_parser_32; + if (yych >= ':') goto basic_json_parser_32; + basic_json_parser_60: ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } + if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE yych = *m_cursor; - if (yych <= '/') - { - goto basic_json_parser_58; - } - if (yych <= '9') - { - goto basic_json_parser_60; - } + if (yych <= '/') goto basic_json_parser_58; + if (yych <= '9') goto basic_json_parser_60; goto basic_json_parser_58; -basic_json_parser_62: + basic_json_parser_62: yych = *++m_cursor; - if (yych == 's') - { - goto basic_json_parser_66; - } + if (yych == 's') goto basic_json_parser_66; goto basic_json_parser_32; -basic_json_parser_63: + basic_json_parser_63: yych = *++m_cursor; - if (yych == 'l') - { - goto basic_json_parser_67; - } + if (yych == 'l') goto basic_json_parser_67; goto basic_json_parser_32; -basic_json_parser_64: + basic_json_parser_64: yych = *++m_cursor; - if (yych == 'e') - { - goto basic_json_parser_69; - } + if (yych == 'e') goto basic_json_parser_69; goto basic_json_parser_32; -basic_json_parser_65: + basic_json_parser_65: ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } + if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE yych = *m_cursor; - if (yych <= '@') - { - if (yych <= '/') - { - goto basic_json_parser_32; - } - if (yych <= '9') - { - goto basic_json_parser_71; - } + if (yych <= '@') { + if (yych <= '/') goto basic_json_parser_32; + if (yych <= '9') goto basic_json_parser_71; + goto basic_json_parser_32; + } else { + if (yych <= 'F') goto basic_json_parser_71; + if (yych <= '`') goto basic_json_parser_32; + if (yych <= 'f') goto basic_json_parser_71; goto basic_json_parser_32; } - else - { - if (yych <= 'F') - { - goto basic_json_parser_71; - } - if (yych <= '`') - { - goto basic_json_parser_32; - } - if (yych <= 'f') - { - goto basic_json_parser_71; - } - goto basic_json_parser_32; - } -basic_json_parser_66: + basic_json_parser_66: yych = *++m_cursor; - if (yych == 'e') - { - goto basic_json_parser_72; - } + if (yych == 'e') goto basic_json_parser_72; goto basic_json_parser_32; -basic_json_parser_67: + basic_json_parser_67: ++m_cursor; - { - last_token_type = token_type::literal_null; - break; - } -basic_json_parser_69: + { last_token_type = token_type::literal_null; break; } + basic_json_parser_69: ++m_cursor; - { - last_token_type = token_type::literal_true; - break; - } -basic_json_parser_71: + { last_token_type = token_type::literal_true; break; } + basic_json_parser_71: ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } + if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE yych = *m_cursor; - if (yych <= '@') - { - if (yych <= '/') - { - goto basic_json_parser_32; - } - if (yych <= '9') - { - goto basic_json_parser_74; - } + if (yych <= '@') { + if (yych <= '/') goto basic_json_parser_32; + if (yych <= '9') goto basic_json_parser_74; + goto basic_json_parser_32; + } else { + if (yych <= 'F') goto basic_json_parser_74; + if (yych <= '`') goto basic_json_parser_32; + if (yych <= 'f') goto basic_json_parser_74; goto basic_json_parser_32; } - else - { - if (yych <= 'F') - { - goto basic_json_parser_74; - } - if (yych <= '`') - { - goto basic_json_parser_32; - } - if (yych <= 'f') - { - goto basic_json_parser_74; - } - goto basic_json_parser_32; - } -basic_json_parser_72: + basic_json_parser_72: ++m_cursor; - { - last_token_type = token_type::literal_false; - break; - } -basic_json_parser_74: + { last_token_type = token_type::literal_false; break; } + basic_json_parser_74: ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } + if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE yych = *m_cursor; - if (yych <= '@') - { - if (yych <= '/') - { - goto basic_json_parser_32; - } - if (yych <= '9') - { - goto basic_json_parser_30; - } + if (yych <= '@') { + if (yych <= '/') goto basic_json_parser_32; + if (yych <= '9') goto basic_json_parser_30; goto basic_json_parser_32; - } - else - { - if (yych <= 'F') - { - goto basic_json_parser_30; - } - if (yych <= '`') - { - goto basic_json_parser_32; - } - if (yych <= 'f') - { - goto basic_json_parser_30; - } + } else { + if (yych <= 'F') goto basic_json_parser_30; + if (yych <= '`') goto basic_json_parser_32; + if (yych <= 'f') goto basic_json_parser_30; goto basic_json_parser_32; } } diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index fc7cf965..3a53f16c 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -6168,6 +6168,21 @@ class basic_json thousands_sep(!loc->thousands_sep ? '\0' : loc->thousands_sep[0]), decimal_point(!loc->decimal_point ? '\0' : loc->decimal_point[0]) {} + serializer(const serializer& cpy) + : o(cpy.o), + thousands_sep(cpy.thousands_sep), + decimal_point(cpy.decimal_point), + loc(cpy.loc), + indent_string(cpy.indent_string) + {} + serializer& operator=(const serializer& rhs) { + o = rhs.o; + thousands_sep = rhs.thousands_sep; + decimal_point = rhs.decimal_point; + loc = rhs.loc; + indent_string = rhs.indent_string; + return *this; + } /*! @brief internal implementation of the serialization function diff --git a/test/src/unit-udt.cpp b/test/src/unit-udt.cpp index a8103082..6b37a106 100644 --- a/test/src/unit-udt.cpp +++ b/test/src/unit-udt.cpp @@ -49,16 +49,19 @@ enum class country struct age { int m_val; + age(int rhs=0) : m_val(rhs) {} }; struct name { std::string m_val; + name(const std::string rhs="") : m_val(rhs) {} }; struct address { std::string m_val; + address(const std::string rhs="") : m_val(rhs) {} }; struct person @@ -66,18 +69,24 @@ struct person age m_age; name m_name; country m_country; + person() : m_age(),m_name(),m_country() {} + person(const age& a, const name& n, const country& c) : m_age(a), m_name(n), m_country(c) {} }; struct contact { person m_person; address m_address; + contact() : m_person(), m_address() {} + contact(const person& p, const address& a) : m_person(p), m_address(a) {} }; struct contact_book { name m_book_name; std::vector m_contacts; + contact_book() : m_book_name(), m_contacts() {} + contact_book(const name& n, const std::vector& c) : m_book_name(n), m_contacts(c) {} }; } @@ -319,6 +328,8 @@ namespace udt struct legacy_type { std::string number; + legacy_type() : number() {} + legacy_type(const std::string& n) : number(n) {} }; } @@ -593,6 +604,8 @@ struct small_pod struct non_pod { std::string s; + non_pod() : s() {} + non_pod(const std::string& S) : s(S) {} }; template From 5b809b97374488c628c8cb7e27d614cb19cfb24a Mon Sep 17 00:00:00 2001 From: Ted Lyngmo Date: Sat, 11 Mar 2017 13:56:38 +0100 Subject: [PATCH 059/530] Fix -Weffc++ warnings (GNU 6.3.1) --- src/json.hpp | 4 ++++ src/json.hpp.re2c | 4 ++++ 2 files changed, 8 insertions(+) diff --git a/src/json.hpp b/src/json.hpp index f2e02628..8883acd9 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -6159,6 +6159,10 @@ class basic_json */ class serializer { + private: + serializer(const serializer&) = delete; + serializer& operator=(const serializer&) = delete; + public: /*! @param[in] s output stream to serialize to diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 3a53f16c..6e90e6b8 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -6159,6 +6159,10 @@ class basic_json */ class serializer { + private: + serializer(const serializer&) = delete; + serializer& operator=(const serializer&) = delete; + public: /*! @param[in] s output stream to serialize to From 70bbf1952739f2890d1d19186eb1dcad11b23486 Mon Sep 17 00:00:00 2001 From: Ted Lyngmo Date: Sat, 11 Mar 2017 13:56:38 +0100 Subject: [PATCH 060/530] Fix -Weffc++ warnings (GNU 6.3.1) --- src/json.hpp | 15 --------------- src/json.hpp.re2c | 15 --------------- 2 files changed, 30 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 8883acd9..6d5b5586 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -6172,21 +6172,6 @@ class basic_json thousands_sep(!loc->thousands_sep ? '\0' : loc->thousands_sep[0]), decimal_point(!loc->decimal_point ? '\0' : loc->decimal_point[0]) {} - serializer(const serializer& cpy) - : o(cpy.o), - thousands_sep(cpy.thousands_sep), - decimal_point(cpy.decimal_point), - loc(cpy.loc), - indent_string(cpy.indent_string) - {} - serializer& operator=(const serializer& rhs) { - o = rhs.o; - thousands_sep = rhs.thousands_sep; - decimal_point = rhs.decimal_point; - loc = rhs.loc; - indent_string = rhs.indent_string; - return *this; - } /*! @brief internal implementation of the serialization function diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 6e90e6b8..21639efc 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -6172,21 +6172,6 @@ class basic_json thousands_sep(!loc->thousands_sep ? '\0' : loc->thousands_sep[0]), decimal_point(!loc->decimal_point ? '\0' : loc->decimal_point[0]) {} - serializer(const serializer& cpy) - : o(cpy.o), - thousands_sep(cpy.thousands_sep), - decimal_point(cpy.decimal_point), - loc(cpy.loc), - indent_string(cpy.indent_string) - {} - serializer& operator=(const serializer& rhs) { - o = rhs.o; - thousands_sep = rhs.thousands_sep; - decimal_point = rhs.decimal_point; - loc = rhs.loc; - indent_string = rhs.indent_string; - return *this; - } /*! @brief internal implementation of the serialization function From f5f6dac8009fb481118be94ce18e043a057bd036 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sat, 11 Mar 2017 15:32:44 +0100 Subject: [PATCH 061/530] :sparkles: added overload for std::vector #494 Adds a to_json function for std::vector to allow implicit conversion from bit vectors to basic_json. --- src/json.hpp | 19 +++++++++++++++++++ src/json.hpp.re2c | 19 +++++++++++++++++++ test/src/unit-regression.cpp | 9 +++++++++ 3 files changed, 47 insertions(+) diff --git a/src/json.hpp b/src/json.hpp index ff0ce5ad..432cf553 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -324,6 +324,19 @@ struct external_constructor j.m_value.array = j.template create(begin(arr), end(arr)); j.assert_invariant(); } + + template + static void construct(BasicJsonType& j, const std::vector& arr) + { + j.m_type = value_t::array; + j.m_value = value_t::array; + j.m_value.array->reserve(arr.size()); + for (bool x : arr) + { + j.m_value.array->push_back(x); + } + j.assert_invariant(); + } }; template<> @@ -562,6 +575,12 @@ void to_json(BasicJsonType& j, UnscopedEnumType e) noexcept external_constructor::construct(j, e); } +template +void to_json(BasicJsonType& j, std::vector e) noexcept +{ + external_constructor::construct(j, e); +} + template < typename BasicJsonType, typename CompatibleArrayType, enable_if_t < diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index fc7cf965..21b4fd3a 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -324,6 +324,19 @@ struct external_constructor j.m_value.array = j.template create(begin(arr), end(arr)); j.assert_invariant(); } + + template + static void construct(BasicJsonType& j, const std::vector& arr) + { + j.m_type = value_t::array; + j.m_value = value_t::array; + j.m_value.array->reserve(arr.size()); + for (bool x : arr) + { + j.m_value.array->push_back(x); + } + j.assert_invariant(); + } }; template<> @@ -562,6 +575,12 @@ void to_json(BasicJsonType& j, UnscopedEnumType e) noexcept external_constructor::construct(j, e); } +template +void to_json(BasicJsonType& j, std::vector e) noexcept +{ + external_constructor::construct(j, e); +} + template < typename BasicJsonType, typename CompatibleArrayType, enable_if_t < diff --git a/test/src/unit-regression.cpp b/test/src/unit-regression.cpp index 7980371b..bfd2c954 100644 --- a/test/src/unit-regression.cpp +++ b/test/src/unit-regression.cpp @@ -795,4 +795,13 @@ TEST_CASE("regression tests") std::string s2 = j2.dump(); CHECK(s1 == s2); } + + SECTION("issue #494 - conversion from vector to json fails to build") + { + std::vector boolVector = {false, true, false, false}; + json j; + j["bool_vector"] = boolVector; + + CHECK(j["bool_vector"].dump() == "[false,true,false,false]"); + } } From f4126e4dd84dfb723acb34dd3dbfcd6364b11fb2 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sat, 11 Mar 2017 15:44:14 +0100 Subject: [PATCH 062/530] :sparkles: added overload for std::less #486 MSVC needs this overload to compile code containing a std::map that uses nlohmann::detail::operator as key. --- src/json.hpp | 16 ++++++++++++++++ src/json.hpp.re2c | 16 ++++++++++++++++ test/src/unit-regression.cpp | 7 +++++++ 3 files changed, 39 insertions(+) diff --git a/src/json.hpp b/src/json.hpp index ff0ce5ad..6f52255b 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -13025,6 +13025,22 @@ struct hash return h(j.dump()); } }; + +/// specialization for std::less +template <> +struct less<::nlohmann::detail::value_t> +{ + /*! + @brief compare two value_t enum values + @since version 3.0.0 + */ + bool operator()(nlohmann::detail::value_t lhs, + nlohmann::detail::value_t rhs) const noexcept + { + return nlohmann::detail::operator<(lhs, rhs); + } +}; + } // namespace std /*! diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index fc7cf965..7c2c24bf 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -12059,6 +12059,22 @@ struct hash return h(j.dump()); } }; + +/// specialization for std::less +template <> +struct less<::nlohmann::detail::value_t> +{ + /*! + @brief compare two value_t enum values + @since version 3.0.0 + */ + bool operator()(nlohmann::detail::value_t lhs, + nlohmann::detail::value_t rhs) const noexcept + { + return nlohmann::detail::operator<(lhs, rhs); + } +}; + } // namespace std /*! diff --git a/test/src/unit-regression.cpp b/test/src/unit-regression.cpp index 7980371b..f7b56050 100644 --- a/test/src/unit-regression.cpp +++ b/test/src/unit-regression.cpp @@ -795,4 +795,11 @@ TEST_CASE("regression tests") std::string s2 = j2.dump(); CHECK(s1 == s2); } + + SECTION("issue #486 - json::value_t can't be a map's key type in VC++ 2015") + { + // the code below must compile with MSVC + std::map jsonTypes ; + jsonTypes[json::value_t::array] = "array"; + } } From d9e2dd03bf4faabe63f09c7853939d67eb04539d Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sat, 11 Mar 2017 15:46:52 +0100 Subject: [PATCH 063/530] :hammer: fixed interface for to_json function --- src/json.hpp | 2 +- src/json.hpp.re2c | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 432cf553..d3601078 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -576,7 +576,7 @@ void to_json(BasicJsonType& j, UnscopedEnumType e) noexcept } template -void to_json(BasicJsonType& j, std::vector e) noexcept +void to_json(BasicJsonType& j, const std::vector& e) { external_constructor::construct(j, e); } diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 21b4fd3a..1560f1ef 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -576,7 +576,7 @@ void to_json(BasicJsonType& j, UnscopedEnumType e) noexcept } template -void to_json(BasicJsonType& j, std::vector e) noexcept +void to_json(BasicJsonType& j, const std::vector& e) { external_constructor::construct(j, e); } From c7afb34e57fc7d343a6c6f3ddee859fe92a7cf92 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sat, 11 Mar 2017 16:01:26 +0100 Subject: [PATCH 064/530] :art: cleanup after #496 MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Ran “make pretty” and added a note to the README file. --- README.md | 2 +- src/json.hpp | 1183 ++++++++++++++++++++++++++++++----------- test/src/unit-udt.cpp | 8 +- 3 files changed, 870 insertions(+), 323 deletions(-) diff --git a/README.md b/README.md index 94120db6..2a896210 100644 --- a/README.md +++ b/README.md @@ -828,7 +828,7 @@ I deeply appreciate the help of the following people. - [rswanson-ihi](https://github.com/rswanson-ihi) noted a typo in the README. - [Mihai Stan](https://github.com/stanmihai4) fixed a bug in the comparison with `nullptr`s. - [Tushar Maheshwari](https://github.com/tusharpm) added [cotire](https://github.com/sakra/cotire) support to speed up the compilation. -- [TedLyngmo](https://github.com/TedLyngmo) noted a typo in the README. +- [TedLyngmo](https://github.com/TedLyngmo) noted a typo in the README and fixed some `-Weffc++` warnings. Thanks a lot for helping out! Please [let me know](mailto:mail@nlohmann.me) if I forgot someone. diff --git a/src/json.hpp b/src/json.hpp index 6d5b5586..32ba8d10 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -9863,11 +9863,12 @@ class basic_json { lexer_char_t yych; unsigned int yyaccept = 0; - static const unsigned char yybm[] = { - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 32, 32, 0, 0, 32, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, + static const unsigned char yybm[] = + { + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 32, 32, 0, 0, 32, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, 160, 128, 0, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 192, 192, 192, 192, 192, 192, 192, 192, @@ -9880,466 +9881,1012 @@ class basic_json 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, 128, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, }; - if ((m_limit - m_cursor) < 5) fill_line_buffer(5); // LCOV_EXCL_LINE + if ((m_limit - m_cursor) < 5) + { + fill_line_buffer(5); // LCOV_EXCL_LINE + } yych = *m_cursor; - if (yybm[0+yych] & 32) { + if (yybm[0 + yych] & 32) + { goto basic_json_parser_6; } - if (yych <= '[') { - if (yych <= '-') { - if (yych <= '"') { - if (yych <= 0x00) goto basic_json_parser_2; - if (yych <= '!') goto basic_json_parser_4; + if (yych <= '[') + { + if (yych <= '-') + { + if (yych <= '"') + { + if (yych <= 0x00) + { + goto basic_json_parser_2; + } + if (yych <= '!') + { + goto basic_json_parser_4; + } goto basic_json_parser_9; - } else { - if (yych <= '+') goto basic_json_parser_4; - if (yych <= ',') goto basic_json_parser_10; + } + else + { + if (yych <= '+') + { + goto basic_json_parser_4; + } + if (yych <= ',') + { + goto basic_json_parser_10; + } goto basic_json_parser_12; } - } else { - if (yych <= '9') { - if (yych <= '/') goto basic_json_parser_4; - if (yych <= '0') goto basic_json_parser_13; + } + else + { + if (yych <= '9') + { + if (yych <= '/') + { + goto basic_json_parser_4; + } + if (yych <= '0') + { + goto basic_json_parser_13; + } goto basic_json_parser_15; - } else { - if (yych <= ':') goto basic_json_parser_17; - if (yych <= 'Z') goto basic_json_parser_4; + } + else + { + if (yych <= ':') + { + goto basic_json_parser_17; + } + if (yych <= 'Z') + { + goto basic_json_parser_4; + } goto basic_json_parser_19; } } - } else { - if (yych <= 'n') { - if (yych <= 'e') { - if (yych == ']') goto basic_json_parser_21; + } + else + { + if (yych <= 'n') + { + if (yych <= 'e') + { + if (yych == ']') + { + goto basic_json_parser_21; + } goto basic_json_parser_4; - } else { - if (yych <= 'f') goto basic_json_parser_23; - if (yych <= 'm') goto basic_json_parser_4; + } + else + { + if (yych <= 'f') + { + goto basic_json_parser_23; + } + if (yych <= 'm') + { + goto basic_json_parser_4; + } goto basic_json_parser_24; } - } else { - if (yych <= 'z') { - if (yych == 't') goto basic_json_parser_25; + } + else + { + if (yych <= 'z') + { + if (yych == 't') + { + goto basic_json_parser_25; + } goto basic_json_parser_4; - } else { - if (yych <= '{') goto basic_json_parser_26; - if (yych == '}') goto basic_json_parser_28; + } + else + { + if (yych <= '{') + { + goto basic_json_parser_26; + } + if (yych == '}') + { + goto basic_json_parser_28; + } goto basic_json_parser_4; } } } - basic_json_parser_2: +basic_json_parser_2: ++m_cursor; - { last_token_type = token_type::end_of_input; break; } - basic_json_parser_4: + { + last_token_type = token_type::end_of_input; + break; + } +basic_json_parser_4: ++m_cursor; - basic_json_parser_5: - { last_token_type = token_type::parse_error; break; } - basic_json_parser_6: +basic_json_parser_5: + { + last_token_type = token_type::parse_error; + break; + } +basic_json_parser_6: ++m_cursor; - if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE + if (m_limit <= m_cursor) + { + fill_line_buffer(1); // LCOV_EXCL_LINE + } yych = *m_cursor; - if (yybm[0+yych] & 32) { + if (yybm[0 + yych] & 32) + { goto basic_json_parser_6; } - { continue; } - basic_json_parser_9: + { + continue; + } +basic_json_parser_9: yyaccept = 0; yych = *(m_marker = ++m_cursor); - if (yych <= 0x1F) goto basic_json_parser_5; - if (yych <= 0x7F) goto basic_json_parser_31; - if (yych <= 0xC1) goto basic_json_parser_5; - if (yych <= 0xF4) goto basic_json_parser_31; + if (yych <= 0x1F) + { + goto basic_json_parser_5; + } + if (yych <= 0x7F) + { + goto basic_json_parser_31; + } + if (yych <= 0xC1) + { + goto basic_json_parser_5; + } + if (yych <= 0xF4) + { + goto basic_json_parser_31; + } goto basic_json_parser_5; - basic_json_parser_10: +basic_json_parser_10: ++m_cursor; - { last_token_type = token_type::value_separator; break; } - basic_json_parser_12: + { + last_token_type = token_type::value_separator; + break; + } +basic_json_parser_12: yych = *++m_cursor; - if (yych <= '/') goto basic_json_parser_5; - if (yych <= '0') goto basic_json_parser_43; - if (yych <= '9') goto basic_json_parser_45; + if (yych <= '/') + { + goto basic_json_parser_5; + } + if (yych <= '0') + { + goto basic_json_parser_43; + } + if (yych <= '9') + { + goto basic_json_parser_45; + } goto basic_json_parser_5; - basic_json_parser_13: +basic_json_parser_13: yyaccept = 1; yych = *(m_marker = ++m_cursor); - if (yych <= '9') { - if (yych == '.') goto basic_json_parser_47; - if (yych >= '0') goto basic_json_parser_48; - } else { - if (yych <= 'E') { - if (yych >= 'E') goto basic_json_parser_51; - } else { - if (yych == 'e') goto basic_json_parser_51; + if (yych <= '9') + { + if (yych == '.') + { + goto basic_json_parser_47; + } + if (yych >= '0') + { + goto basic_json_parser_48; } } - basic_json_parser_14: - { last_token_type = token_type::value_unsigned; break; } - basic_json_parser_15: + else + { + if (yych <= 'E') + { + if (yych >= 'E') + { + goto basic_json_parser_51; + } + } + else + { + if (yych == 'e') + { + goto basic_json_parser_51; + } + } + } +basic_json_parser_14: + { + last_token_type = token_type::value_unsigned; + break; + } +basic_json_parser_15: yyaccept = 1; m_marker = ++m_cursor; - if ((m_limit - m_cursor) < 3) fill_line_buffer(3); // LCOV_EXCL_LINE + if ((m_limit - m_cursor) < 3) + { + fill_line_buffer(3); // LCOV_EXCL_LINE + } yych = *m_cursor; - if (yybm[0+yych] & 64) { + if (yybm[0 + yych] & 64) + { goto basic_json_parser_15; } - if (yych <= 'D') { - if (yych == '.') goto basic_json_parser_47; - goto basic_json_parser_14; - } else { - if (yych <= 'E') goto basic_json_parser_51; - if (yych == 'e') goto basic_json_parser_51; + if (yych <= 'D') + { + if (yych == '.') + { + goto basic_json_parser_47; + } goto basic_json_parser_14; } - basic_json_parser_17: + else + { + if (yych <= 'E') + { + goto basic_json_parser_51; + } + if (yych == 'e') + { + goto basic_json_parser_51; + } + goto basic_json_parser_14; + } +basic_json_parser_17: ++m_cursor; - { last_token_type = token_type::name_separator; break; } - basic_json_parser_19: + { + last_token_type = token_type::name_separator; + break; + } +basic_json_parser_19: ++m_cursor; - { last_token_type = token_type::begin_array; break; } - basic_json_parser_21: + { + last_token_type = token_type::begin_array; + break; + } +basic_json_parser_21: ++m_cursor; - { last_token_type = token_type::end_array; break; } - basic_json_parser_23: + { + last_token_type = token_type::end_array; + break; + } +basic_json_parser_23: yyaccept = 0; yych = *(m_marker = ++m_cursor); - if (yych == 'a') goto basic_json_parser_52; + if (yych == 'a') + { + goto basic_json_parser_52; + } goto basic_json_parser_5; - basic_json_parser_24: +basic_json_parser_24: yyaccept = 0; yych = *(m_marker = ++m_cursor); - if (yych == 'u') goto basic_json_parser_53; + if (yych == 'u') + { + goto basic_json_parser_53; + } goto basic_json_parser_5; - basic_json_parser_25: +basic_json_parser_25: yyaccept = 0; yych = *(m_marker = ++m_cursor); - if (yych == 'r') goto basic_json_parser_54; + if (yych == 'r') + { + goto basic_json_parser_54; + } goto basic_json_parser_5; - basic_json_parser_26: +basic_json_parser_26: ++m_cursor; - { last_token_type = token_type::begin_object; break; } - basic_json_parser_28: + { + last_token_type = token_type::begin_object; + break; + } +basic_json_parser_28: ++m_cursor; - { last_token_type = token_type::end_object; break; } - basic_json_parser_30: + { + last_token_type = token_type::end_object; + break; + } +basic_json_parser_30: ++m_cursor; - if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE + if (m_limit <= m_cursor) + { + fill_line_buffer(1); // LCOV_EXCL_LINE + } yych = *m_cursor; - basic_json_parser_31: - if (yybm[0+yych] & 128) { +basic_json_parser_31: + if (yybm[0 + yych] & 128) + { goto basic_json_parser_30; } - if (yych <= 0xE0) { - if (yych <= '\\') { - if (yych <= 0x1F) goto basic_json_parser_32; - if (yych <= '"') goto basic_json_parser_33; + if (yych <= 0xE0) + { + if (yych <= '\\') + { + if (yych <= 0x1F) + { + goto basic_json_parser_32; + } + if (yych <= '"') + { + goto basic_json_parser_33; + } goto basic_json_parser_35; - } else { - if (yych <= 0xC1) goto basic_json_parser_32; - if (yych <= 0xDF) goto basic_json_parser_36; + } + else + { + if (yych <= 0xC1) + { + goto basic_json_parser_32; + } + if (yych <= 0xDF) + { + goto basic_json_parser_36; + } goto basic_json_parser_37; } - } else { - if (yych <= 0xEF) { - if (yych == 0xED) goto basic_json_parser_39; + } + else + { + if (yych <= 0xEF) + { + if (yych == 0xED) + { + goto basic_json_parser_39; + } goto basic_json_parser_38; - } else { - if (yych <= 0xF0) goto basic_json_parser_40; - if (yych <= 0xF3) goto basic_json_parser_41; - if (yych <= 0xF4) goto basic_json_parser_42; + } + else + { + if (yych <= 0xF0) + { + goto basic_json_parser_40; + } + if (yych <= 0xF3) + { + goto basic_json_parser_41; + } + if (yych <= 0xF4) + { + goto basic_json_parser_42; + } } } - basic_json_parser_32: +basic_json_parser_32: m_cursor = m_marker; - if (yyaccept <= 1) { - if (yyaccept == 0) { + if (yyaccept <= 1) + { + if (yyaccept == 0) + { goto basic_json_parser_5; - } else { + } + else + { goto basic_json_parser_14; } - } else { - if (yyaccept == 2) { + } + else + { + if (yyaccept == 2) + { goto basic_json_parser_44; - } else { + } + else + { goto basic_json_parser_58; } } - basic_json_parser_33: +basic_json_parser_33: ++m_cursor; - { last_token_type = token_type::value_string; break; } - basic_json_parser_35: + { + last_token_type = token_type::value_string; + break; + } +basic_json_parser_35: ++m_cursor; - if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE + if (m_limit <= m_cursor) + { + fill_line_buffer(1); // LCOV_EXCL_LINE + } yych = *m_cursor; - if (yych <= 'e') { - if (yych <= '/') { - if (yych == '"') goto basic_json_parser_30; - if (yych <= '.') goto basic_json_parser_32; - goto basic_json_parser_30; - } else { - if (yych <= '\\') { - if (yych <= '[') goto basic_json_parser_32; + if (yych <= 'e') + { + if (yych <= '/') + { + if (yych == '"') + { goto basic_json_parser_30; - } else { - if (yych == 'b') goto basic_json_parser_30; + } + if (yych <= '.') + { goto basic_json_parser_32; } + goto basic_json_parser_30; } - } else { - if (yych <= 'q') { - if (yych <= 'f') goto basic_json_parser_30; - if (yych == 'n') goto basic_json_parser_30; - goto basic_json_parser_32; - } else { - if (yych <= 's') { - if (yych <= 'r') goto basic_json_parser_30; - goto basic_json_parser_32; - } else { - if (yych <= 't') goto basic_json_parser_30; - if (yych <= 'u') goto basic_json_parser_55; + else + { + if (yych <= '\\') + { + if (yych <= '[') + { + goto basic_json_parser_32; + } + goto basic_json_parser_30; + } + else + { + if (yych == 'b') + { + goto basic_json_parser_30; + } goto basic_json_parser_32; } } } - basic_json_parser_36: + else + { + if (yych <= 'q') + { + if (yych <= 'f') + { + goto basic_json_parser_30; + } + if (yych == 'n') + { + goto basic_json_parser_30; + } + goto basic_json_parser_32; + } + else + { + if (yych <= 's') + { + if (yych <= 'r') + { + goto basic_json_parser_30; + } + goto basic_json_parser_32; + } + else + { + if (yych <= 't') + { + goto basic_json_parser_30; + } + if (yych <= 'u') + { + goto basic_json_parser_55; + } + goto basic_json_parser_32; + } + } + } +basic_json_parser_36: ++m_cursor; - if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE + if (m_limit <= m_cursor) + { + fill_line_buffer(1); // LCOV_EXCL_LINE + } yych = *m_cursor; - if (yych <= 0x7F) goto basic_json_parser_32; - if (yych <= 0xBF) goto basic_json_parser_30; + if (yych <= 0x7F) + { + goto basic_json_parser_32; + } + if (yych <= 0xBF) + { + goto basic_json_parser_30; + } goto basic_json_parser_32; - basic_json_parser_37: +basic_json_parser_37: ++m_cursor; - if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE + if (m_limit <= m_cursor) + { + fill_line_buffer(1); // LCOV_EXCL_LINE + } yych = *m_cursor; - if (yych <= 0x9F) goto basic_json_parser_32; - if (yych <= 0xBF) goto basic_json_parser_36; + if (yych <= 0x9F) + { + goto basic_json_parser_32; + } + if (yych <= 0xBF) + { + goto basic_json_parser_36; + } goto basic_json_parser_32; - basic_json_parser_38: +basic_json_parser_38: ++m_cursor; - if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE + if (m_limit <= m_cursor) + { + fill_line_buffer(1); // LCOV_EXCL_LINE + } yych = *m_cursor; - if (yych <= 0x7F) goto basic_json_parser_32; - if (yych <= 0xBF) goto basic_json_parser_36; + if (yych <= 0x7F) + { + goto basic_json_parser_32; + } + if (yych <= 0xBF) + { + goto basic_json_parser_36; + } goto basic_json_parser_32; - basic_json_parser_39: +basic_json_parser_39: ++m_cursor; - if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE + if (m_limit <= m_cursor) + { + fill_line_buffer(1); // LCOV_EXCL_LINE + } yych = *m_cursor; - if (yych <= 0x7F) goto basic_json_parser_32; - if (yych <= 0x9F) goto basic_json_parser_36; + if (yych <= 0x7F) + { + goto basic_json_parser_32; + } + if (yych <= 0x9F) + { + goto basic_json_parser_36; + } goto basic_json_parser_32; - basic_json_parser_40: +basic_json_parser_40: ++m_cursor; - if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE + if (m_limit <= m_cursor) + { + fill_line_buffer(1); // LCOV_EXCL_LINE + } yych = *m_cursor; - if (yych <= 0x8F) goto basic_json_parser_32; - if (yych <= 0xBF) goto basic_json_parser_38; + if (yych <= 0x8F) + { + goto basic_json_parser_32; + } + if (yych <= 0xBF) + { + goto basic_json_parser_38; + } goto basic_json_parser_32; - basic_json_parser_41: +basic_json_parser_41: ++m_cursor; - if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE + if (m_limit <= m_cursor) + { + fill_line_buffer(1); // LCOV_EXCL_LINE + } yych = *m_cursor; - if (yych <= 0x7F) goto basic_json_parser_32; - if (yych <= 0xBF) goto basic_json_parser_38; + if (yych <= 0x7F) + { + goto basic_json_parser_32; + } + if (yych <= 0xBF) + { + goto basic_json_parser_38; + } goto basic_json_parser_32; - basic_json_parser_42: +basic_json_parser_42: ++m_cursor; - if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE + if (m_limit <= m_cursor) + { + fill_line_buffer(1); // LCOV_EXCL_LINE + } yych = *m_cursor; - if (yych <= 0x7F) goto basic_json_parser_32; - if (yych <= 0x8F) goto basic_json_parser_38; + if (yych <= 0x7F) + { + goto basic_json_parser_32; + } + if (yych <= 0x8F) + { + goto basic_json_parser_38; + } goto basic_json_parser_32; - basic_json_parser_43: +basic_json_parser_43: yyaccept = 2; yych = *(m_marker = ++m_cursor); - if (yych <= '9') { - if (yych == '.') goto basic_json_parser_47; - if (yych >= '0') goto basic_json_parser_48; - } else { - if (yych <= 'E') { - if (yych >= 'E') goto basic_json_parser_51; - } else { - if (yych == 'e') goto basic_json_parser_51; + if (yych <= '9') + { + if (yych == '.') + { + goto basic_json_parser_47; + } + if (yych >= '0') + { + goto basic_json_parser_48; } } - basic_json_parser_44: - { last_token_type = token_type::value_integer; break; } - basic_json_parser_45: + else + { + if (yych <= 'E') + { + if (yych >= 'E') + { + goto basic_json_parser_51; + } + } + else + { + if (yych == 'e') + { + goto basic_json_parser_51; + } + } + } +basic_json_parser_44: + { + last_token_type = token_type::value_integer; + break; + } +basic_json_parser_45: yyaccept = 2; m_marker = ++m_cursor; - if ((m_limit - m_cursor) < 3) fill_line_buffer(3); // LCOV_EXCL_LINE + if ((m_limit - m_cursor) < 3) + { + fill_line_buffer(3); // LCOV_EXCL_LINE + } yych = *m_cursor; - if (yych <= '9') { - if (yych == '.') goto basic_json_parser_47; - if (yych <= '/') goto basic_json_parser_44; + if (yych <= '9') + { + if (yych == '.') + { + goto basic_json_parser_47; + } + if (yych <= '/') + { + goto basic_json_parser_44; + } goto basic_json_parser_45; - } else { - if (yych <= 'E') { - if (yych <= 'D') goto basic_json_parser_44; + } + else + { + if (yych <= 'E') + { + if (yych <= 'D') + { + goto basic_json_parser_44; + } goto basic_json_parser_51; - } else { - if (yych == 'e') goto basic_json_parser_51; + } + else + { + if (yych == 'e') + { + goto basic_json_parser_51; + } goto basic_json_parser_44; } } - basic_json_parser_47: +basic_json_parser_47: yych = *++m_cursor; - if (yych <= '/') goto basic_json_parser_32; - if (yych <= '9') goto basic_json_parser_56; - goto basic_json_parser_32; - basic_json_parser_48: - ++m_cursor; - if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE - yych = *m_cursor; - if (yych <= '/') goto basic_json_parser_50; - if (yych <= '9') goto basic_json_parser_48; - basic_json_parser_50: - { last_token_type = token_type::parse_error; break; } - basic_json_parser_51: - yych = *++m_cursor; - if (yych <= ',') { - if (yych == '+') goto basic_json_parser_59; - goto basic_json_parser_32; - } else { - if (yych <= '-') goto basic_json_parser_59; - if (yych <= '/') goto basic_json_parser_32; - if (yych <= '9') goto basic_json_parser_60; + if (yych <= '/') + { goto basic_json_parser_32; } - basic_json_parser_52: - yych = *++m_cursor; - if (yych == 'l') goto basic_json_parser_62; + if (yych <= '9') + { + goto basic_json_parser_56; + } goto basic_json_parser_32; - basic_json_parser_53: - yych = *++m_cursor; - if (yych == 'l') goto basic_json_parser_63; - goto basic_json_parser_32; - basic_json_parser_54: - yych = *++m_cursor; - if (yych == 'u') goto basic_json_parser_64; - goto basic_json_parser_32; - basic_json_parser_55: +basic_json_parser_48: ++m_cursor; - if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE + if (m_limit <= m_cursor) + { + fill_line_buffer(1); // LCOV_EXCL_LINE + } yych = *m_cursor; - if (yych <= '@') { - if (yych <= '/') goto basic_json_parser_32; - if (yych <= '9') goto basic_json_parser_65; - goto basic_json_parser_32; - } else { - if (yych <= 'F') goto basic_json_parser_65; - if (yych <= '`') goto basic_json_parser_32; - if (yych <= 'f') goto basic_json_parser_65; + if (yych <= '/') + { + goto basic_json_parser_50; + } + if (yych <= '9') + { + goto basic_json_parser_48; + } +basic_json_parser_50: + { + last_token_type = token_type::parse_error; + break; + } +basic_json_parser_51: + yych = *++m_cursor; + if (yych <= ',') + { + if (yych == '+') + { + goto basic_json_parser_59; + } goto basic_json_parser_32; } - basic_json_parser_56: + else + { + if (yych <= '-') + { + goto basic_json_parser_59; + } + if (yych <= '/') + { + goto basic_json_parser_32; + } + if (yych <= '9') + { + goto basic_json_parser_60; + } + goto basic_json_parser_32; + } +basic_json_parser_52: + yych = *++m_cursor; + if (yych == 'l') + { + goto basic_json_parser_62; + } + goto basic_json_parser_32; +basic_json_parser_53: + yych = *++m_cursor; + if (yych == 'l') + { + goto basic_json_parser_63; + } + goto basic_json_parser_32; +basic_json_parser_54: + yych = *++m_cursor; + if (yych == 'u') + { + goto basic_json_parser_64; + } + goto basic_json_parser_32; +basic_json_parser_55: + ++m_cursor; + if (m_limit <= m_cursor) + { + fill_line_buffer(1); // LCOV_EXCL_LINE + } + yych = *m_cursor; + if (yych <= '@') + { + if (yych <= '/') + { + goto basic_json_parser_32; + } + if (yych <= '9') + { + goto basic_json_parser_65; + } + goto basic_json_parser_32; + } + else + { + if (yych <= 'F') + { + goto basic_json_parser_65; + } + if (yych <= '`') + { + goto basic_json_parser_32; + } + if (yych <= 'f') + { + goto basic_json_parser_65; + } + goto basic_json_parser_32; + } +basic_json_parser_56: yyaccept = 3; m_marker = ++m_cursor; - if ((m_limit - m_cursor) < 3) fill_line_buffer(3); // LCOV_EXCL_LINE - yych = *m_cursor; - if (yych <= 'D') { - if (yych <= '/') goto basic_json_parser_58; - if (yych <= '9') goto basic_json_parser_56; - } else { - if (yych <= 'E') goto basic_json_parser_51; - if (yych == 'e') goto basic_json_parser_51; + if ((m_limit - m_cursor) < 3) + { + fill_line_buffer(3); // LCOV_EXCL_LINE } - basic_json_parser_58: - { last_token_type = token_type::value_float; break; } - basic_json_parser_59: - yych = *++m_cursor; - if (yych <= '/') goto basic_json_parser_32; - if (yych >= ':') goto basic_json_parser_32; - basic_json_parser_60: - ++m_cursor; - if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE yych = *m_cursor; - if (yych <= '/') goto basic_json_parser_58; - if (yych <= '9') goto basic_json_parser_60; + if (yych <= 'D') + { + if (yych <= '/') + { + goto basic_json_parser_58; + } + if (yych <= '9') + { + goto basic_json_parser_56; + } + } + else + { + if (yych <= 'E') + { + goto basic_json_parser_51; + } + if (yych == 'e') + { + goto basic_json_parser_51; + } + } +basic_json_parser_58: + { + last_token_type = token_type::value_float; + break; + } +basic_json_parser_59: + yych = *++m_cursor; + if (yych <= '/') + { + goto basic_json_parser_32; + } + if (yych >= ':') + { + goto basic_json_parser_32; + } +basic_json_parser_60: + ++m_cursor; + if (m_limit <= m_cursor) + { + fill_line_buffer(1); // LCOV_EXCL_LINE + } + yych = *m_cursor; + if (yych <= '/') + { + goto basic_json_parser_58; + } + if (yych <= '9') + { + goto basic_json_parser_60; + } goto basic_json_parser_58; - basic_json_parser_62: +basic_json_parser_62: yych = *++m_cursor; - if (yych == 's') goto basic_json_parser_66; + if (yych == 's') + { + goto basic_json_parser_66; + } goto basic_json_parser_32; - basic_json_parser_63: +basic_json_parser_63: yych = *++m_cursor; - if (yych == 'l') goto basic_json_parser_67; + if (yych == 'l') + { + goto basic_json_parser_67; + } goto basic_json_parser_32; - basic_json_parser_64: +basic_json_parser_64: yych = *++m_cursor; - if (yych == 'e') goto basic_json_parser_69; + if (yych == 'e') + { + goto basic_json_parser_69; + } goto basic_json_parser_32; - basic_json_parser_65: +basic_json_parser_65: ++m_cursor; - if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE + if (m_limit <= m_cursor) + { + fill_line_buffer(1); // LCOV_EXCL_LINE + } yych = *m_cursor; - if (yych <= '@') { - if (yych <= '/') goto basic_json_parser_32; - if (yych <= '9') goto basic_json_parser_71; - goto basic_json_parser_32; - } else { - if (yych <= 'F') goto basic_json_parser_71; - if (yych <= '`') goto basic_json_parser_32; - if (yych <= 'f') goto basic_json_parser_71; + if (yych <= '@') + { + if (yych <= '/') + { + goto basic_json_parser_32; + } + if (yych <= '9') + { + goto basic_json_parser_71; + } goto basic_json_parser_32; } - basic_json_parser_66: - yych = *++m_cursor; - if (yych == 'e') goto basic_json_parser_72; - goto basic_json_parser_32; - basic_json_parser_67: - ++m_cursor; - { last_token_type = token_type::literal_null; break; } - basic_json_parser_69: - ++m_cursor; - { last_token_type = token_type::literal_true; break; } - basic_json_parser_71: - ++m_cursor; - if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE - yych = *m_cursor; - if (yych <= '@') { - if (yych <= '/') goto basic_json_parser_32; - if (yych <= '9') goto basic_json_parser_74; - goto basic_json_parser_32; - } else { - if (yych <= 'F') goto basic_json_parser_74; - if (yych <= '`') goto basic_json_parser_32; - if (yych <= 'f') goto basic_json_parser_74; + else + { + if (yych <= 'F') + { + goto basic_json_parser_71; + } + if (yych <= '`') + { + goto basic_json_parser_32; + } + if (yych <= 'f') + { + goto basic_json_parser_71; + } goto basic_json_parser_32; } - basic_json_parser_72: +basic_json_parser_66: + yych = *++m_cursor; + if (yych == 'e') + { + goto basic_json_parser_72; + } + goto basic_json_parser_32; +basic_json_parser_67: ++m_cursor; - { last_token_type = token_type::literal_false; break; } - basic_json_parser_74: + { + last_token_type = token_type::literal_null; + break; + } +basic_json_parser_69: ++m_cursor; - if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE + { + last_token_type = token_type::literal_true; + break; + } +basic_json_parser_71: + ++m_cursor; + if (m_limit <= m_cursor) + { + fill_line_buffer(1); // LCOV_EXCL_LINE + } yych = *m_cursor; - if (yych <= '@') { - if (yych <= '/') goto basic_json_parser_32; - if (yych <= '9') goto basic_json_parser_30; + if (yych <= '@') + { + if (yych <= '/') + { + goto basic_json_parser_32; + } + if (yych <= '9') + { + goto basic_json_parser_74; + } goto basic_json_parser_32; - } else { - if (yych <= 'F') goto basic_json_parser_30; - if (yych <= '`') goto basic_json_parser_32; - if (yych <= 'f') goto basic_json_parser_30; + } + else + { + if (yych <= 'F') + { + goto basic_json_parser_74; + } + if (yych <= '`') + { + goto basic_json_parser_32; + } + if (yych <= 'f') + { + goto basic_json_parser_74; + } + goto basic_json_parser_32; + } +basic_json_parser_72: + ++m_cursor; + { + last_token_type = token_type::literal_false; + break; + } +basic_json_parser_74: + ++m_cursor; + if (m_limit <= m_cursor) + { + fill_line_buffer(1); // LCOV_EXCL_LINE + } + yych = *m_cursor; + if (yych <= '@') + { + if (yych <= '/') + { + goto basic_json_parser_32; + } + if (yych <= '9') + { + goto basic_json_parser_30; + } + goto basic_json_parser_32; + } + else + { + if (yych <= 'F') + { + goto basic_json_parser_30; + } + if (yych <= '`') + { + goto basic_json_parser_32; + } + if (yych <= 'f') + { + goto basic_json_parser_30; + } goto basic_json_parser_32; } } diff --git a/test/src/unit-udt.cpp b/test/src/unit-udt.cpp index 6b37a106..6aa469fe 100644 --- a/test/src/unit-udt.cpp +++ b/test/src/unit-udt.cpp @@ -49,19 +49,19 @@ enum class country struct age { int m_val; - age(int rhs=0) : m_val(rhs) {} + age(int rhs = 0) : m_val(rhs) {} }; struct name { std::string m_val; - name(const std::string rhs="") : m_val(rhs) {} + name(const std::string rhs = "") : m_val(rhs) {} }; struct address { std::string m_val; - address(const std::string rhs="") : m_val(rhs) {} + address(const std::string rhs = "") : m_val(rhs) {} }; struct person @@ -69,7 +69,7 @@ struct person age m_age; name m_name; country m_country; - person() : m_age(),m_name(),m_country() {} + person() : m_age(), m_name(), m_country() {} person(const age& a, const name& n, const country& c) : m_age(a), m_name(n), m_country(c) {} }; From 65dfc97d401e17befa4d9a2320660b381529304e Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sat, 11 Mar 2017 17:44:54 +0100 Subject: [PATCH 065/530] :hammer: using __cpp_exceptions to detect exception support #498 MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit I used __EXCEPTIONS to detect whether exceptions are supported. Apparently, this is a macro that is only used by libstdc++ (https://gcc.gnu.org/bugzilla/show_bug.cgi?id=64276). It’s much cleaner to use __cpp_exceptions as it is in the standard since C++98. Note that compiling the unit-tests with “-fno-exceptions” still does not work, because Catch uses throw internally. However, the library’s exceptions can be switched off by defining JSON_NOEXCEPTION. --- src/json.hpp | 2 +- src/json.hpp.re2c | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 33456a51..9fb5fefe 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -82,7 +82,7 @@ SOFTWARE. #endif // allow to disable exceptions -#if not defined(JSON_NOEXCEPTION) || defined(__EXCEPTIONS) +#if __cpp_exceptions && not defined(JSON_NOEXCEPTION) #define JSON_THROW(exception) throw exception #define JSON_TRY try #define JSON_CATCH(exception) catch(exception) diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 180f5d3b..a837df06 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -82,7 +82,7 @@ SOFTWARE. #endif // allow to disable exceptions -#if not defined(JSON_NOEXCEPTION) || defined(__EXCEPTIONS) +#if __cpp_exceptions && not defined(JSON_NOEXCEPTION) #define JSON_THROW(exception) throw exception #define JSON_TRY try #define JSON_CATCH(exception) catch(exception) From e3e6cbecc7a150590636d13b17a0260b5c46d90e Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sat, 11 Mar 2017 17:59:24 +0100 Subject: [PATCH 066/530] :checkered_flag: added check for _CPPUNWIND MSVC does not define __cpp_exceptions, but seems to use _CPPUNWIND when exception support is switched on, see https://msdn.microsoft.com/en-us/library/b0084kay.aspx. --- src/json.hpp | 2 +- src/json.hpp.re2c | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 9fb5fefe..f9a92820 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -82,7 +82,7 @@ SOFTWARE. #endif // allow to disable exceptions -#if __cpp_exceptions && not defined(JSON_NOEXCEPTION) +#if (__cpp_exceptions || defined(_CPPUNWIND)) && not defined(JSON_NOEXCEPTION) #define JSON_THROW(exception) throw exception #define JSON_TRY try #define JSON_CATCH(exception) catch(exception) diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index a837df06..2cad18aa 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -82,7 +82,7 @@ SOFTWARE. #endif // allow to disable exceptions -#if __cpp_exceptions && not defined(JSON_NOEXCEPTION) +#if (__cpp_exceptions || defined(_CPPUNWIND)) && not defined(JSON_NOEXCEPTION) #define JSON_THROW(exception) throw exception #define JSON_TRY try #define JSON_CATCH(exception) catch(exception) From 122afbf1280413ade3719d6af2c37352215e30f0 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sat, 11 Mar 2017 18:43:21 +0100 Subject: [PATCH 067/530] :hammer: added defined() check --- src/json.hpp | 2 +- src/json.hpp.re2c | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index f9a92820..725310f6 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -82,7 +82,7 @@ SOFTWARE. #endif // allow to disable exceptions -#if (__cpp_exceptions || defined(_CPPUNWIND)) && not defined(JSON_NOEXCEPTION) +#if (defined(__cpp_exceptions) || defined(_CPPUNWIND)) && not defined(JSON_NOEXCEPTION) #define JSON_THROW(exception) throw exception #define JSON_TRY try #define JSON_CATCH(exception) catch(exception) diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 2cad18aa..9172d2bf 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -82,7 +82,7 @@ SOFTWARE. #endif // allow to disable exceptions -#if (__cpp_exceptions || defined(_CPPUNWIND)) && not defined(JSON_NOEXCEPTION) +#if (defined(__cpp_exceptions) || defined(_CPPUNWIND)) && not defined(JSON_NOEXCEPTION) #define JSON_THROW(exception) throw exception #define JSON_TRY try #define JSON_CATCH(exception) catch(exception) From 4e49829851e536bd82220872e5ea733221eaf10d Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sat, 11 Mar 2017 19:26:12 +0100 Subject: [PATCH 068/530] :ambulance: fix for #493 Added a test to check if the input stream is good() before executing getline on it. Also added two test cases that set the failbit and badbit before calling file_line_buffer. --- src/json.hpp | 7 +++++++ src/json.hpp.re2c | 7 +++++++ test/src/unit-regression.cpp | 32 ++++++++++++++++++++++++++++++++ 3 files changed, 46 insertions(+) diff --git a/src/json.hpp b/src/json.hpp index 33456a51..1ed8cca2 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -10988,6 +10988,13 @@ basic_json_parser_74: m_line_buffer.erase(0, num_processed_chars); // read next line from input stream m_line_buffer_tmp.clear(); + + // check if stream is still good + if (not m_stream->good()) + { + JSON_THROW(std::invalid_argument("stream error")); + } + std::getline(*m_stream, m_line_buffer_tmp, '\n'); // add line with newline symbol to the line buffer diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 180f5d3b..c58b29a8 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -10022,6 +10022,13 @@ class basic_json m_line_buffer.erase(0, num_processed_chars); // read next line from input stream m_line_buffer_tmp.clear(); + + // check if stream is still good + if (not m_stream->good()) + { + JSON_THROW(std::invalid_argument("stream error")); + } + std::getline(*m_stream, m_line_buffer_tmp, '\n'); // add line with newline symbol to the line buffer diff --git a/test/src/unit-regression.cpp b/test/src/unit-regression.cpp index bfd2c954..368c1470 100644 --- a/test/src/unit-regression.cpp +++ b/test/src/unit-regression.cpp @@ -28,6 +28,7 @@ SOFTWARE. #include "catch.hpp" +#define private public #include "json.hpp" using nlohmann::json; @@ -804,4 +805,35 @@ TEST_CASE("regression tests") CHECK(j["bool_vector"].dump() == "[false,true,false,false]"); } + + SECTION("issue #495 - fill_line_buffer incorrectly tests m_stream for eof but not fail or bad bits") + { + SECTION("setting failbit") + { + std::stringstream ss; + ss << "[1,2,3\n,4,5]"; + + json::lexer l(ss); + + CHECK(l.m_line_buffer == "[1,2,3\n"); + + l.m_stream->setstate(std::ios_base::failbit); + + CHECK_THROWS_AS(l.fill_line_buffer(), std::invalid_argument); + } + + SECTION("setting badbit") + { + std::stringstream ss; + ss << "[1,2,3\n,4,5]"; + + json::lexer l(ss); + + CHECK(l.m_line_buffer == "[1,2,3\n"); + + l.m_stream->setstate(std::ios_base::badbit); + + CHECK_THROWS_AS(l.fill_line_buffer(), std::invalid_argument); + } + } } From 4b9c2f128740842abb8ba60956778bb59af15bd4 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sat, 11 Mar 2017 20:16:13 +0100 Subject: [PATCH 069/530] :hammer: added __EXCEPTIONS to the list --- src/json.hpp | 2 +- src/json.hpp.re2c | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 725310f6..c7b0154a 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -82,7 +82,7 @@ SOFTWARE. #endif // allow to disable exceptions -#if (defined(__cpp_exceptions) || defined(_CPPUNWIND)) && not defined(JSON_NOEXCEPTION) +#if (defined(__cpp_exceptions) || defined(__EXCEPTIONS) || defined(_CPPUNWIND)) && not defined(JSON_NOEXCEPTION) #define JSON_THROW(exception) throw exception #define JSON_TRY try #define JSON_CATCH(exception) catch(exception) diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 9172d2bf..3ee67b1a 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -82,7 +82,7 @@ SOFTWARE. #endif // allow to disable exceptions -#if (defined(__cpp_exceptions) || defined(_CPPUNWIND)) && not defined(JSON_NOEXCEPTION) +#if (defined(__cpp_exceptions) || defined(__EXCEPTIONS) || defined(_CPPUNWIND)) && not defined(JSON_NOEXCEPTION) #define JSON_THROW(exception) throw exception #define JSON_TRY try #define JSON_CATCH(exception) catch(exception) From 01470f388b496050be164b9fe6bcc613d1203109 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 12 Mar 2017 11:04:26 +0100 Subject: [PATCH 070/530] :construction_worker: fixed no_exceptions test case This test case relied on logics that have been replaced by CMake with #461. This change enables compilation and execution of the test suite without exceptions by adding an after_success task. --- .travis.yml | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/.travis.yml b/.travis.yml index c3cba69e..5060b728 100644 --- a/.travis.yml +++ b/.travis.yml @@ -78,13 +78,13 @@ matrix: env: - COMPILER=g++-4.9 - SPECIAL=no_exceptions - - TEST_PATTERN=-e \"*\" addons: apt: sources: ['ubuntu-toolchain-r-test'] packages: [g++-4.9, cppcheck] - before_script: - - CPPFLAGS="-DJSON_NOEXCEPTION" make + after_success: + - make clean + - CPPFLAGS="-DJSON_NOEXCEPTION" make check TEST_PATTERN=-e \"*\"" # Coveralls (http://gronlier.fr/blog/2015/01/adding-code-coverage-to-your-c-project/) From e3e941ef2b9a5405cff77c64e52b3c5f20567029 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 12 Mar 2017 11:19:27 +0100 Subject: [PATCH 071/530] :construction_worker: fixed a syntax error --- .travis.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.travis.yml b/.travis.yml index 5060b728..dcf7b6a2 100644 --- a/.travis.yml +++ b/.travis.yml @@ -84,7 +84,7 @@ matrix: packages: [g++-4.9, cppcheck] after_success: - make clean - - CPPFLAGS="-DJSON_NOEXCEPTION" make check TEST_PATTERN=-e \"*\"" + - CPPFLAGS="-DJSON_NOEXCEPTION" make check TEST_PATTERN="-e \"*\"" # Coveralls (http://gronlier.fr/blog/2015/01/adding-code-coverage-to-your-c-project/) From 28dbe4e6515f4160184bf9d19496fa5682d0b2fc Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 12 Mar 2017 13:49:39 +0100 Subject: [PATCH 072/530] :memo: overworked documentation for the at functions Added all possible exceptions to the examples of the at functions. --- doc/examples/at__object_t_key_type | Bin 130960 -> 0 bytes doc/examples/at__object_t_key_type.cpp | 16 ++- doc/examples/at__object_t_key_type.link | 2 +- doc/examples/at__object_t_key_type.output | 1 + doc/examples/at__object_t_key_type_const.cpp | 16 ++- doc/examples/at__object_t_key_type_const.link | 2 +- .../at__object_t_key_type_const.output | 1 + doc/examples/at__size_type.cpp | 16 ++- doc/examples/at__size_type.link | 2 +- doc/examples/at__size_type.output | 1 + doc/examples/at__size_type_const.cpp | 20 +++- doc/examples/at__size_type_const.link | 2 +- doc/examples/at__size_type_const.output | 1 + doc/examples/at_json_pointer.cpp | 50 ++++++++- doc/examples/at_json_pointer.link | 2 +- doc/examples/at_json_pointer.output | 4 + doc/examples/at_json_pointer_const.cpp | 40 ++++++- doc/examples/at_json_pointer_const.link | 2 +- doc/examples/at_json_pointer_const.output | 3 + src/json.hpp | 104 +++++++++++------- src/json.hpp.re2c | 104 +++++++++++------- 21 files changed, 296 insertions(+), 93 deletions(-) delete mode 100755 doc/examples/at__object_t_key_type diff --git a/doc/examples/at__object_t_key_type b/doc/examples/at__object_t_key_type deleted file mode 100755 index 03f48e68ca3efe94e36855f9d69bd72b6854b4ff..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 130960 zcmeFa4Sc0nSueU%Iwb{4QVWqWcwRC1InQScf(?h?4}l4-|DsA6-FgHaFz^6^Fo2p>)Z2kH5LJnzoC zGns;V?>*k%uk)Lo=Y7`4v%a78tY@|p;ll^_J#+AdgU@~zUEBY4w4*y>q%)2L4j+E*;P7+XgDC#UHY$N5Z5f-K@U}?f2Mc%$aU9`UiSvLY^T@%v!?g9C@yi@$$dzv_bE6 zuf1WbLAKLdd9mreCx{SlYmYK=kM_S!eS`n9h({OZ@f@^!-7+xNz6O>ZhPLVSCk z$|TwErci{#hp*oE+f+9iIPy>tEpPZ>)kMuZIu6_WHv&y!q!}_PXm2kKXu( z^U^EGkzQ1x$g+PtR$u;e`0%SQHyd$?Z>RUTHhR~;^q%rJAN2a8n8eHU zSl%n!=)K~lH@uZaT9N7P}KQnms{(=XLMmtPLSm69O z{VT?quSh};KU}Fy^zjSfdI5R?#}Q!W^&OQSbcFdEIx2Ih?BQSNP?NH6^x!xBi^w0s zjr?*)rSJ8X%KhPX5I_D~iXYuRj^AbYo$ueuO&yh2?BMx8xb@=q0Q{afcEcNSwi z_QY4b`T8R4!{N?+eBkYaU)cYTm)-Z)pZwwHKIx7B_}a(7O?~D8_e13SRerp{0r_t? z%FB4gwl0vrH%Ao{`ki~Vb2qgiC4c&`RXr+uY3~C)gJyiZ@%Hi z>wf+T&xj}pU8(#SeoT7dEgh9rK=wJ}tj~M@-CG0q*1)|raBmIVTLbsjz`ZqaZw=gA z1OMJ@;7~RBLiO000SLdpW&BeGXVMDWA(7E=xL1()yc(f1c^9*^GwEB%A2iF*&s;P zj*#S1Z4`h~d^K&*HsQ&jUA1M_R8ch2?1k5`X&y99ffmgwX`V1meuJihizb5>G}){L!2t$D zzft&_K#O1>9P=wokl!Gv;3CMNErN4f1a~uFlJzQlO`t`v2ab8(sZ#SB1QlEa8MGi+ ztFQ@@+B`wF>z1q8O=@$*(oBv?w&)-Wa;ycmY{wOYRf9Exr2(y>0g*m(z-2!(l8Jx@ zbT_k9+Kt;3zl7Z{QqF} zG;?BUO^(R923m8|j>c-8a;$Y5^ceIKoUeLJAZe~4C&bTa&+R1iZ z!`vmc-Ne#Nj!8D)3hb9-EwHK`4;l;`j1Zi!jx!PPoPrFX3(hc4?`9Ff!CQGso3r#z z8}vpOq)&eHHiN9jERIU6TlydGNqd()}Av z#Wj8-b;lzQR6Oy(^LX13#ZH~yEW6@~12dje z|6MTp?WsTQTsNfN>(s|vsSlG%QFbC*sUHNEe+DxO;_(})J04S4Jf}{D*g!zqx6|QwCnMDaT2wdSn2(q$zd=>OMU_EIRL@!#?5c4eYW+p57Ib7abvCP}K8d={ z8L4P$B3n(p1}y&udJN+68;Lm{n_BVQ)U5IK*PYw5)*+>G=#l?#>k9eG9*o2%+Mj%N zyZ=1>yLO;Sg>($(?Qwn)anc%llw_PMmb+>nQ>mLA>G=oXjf~H+=xKj7eQP(So>i6$ zf=ycE*WD{%y??Dq>2&^PbNUsyx3OJ1bUo@gLRKnTKTMuMk&mMF#x|-cU^=#u{|=|b$y43e-u_Db+c`YK=Kn~BtZ-eA zy1GTL_v)#|eN!J5#pDnuAIz0%GhaW0}z1QB}^{9dK=kM1Si*(PQzjAq=bKAcx{DV*3WB7rojcPV_mpT+gXw=@+ z;9ab|{KtP3-6<+0@_+2i2C5qCvx_+AdVPjI=9#pic{QyGOZ%$n8}n)hCX||HSP0Xc zXeY;eDqZhBi7K4Ro&|@qHU91BI{rivj>w(WbUz|PVmF(EjLyjEKo49v(XUft+_MN- z89a|TP>y1LBZ%hvA}3qK1}Ctj8%Gd_2AWnuF&ky9G^0;7flJy@8MMgalAsqJ`TIV> z6jaErTk6bpDk)>~a~*?w7Qs3cs)10>Om=h%84{tS)5l=z+P(JnDxzwT^usmDp3Jy6D{hl6`DcH!qmt+Nb54Igc4kq-R(Yu=4UtV7%vm+L z3l3a3#u9l zsk@|2Rova+8S->}qhT)p=9Q?i^dSuaRdhc2cyRbPp+9VE+4a3hZ@XJeQ%q%>SN=bK z9Hg@*+c5tqzR$Xmqr-876m7(&&F_yC^9qw?k=FkXIbTo*S;{{{BJ3dl2?0B*SvCU* zUNa(aC;I7>z9i)4FL`vUXY6_)R(`13S;$C^GdKgrJiF(k$zOnA{-YvXJBZjtra-WwWYyNv%r%_KaECy6u#2IPqT#&t+kqW@Y8JD!B4OMRjXk!;h|pZ&aG$0iyqQy zSk@C8_Wlro(XfBMmq68(|Ni4p!@q@OwzbqfH7rxv7CisXZqUq{tkrx2-)G%8?G4KY z$}eMz!XpuY&t*!4KmEENpBl-v?3F zO{c*#h(;R1U{b@Op$;t!oVaXYYV5tJPqgEqWuW++ADFtVnjLyCTowmT*q2U|i=EKm zbh24MV--yV$fiTOgJ@r%DV&obwBE(lq!TWyknvaGf_5c@)6wbteHb`fxjJ@=Ik&L0 zlnfouDbJi!zR5GaoN*;xguDXs8*oFn!OSzZmV1H#o{k}3x8?7AB-&u|6r>7P*H(QU z%>U@I@JTLwO#Z9*m~NAYIGulr-rMAxV`pG#fJnhHkrB^S9EedCIdt99Id#Y6ddF1# zq3DfR5a4gu@vBfIg*s8AGvuDh^$U`W(?HkpWCzdtK!YRCzoh32m~>{It(sPyq5-lV z1gfS#$2LJdU{It-*YekJqKe>cK^EEM)uq9C&_jRYxaAlCnc+6&Fkmeq|@JUpX=Om2=6y z-DIap_J5e{pONgBAG3|@eWnJ#yc5|kpBVh|xn!ScvR9DoCr$SENcJ}hvL0Av`Ma2idHee@UFJv*~OXNu{eze0>LLPNGhdz`!8z4()U*A9; zq~A6P-O}x3lkyHc8v}q{$M1)p0&XoK2y~%PX1@aYKO!tV2EgoEJ~Q@lz!38r?9A24 z)5v_o5|KgD@R%@28tS}h0A-~*xzvlSO2~nHpw!1uAFLlh3}`Y$;HaOmRr5@&xsoyT zUi^}~@?jP>^|-)vE(#Q{Fgt9G+i7&jDa;2GsNFgE&0<{7pq2qSlN_DTpZ~+?)R?e9 zjcd2#fUWglH_}IX-@Co5$tIex2f>lWr9Ar@D-rx<;n4*fZs*_U@aum~e>!%io2{5M z$!5_|0+UzKr2?0(9Vy#KtJ*nv~Ysb`%(Sc(L8B2l0V<0Vd)Ji=%|DTUi-(oXq25%p)kO%fG zeh7*P?q;dyjOFOu8>cTq6W)aZf;t^r!vgLVs1KC3sF-I#Whc@#Ev!hI&>a&2d zJ$=SUVjRcp07s((xD+{fa>TX#a_r13U6P{uFM0udPqeM!McRHulCvbKO;Ma5_L!0@YO*OY_e0zO&gUBXgqn+5m z(1%6)oJSwlyA>WhRhbyebBhCL=;Wg4S4I0wQ!0a&kvc~#`3SHhWk9tjuw{c`a+M{m zyVwl)tilCfD3DyMC)e&%%`P0k0QMr!%2;&BKYs)&H)}4K&^QTU50@aTnY#Dup~9hK zT9+U8Y80|Q?Fd>g!UjM=GC8JoN3UDA;iXmVW+vIJLdM=^_|;cTY+1lNooP*27WrN%O21v3t;vzld&lZl0=fmJ(E&)w?6%^m|Y z#hU6Y&D1{05Qb1)8g6pqglp8H1vD}thQwdw4|#aVSJAN@ol?lsAdQUK_tp>*{ci_p zWU`;4;>6ePK6|pzE`5t}vMYaqYme;mMF`4}5N`nX;iqA&=Z`GXL%En!?VE6KD`~u6 zjUCLkP@|1#NLtdc(u6h|aCF{cFb`skVqW94uc;oZHVtgdXCCHMyBwMZ2buw#sV4WU zPBuB(E9h1;NVn_w(^OX5i%^w7QGo&(%s^DVOF7KVJ^Z!Ijiu~65jlUu*D-@}anoLj zp`0o0A0tQ z;d&$j3jHQDK;YU`3K-!}M>thixi)_hvGT`AaH@&wu~TF>QX;UEUAE zpqeQZff5kBJCHpS9#gez)O^w*p1uH;^GOTFcTp2n-$XSWNdD$PQs2N={zPUZ)i*rd z)3q|>j*2CV1F3Y4L+O7QN}t`i|K#T`9P0Q>jQnvVb0E1{of|t~Omwe?MeKWJTeH@6 zFlFE(?qU?-^x!)x>(b*zq)AyuG5-1=;}85hP7J=|F&$I2L7-=AEEM8SqE057EZYTL zHBQ;5}wxvb#SE#TxceYi5Y=u>ve?{GJ8Q&h8sg#r$)e4N+fT6zlkTMgz z6}oFMEe}XoQ!vCdoJ7C{L9<<6r)n&H$nyl0i~Pwy2s*DoP%;Ot>yzveD7ULTYqIRy zyz<{f+Gkxm?OL^*>3*6yyQoOKsNv(cq6siPc7xun@I*c&rk}2Kc%chC#K>Hs~s>BkiG`=Izd}}Um^)5H_QWj2Uoj}|Bw>5ek#A~ zDNG&0-Q?+RjIg!!PcWV&oKSvp#aie?CEu?<+PT+ADl&dS{s$^Yn7BL>j8~ueHD3ii zY$T#sVgsDz^8fw=Tl%hE$}h>-H_?MINgdrx@d-IO{{u*Z7G1g?H38a%{sY;J_1xo; z5rlwT)VBWT{uqM9rdZc4V|V5MwH0#PdVAmV_dV}9Fk(Eyz4lBdLgUtG zWS!_ZU|pOE4rDjyq>nz?z?w1|Q0b}OpDQ2L+)N%w&y>yewlP zk;5A?N*iQ4@S_V-5SBDJBoU0Go+Ttm1K2MOKP`X}(xOHzjzb>`qb;d_Kz#rd_9q(% zSS76`7$C7HUgM>c2DBANkl_(PzTPNBo-}%`@Bsl&ULh*gN0qez5G%$ez@0ToNeKf} znXxG_cF1J>L0G#V1{_&G6bx_f1iz3CS=ZbDkojrpi)WE#l_+g4pvv=i{g*9)tB)yi z11I(3Uy2SG}tcN1lpAw=rpqL8W+tvk7dAkXS0l zj-1XTD@pc#{9UKo$g_;cA`)AM1hNQc@JM`ncVPvP>=Z%^SSs1-4q~*~_?1u6I_UB@ zLu$r^`4hk_Mt>4^BuYv-E#-k}r8Q;;vM=2+k*yFU*)jl(G=zA)1nHRl4C-oiH-p3r zrKQ>w9G!K$#Of-o&nQxTR*?cwc$}~UgK`67H?0$$qQ0$OsSDykys4L@jdgw=-$(f};MfJ2SQ z-A3kENEj1n$>0uI>ok86z{oTv0473gNmjS6M!SHYC|F$}srMT*U?zI(&`SWr24M{W z=unLbs2UlV)i~{%n=02P5l}d2Lc;{EO{IX^{O~t-sLkW7&Dk#?Fs1}jIChG&+l%(S zWZz-U!0bo)vo4V_LU}(b^qIYyU#)-G$e)j*@A>G|9djL1hp@|_jS3#>#Z0}Czc4(j zlbaXFyN3>oFf66-1JIaMPCXZ?V~Ya25wx^fEar62=P{R0Ty;U$I~V28*?af}G5^IJ4Re;|uY z8odHiE_A))RoYhF^<9EAlh@0CU;gx6a$7aKpk2aaS{K zh#Ka#8M!q?4e{<&82H`@9bsu^Kk{f`0@fD-8-?RgauTxMxI$rp{^&F5eHJLR9D5P(k1|J_(f zuD=M}w4DDd+w#B67;&5<&4?ibi*{TkR!FHjy1gVy?$eb`p`fx=4 zyX{8|A$#d)kq&+wHoydITLz4>rSng*QO0Q*7bp4akRN2v1Vb6C)l2^R&oF&!#Zxb6 zkpH*tLjFGm{!z~V9S9C)8_+!D<_6R06MAvU&#KyE&mTH=N|@;3G}eg!5&Bl zF)a&O0AGWN%iBRCH(r1;EO8z@h<>jT(Iq%gx`u}Jtv=3+U4;sA$G z0qx8Aj<`_sAG_^1m8fM}(zu?z0Q^)y+V}i@pHeB+)6#~0Zp%M(0ZYjVy2+lEI;KU7 zyo$`ngbo9#TNU)mlq{b6u;9fa-w#29#nispG$**hDOybZvceyo%D;w>)_xNwkfb?} z+QS|J{vyj>C+K7jh6%lpbuQ5`=ME~h+q$dCrNAD$g{6v}yuOo5zDWd#BRtvV{}HPx z2rB)e)9e(W{k*^G6**8h#^VV2=b=z7gce7!wjtZ~B+|hlGj$88$GL;|n;mlH*@hz8 zL1i`5aR+GJ!~p98p`o{~0$J#>UpGPMB?`5DXeq43PwIURX>i%jHmSWPQ&mtejfx4? z^Qun}A>Cnz8~|9VJq>@+T(DB*SIbJ3h^$l@s%6Hg(6Umc|N1goT*Y?vLvrax;q$+& zVC<46>YWbNEK{Y#aw|3jJ(Z)0N0Vgg>Z+Mtwu%6=0B#bm3+E3hg1P0O!+85?_r zvF9hyb<3r&?VkKL`pH%V8I2Potr>7dw2QsJRby*T$zVS^Va7uk>tUy& zBPXr1lXH$T?3$44o#a@Lstf|%CLrJ$ z8svJMonHG9pkYpayQBJYsA@z7Mdq;(j9@hr6lKcKZTXYGOJ-4909y2}BOYsXLhJz% z_3m5DrHL)R3wtMHq0;*Qh5a{%kP69d%WT)YmR2%UoF)5t_&=*k=1WyF;VPjMCXj!x zODz;DGCITjq6Z3Oi^{#R)R@o0qIp;37bjq)5J$WW1!o8Xoem^`Rf;hIj1FT0 z<_x`d5x}V!V**%hFh{U`aK@_g3-wV^F9+q=rXPY_Nk_;5CME!$Y>c2(6OCyTp8{yr z1QFE4lw$}$J7r7&Sukv8QlB5hQF8T!`aF|>64NOOD+Il4QGOOf(E$}Cgi_1OAu^{= zuO?Si(_aGb!`55+M$87Uhtwgv2hd(KZGj}qjR|;t#BQmSWXrKs`vef?e6q=})+|g! zv+ybdx6X>tS+g)dwq{|>zY{27{v)u{z&H6u{!o8&8qV+UZ_dT}-J1$S87Kb+zp)z+ zVJyz`^}#vGegc2jp@_FuqvZCM9o$~DgWKL6+#U|M$f&d*TV(UsI8Vy=VSjfsf6wvv zL4H5e-^%x1e#enh=n`KhREJ^noAAAZ{t~U^{XxKueChAmr||J)#*EFvy=!#QuL=b2V zlJ**SbPkvUZUEvZ_CZ;Cox2q%Uj$!(_X2_~0xkkB11>|^2r&Xezm6`1zOpz17ZQ2w z)t*y6?rrieN9aGN7m2BAvH4lN$YuARfaq6M?YD8&kxEC8IYy$XMkDPeFNm&jp| z)TfLIBsSE-;dweHv9Tb-9PIp!c&g@N7KP@8~lV0?tDTAcepG9$^5l{LAsLP3x3n1qzVPr5n;A*4O7=b0`k!~$oZ}T zfiMtCWtPx&%Vm>a?3ny6+C{3Y7WeIT5>-VEZEc$oQCPT!{=jlsCn6QxHwg5ZfB+^k zV*-T;B}uBxeMtp9QJ~l*b-yUiS=~Vc1Dr%x9&Kub=18r~I`Yk8RTV-%?({K5fkJ4u zZJu>wh&nG?Z9;RKz+9TGv@B~?)FzpwRaE4&CUbt_*dvbYxR|hmz_RivO&Qu^#h3tu zb7KOYu)QoO4E=KZ3luEBFz4!Y)AuV#RN3kL&^I^~Yn&w1sXW#?&T7cP@@5Zu>UuB! zh;3F8ImxcxB9Z?j;;%mv{?QEiBSin!6 zXe2zT=Xc^;d4g zp9j3_`HLvr#Miqnt*l=Id>!|KR3?sou+mk#3}3}|%%6M>Nnkr>FvVVsbZXECU2?nD zJmmRgd9c(muP0o?58Ui~7)Kvk0jCt3l^K;eZ5UvRdF21~qZ%z_E5XdHE)2HpGqd@- zv3VqK{kip<$ayuR#idUwJ)%3jk$=?G>-a7Q2O~dNP%01(h1R)di_Cao1-P` z>LUnr7FE!`|JS)JPIRDIrt_QOH~n7p>pl2A1Hb3s_Zs}(gx@>x<8jLE z{fYD64ClX0MlTq93uGKE8P1}hFJOq>kR|D5P<9&i^nxb7XwRp6h{<*fgDFFj9>x2T}nU-AFS7hiSzQZmN9;SnVvB)}s7mBm)&US?o|+IgnsL6xK4U3iuSA z!%pHfbXZ#ZBBs`TOq(WY01DKtwoYiH!|KGrJi5(11`p(kIAI8T>33YB&M>3t13K5( z6h;t?w=m~OwpmN>VBUdE7c!3La#509{+klIFBt-6^Nzc|Ty$Zaf<{uZlAh*A<0hpX zx|TRf8TKrNz0X1NTo~H}sP7fVI&p5Wymf${tl`eR=kl%tI+hrM%}*hkQP_k?CjfJu zNu8lV&DAE5A0G@wdlP|MNTZO57ds)k4KOZv_|L#L(EOnr`Qi}ASzwv`?)g*rybyKb zaf@TzCe*1^t(bzO<2(q!Ov0D}?ItQprtNHyuZln+chj$Byy1{ONB~oQr?o*QQcHZT|Decc{%*AXfhOx2q(gdh8V0mnZdUdj7^gVp4C}JKB12#T;(j zNvaW3FS1+4-I2vj23eAj5*&`zq8fUQT*=`oDyQ#AyAP5xhr<}l>PjMOj35%&n}&O) z-BPf(QAC5m;s|4BH#@;bkIV0BuyJ3Z6u+5{Fj!Qz2478X?xSO3oAts<&yJ}ahtj9O zyVp?#VAJhHssaTo6=oiaL>L6|Mji5qLn^DqcC+{)IO;h3WSCf=W1Lh#Y)^(pa2Te0 ztbL#<5q9$g0J@(s>88OCXW%FXNu$JB1~g9<=NsrmpPF|xfiRp(+w7-e`xY~9=rW5P z#IWEC)4k$^g-5*yKV}Y%u~;Kw1H?vrOKfkl`6hXXK(dbRlkf zbQa(c}-`6__9>wnH$j7EAcao1sV$)xdI_V3~dP8@w~v$V9$3Wfl=kn<)h7IgbwRhIcr zJL3^|4PIvM6*f8bKE{C!n4-qf5OP-rCU>nOKP~GxB5*T1wONf>f=GiU`~%jU=P=l- z5i4K$2z*{#Vz3;6ptYjz<8@21V5zUtG0f>xTh;4?%Cg+E+;If3Y%wO#Vz3VB8%N{f zQJ=Fx51(<$^8`zhoT3`#QV3Y}1qG`w5~KmhU}xw%?)q z09v(41U0Z~utwn8R0^ogcYSP!+WZNG%ine@Gvy09kDVg-gms+OO~dql(ib7o$9el> zXV%DEZ5s_SZ(7O4Ik%=u;k)jVW)G_pJqsY1YbQRVcIa9gqCKroDTy$R6q81+9Z@`Q%N3h+;n#Nf z4!T-~jEy+6dbG!k9PO7C+8=eAmM&=AIhz-VRe#(0yv$f@$BgVz?4wW$EIJWWsDfFB ziHS~h09%uP7};AXU1-JXu2-m+l0h)CxXyu}L(_1Y1*DV+NG8S|6hcXfi-61cfq9+h zl2ZiN^^)u>pnVV^ANUh~Yq(nkitA!A@h}t?SPb>_zYoS$h)Lrj0kQL6c~~zP<;S!P zx%|Nbmz)MLqZ}>LtK!4~k`kjF8F8Z+8Tya;I#Z6q0 zDBc%h7N5N?cH`po6&WQ1S5R{U$WLo%6M@@^b9IEkLtCUVv`=C^dk&*~ z8e-+|{V;sWNBLaEv3_V*6zMz{Sdzp^>>McHL+p7-Y0CJrIh16TpU1?$gphEF;6gwhUrXHZ}pxhNdA2GNpqiyu3CjXRsKT? zdL^|^-hW(390YonR}k0#~D;?`tf>59Zkg@R)3K{9m};3 zb6N)z<#{_Dv>#l_e+=$GRuLQ(ep~*7s@}o1DQrXdPKCDfuE)+y!+;ZAxKqQiQ4y$n zk3fJ@t)# zKOhd*Zb1|mPB$4pe|Hy*_UrF9foXtZ%37gQ7F*l8H5t5$jsqte3_dW`SY?oQL!c!I zB_0;V$$yI}^$}-fN+6E)$AOlx0BtpO1D-Z4>HLdtnW-afOEk|a_b?whf>vRa0K#k$5{4n_5dHRMJhXa`y2nm83P?}L!0a63YSeeRa}a^sZbi)mAbS`Sa7R%> zVSR!F@=o>$Ky;MRaz`QJj-r5D9fi^cx{pAZqF@`^#HQ^h3}?}ZVH;YGI>wqWy_b=7 zCAvGx3tUq>q?Xj@OijQx`k=NcM{RPl6DtXA+OiL;$W4O$SN}@Ahne5%J=>DO-lpPv zb-X0GF?x?Yz>}1kFObKA-l10n!Zt;=-xy_U3Mm`){P{@R3>wk`+)wxr8tgqV*+^s~ zm&spT z&$h(U0U(Me7c7?SQ7t^sd-3Vs>#NpJeim2CJm8}sGR&};=|qyxugv2sf5Rs=*pv{C zM#{5(L_xC=0$lhZtl>_Yxd{S$;AxzWJ$=o^%&Q5PPu`(L&8~cK4ri%FNvBy`Gp*-W zi1$uxUP0=M!ki!6IiPI$g#q+O`xz9DtBO}V1+gNmv8<4`@50JJSOl?bfIDN<7xb0} zvb5<46*}U`xTq8d5QSxaVgou-b(eT>BdEkdcbz{+z2#1CLsS}lN~qcI^mYr>dqu_U zC^H8BD#yUOkq<}JnioSx7>@pm!f3bd8H4?vs;JFet*}$ORI@~J9<|~A=*wgAqKB(? znj0R}MyK;y(0-}?ps3~q1l6?X zAqCOycB~@8DAi}2VY;GVaj|u%28+o-ehKe12~Dlfqa{AB4TX&2h9lJ$R@@dh@^Kjv zF0`O&j4ETip=)p!vl(W%F{*86F~i}T>I`0gOT!v|!-fMlaq+?x%8#`<520CreCgv` ze1r~IDE46DU%)iEs?dLt<6^VQ`V%l>wZ%K1480#BDrw}?h>R5GQ#e6e2&W`_C;qP2 z5xU?`9&~<-=6joeoZigA^w#`X#O%+5!CzLy(HAV2$bj>Ua zUz-5&wh5RHp6*+z;aVN1Bmm743lfLA;{-S^T#{p(;uB||;Fx!>pa~|WliiwtwhUz3?Xn_P8n=VkVQVZdz(JepSHGx{)ElejL^H1IxU3~p#DZXKLVDYf0 zz)2FPCp}m0HYR}WT4GDJegO9#yP(_(Ru_PsfH4CqhWhNlc*sS!n07+|*T)+ZkYb2| z(R;FSsoC%;f#su7Zln_jbl~8?H$qFmPH_?pVGfJA0^A&Kb_Z(5Yx&XE# zO>Dx%gr#+1Ve*W9HTg9O^3LBP)f$Pn^`0G4!`6Z#8u+XaYHPzOZQ4&5IswayPM{*| zxsv?jA5-sP=C^v!wq&5ICYhu@Pk_x2fapE)fQbY>U}&sh*@{4YS)j2fzs#79m4I`u z!ZDvg+7|xZ5hUr*S{I_3r8krx`x#QW?PMv?a0dX2~2^79TCmVw#mQ$a{c27ltP=<6&>9?anaO$t4q)9bh}Am zG|zi&5B@Iu!PxwXi@v*zI|`CJC)Y2co-Y}CPuDweY#Ez9yKs9LDhriS*s4`CUdlwR z!DwfWnRPQOIm%Py=nfo z3>P<$*Cbo_eXf~+r@5v4)7a;);xXWD6w1UP`j>T*vzqGDtCf0*Ez~u-71h94z z@Ny9Q*NR^6SCClg)kcJQi3kO|B7|RkN`6*6Vd9gYG3eB!S2SYov8IrdFTKoupar>RA(?@Rxz`R9P6&vSo6hV zFHScVShpk;KMGkKE($D;1Z7o;Q-ML&Z_XJ9#Seuf*tL)_?qi^w228;E5?~ad;73^r7=bKcWJ=k(uge&Y3M#O6y5UEO z3i~B}j!JAO>4bw)wb?YVT#1JY+k54>8QM`P9rBDl0E$~B!0co{Ivz;w3kM5#et#3* zpohRC1bBYK*r+jq#FweXooc$$*<1ec%3WfsfV40Um~mWzB<@@bi-Az=*l-{%-U*yj znvF##y`bpEaB-C!pAMm7+@Xf;Es%t*M15I~0m88%F}!8cw(59x#Pb?%;MV;`1NQDG zh*KU|sHg(#?ly4{q@ndi(wKJT3M9=rV*=3G7!$yC!o~z#dnA_=q8ALw$|}gE%OVj zpu#xFj-3cRuT*1c8qe0mk=@2DhZ#;kGDxOdM3BFJr`6}QgcWaJh5{iBIF0$KDr9b0 z@!}9jn1Eu&w8P$xu6N!x@%6jU-d31X>x~YZ2P5)`b18iSPMI1TQ5Jl`ZrDb&gurp< zQXye_q6n1dXkBY`aM$NEx$8YYq@U@LE0B43haZ z&%n%tO(T)-SicStqzdRmN5}Cu^?K_szH_sB0Xl$IT}r7b-+4 zd(1#uV@^?8sRs(}Xj!cRrJ9n@1Y$Xc@XA^?OR%Ifr!J3OaEwxs>H@L)Db_k%kxtut z##9!BH3X7a^%!VH%D|P@iWIGb=BdSsbXgQ?D+IPt5NX&b+zCCtYQKDO+>I!0-k5<; z+2SJ_XIRs0hozSJ1`#lP@k}d0Pb!wCZK9$U{>Q1cf}Eamc<7VP#7?w zDuHWLDWEowVR>>+ZN3<>@>l&Q1jc%e6mZE$+XD3AAK?OYyFA)mUJqfk-t&F+@qSTU z3YnSZL6Z!&1Mz~4_y7zna6oYeQ?i(K(wN~OhyIKpF1hs63b59tAI>Yx5u^h9CvS(f z?ak!Ra1lsu=1c@kAh{Md#^9G+3r6C5REoz-eCQKr55Y`el#w2tJ0OD$+_$bf*U8lI zTe`2>Rhj69kmF3q5+4}Z>wa7i2Z4rs7cK%}o0j;7T-abEOYReNq7O_Eh(VPkU|GpFc=_jfunRIFdUP{ju6yrT*fW?5}^rNw2o z5z$iZo^Sxfj7AUr3y(=3g9#AhLNN}M6hQ_wBBDLSB85e6ZV}15!JOoysY~69b zDZj9rqxl{KvvRA*RoC$$=C!p8T|qy_TqF?UnXmK;<+Kt4XI4USiKno$`#NbxFd&-a z$UJBg3H2hwQ@!Zp$`oWpA>^#M#nc%3gZ*Zl$TUG%kQ6tZ`l3@P$?YeQ+?n4>MmVWA z`?mCf+j4;&1Jhk3Sd!g7ssQ=52kf3jw6UOIac(s1vvoNzLay2`S4iJOKJ}qOANYG} z{i1`F<x|HV+J?>D>sDes2vBPf*4Ch7%K>v+ zhcEz}m@9*>9h=7pnfaN&a5>S!Fc(gxSB=6YBXz~=)By`O2o}tWd_`=Vr@8P zp)A>Uw>ZEQmb`I}X=n;roq#AffjmGOrEN#?xY3>Ycj}}l-6UF28Y`?V+Y=ObKQs^f z+c`HJPm2Lh$L2r^;w!a|rFUy!=#~G&2Ma+kEIw^QzGWHt4Fo9JkCTOIe|wQj&ux|`51*KMr)(^${&VY7YSu??S}UgaG4df5~pij_{$(xT!b2<-balEOfnVZ zu@OX$xYbX7Pd$;*oi7OgbSL6F;WniY1RMqh?+|DXD577o$DWw5hSfa9^Y-C5t_aF6 zbOSRe38uqgM>uFdf)im*(3l1?6{?#sa?*_&NPEh`8RA5QXe=uDGJRm!E1;{)5L{Rm z_IHgH`e8*+kTn?%mC~u+CJdP)6=ZJCezOF=!t6w-jHH&Eipy-m($Cg(n1L8P3){Nb z_YbTO?zxECa&X&D{x~GGQPjJ(E71Ji%v4-c42vse5szw~?uB@;+Zmo$*Jx_66t^tJ z=0QBZ6;egv=&>3Hjp7>dcnTv9A%!E)SMyvEHmG&g1{abzTO>PtN?fwTUxXXRA+*W7 zF^}4QKeY!4TRQH!>s3}U z?K(v$j5h&rP-1?vfaKsI7uXgkSfO8Jd8}At>&yYxZi^^0lwU?Va4MX5b2f z4Io=Bfq)1qP!m3M9-i>}HR1U@i)Ricw3CHb%pCoq83F+j&m2?Cm}ia|Wz_~?*sODo z=jCV&t5!K?%l2EepFo&f6;Cg<-|us0{s+IpWTg1p5(u{C0=tf;yFmaR3}8NO4+v69 zFzoo-%*bGWjxP{$RoU(~B(1m)y|~?F zRFQo4N2kON+F{r*Vh}1Ge*!{*iRsYw-D8~-TeXiFJm%I zrnc4G+j4DXlOcr3%P_8cvRx?b5;ymd%?Y8A+#x?xCBdK zHbanRcFY!q)djFhG$t@9p=;GH0%%BM0x&BxCICwUV+2^9S8!=KT7J`x!f;bFG0d>? zYjhI8dQ8ArLdE;CLrpoV0Qjon)OsAppkc>GpZw~(?Pm~zev1j!D+Jga5?1RG zRu@2R3Tp^Jsxc;jchHOxblvjw$uD(Gl7-a9zQ1xFEp-Lx?QaBg9B0Sszm0@&^{Mo_+HtEXIKA}DevwVcpNNL`wB8xz2WsxbrI z*VfG<54&Rofil&qma0fn6+7+@Vq3`TNt ztP8^{7j~=)GzRQv(6D3Z(d#1)GOQq(FjyBshNi<_ymM6BK#sa5AMAc08rbh)6+F;R zrY9$4oIZM@&Gnr&7(GAvV-WFcpJyyYf@IInq;>`aV6w@-5IHWD2a%-q1Ly(k*O|Ub zy;p;!YZ9llDogT)+;I1L?5321ynQPOMU!<*&k+M3mGW-N8gurlw=xS%uhNyh6MAk; z7~|{}P(cdRb5%`Dy$w0yw2L}UV@-aAJ;{bKgR97~B^Wk0i!LS-O0!$x^|sWRA(wE?R?NvNR^pehxV8(<3kSRa7!M(F*JO{y&Q9O*4Mub~0qj zs}a4iMtO_VDmF${+Pj21fOe#y^ufxD5Qs*a9M%gnJw2Ufp%eefLeSb?lqm?i`U0xIqvFFylSo`OgP<5ZxffbS(8 zU+n@vuP=&+Kc?kZJ|~Eli$w=PiO1IF%Y`HYqKnR>z&))&T*$dnA%no22?%UcTbD0V zEEkegP!Iu_JBUCuMC|w!YUni+7l`Ez1|J>$=qWnnhOteB)LuY&o}DchnFxwpUi(2l z0;fW^w|ctmZc5+MIKv{xJR1Zmjv~;$l~C-H6>rUR81*=`Kuf_L%TN+WO$b2dQ^-Q4 z)vJQ>so}!BftsqpSubH=s7Yf@J;^+Ti(j{{nxA$lrvO^_zY;;*HN?Sk2X?C|ZCAPE z;ZO7C1cDUzzeaGZrgYe15pWT~mjTar|0|=TBm$YANzU{%BD8@$E(sP_)R#4eSxjtl ztF(1t@ob1+sil&~eg>hd8=8`Oy|}OyT6Rr^;|!{)aGWJBrb0!<7-p$rDmnFGfKpoAU0;V!vXZ%vkH+$0802B?s8PXhNc;v%a`kt}VcA z{nMC!*-B#KI|TC{tb%?(K^rCzY`@8gHUA^rw^k5L8EX3?Zr1aTfaj$-y9hw zh*V52L15W|1Rw_+6M*K>7(touk_uX10Mm&A)ususq%$UfsaU%KHva^NGbR8IXN;i4 zc52mfA&G!!pMwg-F-&Np>4;^g{R~vdfl?uZK#v0nfOi@bXgeWNE+na-kOEkVxL9t8 z*v_#jZ&>?6f9?DXi+EiCr=W}pfGZgzC{Hf^Tm^)YOpRW)m9uFSLGM` zZGIaSPjtyK9)75AI7qJO-cHlZB?rx1-N;I7=2}tkS|_U@W-il{6EaR8&s?f+JQ)Uu z{5w6Ixh`ZZ&s_XWYQKm9Fxlka*33mDsl5k(i^6UY33rj zedek#XTSY=W$AbETa;FF{ggRbaKqXO9x z0?g$C#xUmXH%!$Qk^Nss#Uf#Tb6o_L zrU)-G8~j}VeNOP_($CuI&zQbJ(BBaKhl)PG+vxMC=@xzDl@skSYbwi*2-G*E4Q;5x zAXg~`13aayIxUvMxk5GtVDn{6z$VZY#cc{;qu7`L&UhOWfIXNo0ay(ZOPalMG?p5J zDYyOl>?aHtA<2(|A;$SU@mvEQ`)jMe6Z)di4iVGRMuD#ipLs~96F z%PP70UwsXr#k+~1HP#d(6M(E@OaQWqF#}EGtB6#{DnVe`fdn9{7!!c3VvL|Xjgtz- zngC=K1*%OGU|1RxfUJT?oYBkzkX4KcKvpqEP@1P|E9F8G0nt7O6~K&QOaQWqF#{EH zxm3s?(BnV?m?4b`KvpqEP%b2?ppXLKTP~IxqE>Yj0q{g)0+>F9)de7{7!!c3VvL|H zt1Om_Oaw)qQP4~PN&{m8kX4KsXii>WkwaDq0;3Kj09nPD0Av+o0wJr|Wr_f@iZKDG zSzQwXkX7W@G$>lV;z1_>Ww|i{=n#Zu0+`^937{p65tRBsbKZ3@2hi&8L{KjW9YX+> zZA<`Dl`#X=%Pb-l^%4YjJCFcm6=MSES;hoHR^+nZST^M8)JFW^eR_tfcuw%CAAR7*nD@Z2HvI-p}tIV*%idD8~ z;1OC@K_RP%f}D_X`dC&`eWwjZ&kte+nbdA#EY54GJ|U~1K#*1NcS}|Q0$BxrTe1qi ztB_RyTCxhh+hmnekV000Q(0C4%9P6r=PBmwER+P0Rp{N7y#gUn2CuTmJ8|Ur@gA9GuWbs>g9Rea!`Z9q{8cXjx|N>IWX$vuKFP;e(neT&jipFq_^VRdwAA{Cug`B zzmsMPN2P|QtLdKnR%s&?eI9P}h>p!+nn#v2ROudA5dPeX8P^w>oJcYhlbz}~Jg0qpG>GobO3b|faXBY|Ah zSCMYl@sGeFyl7mykTZMvBA}ASdtz~?R3=3Vj0zO2*aIx4aUQ(JWC1pzzW z)16MR3}8r1_yTbkXuK>II{P-0As??q1ZghQ;w91D&}gAL1yCi%1pEd#{QBWol103$ z2MbhH&pvgl8WGgYu;Q2B>&DzaLPwl$u%CgZ=|w~;WRV~+;6MW4w#Ec7%NipnPt&A= zktGn@6g~xt{ER9}9JTX{qH*U*`z8UlK~0>XG!N-&j@AUU;6lzhsK7nw)SO&`7Xdg-_zB34vm_ zz<0d1T76v6w_c_&VlXd*5Dx>{c+vJ3e#yiNF~p&_0ZhgHO}2r%dpW3R>3T? zei1gm*QYsoVNpb{M8<2l)m0PW2KvJW+8$Faf4-)GW{I%AR%jn4#PWbs5;j;gGsOwR zIC*GCTk>LZnsCT*hZI&A%lkRd;-qCf4(fB4n0pk@2MTb|SynrwK70reKCPiZc zNh3E#P@WQws1!1YXt9nFx%pjf*v(k!)@?PL02qle198h?L@K6)Ah4k%i}$lPjS0Xm z#smn;Qv#`=B?PcuQ53xs3|7{wCV9XRw2d& zFz*>NP$8?OLI!~W2NJ-%XH1|?6J9PPsi2SoI4Iykx*_7&gRe8(SxxQB3t_;gq>5`T zri~Glr-XrWk%^$li_W6}wx*2;`Cc1tLgk+|&dhWEc~` z*fnOLdg(=^qF#c4l<+%V$l9)pcfA}jNr8A=C=62|=`lyJm?OxtPKL~UI0z?ETrbm# z?#DY|Wz4FBz^xjzW<1V^z#fMoi%TTq4(hmk+8YkVDUn9Ev(zI%dnh?*g6Tn4S`*Bs zQmb{d3SxQcMC62w)5ptG)py!p^!x+h_1N=eEVB7jpVpWcdhD==qEb7l{W<;?6O2l| zSA(Tkp3=KK!IXo<1j8Vu35FAF>%-Qo_$O%T?U0K{6y9STyB$ubhp-JZaAQ#>W z4%HnC5vx`^?=Z!9)mmXG-py3G+H8^89v%WP1HO>~@xu#)(D5VmDzkxedGLf_B|yV^ zD9`L70LEiX0F1|&05mPe4C)&qh()7wz?G!0e@%U|9Y=gI9rAvr3FhNaHTeSGlf>#L z?!)4-!O5dJR_*tLH?CD5{QUY4;0vbVm?3$&`k_4k<^-C0@bl^5=W#_-?>Xm%6JBtI z-{e)O-F`);tW?45i}?7t-f0{sRFyusvA%}(mIG$#^{e0;>pKcrTn@|791tTEn(11^ z2ju~*Y6sWWKkoR2Lyj0ZSM%G~^R4{7mCgTCtqij5iyY#rwByjXLs?yt&<;-^nurXdhI<*eaS}wYqm~t?QkO{KS1$*YxaJoH~hSf%~Dgsi7rX z9ir?AJkWij+B5OByWxCdaA~f%U*vBwYKj0TvMLH95MWg8r(JSUQ1!Zrpno{Jf{Hn4 z@0GIyJs@d=MJESph%~r*8a|~IvsE(N7aN`;Q-a<0S4VDeGt}5Me>UaN8aunUS6l~q+*2!gBaXrIQ z{s5f)0rdqF>XoOjCCG~R7Q5+)t4#&shSWHT76zhm;IBBPMx7HR2WPInTQq&_9~%I| zLOB#KY=>VwJRS<4P&paHwyfh#h_o?0QfSiiaxAbBJ6a`UTh2m5WGpK#YMM2%DMjFY zZjKWt$^xsRgZrARq`9b&am`inz^vK_VzY?()ib_IPL3(%E5;EQk9b++J!nu5KQ#?r zja9AJ))@8t?>&6)mr-}PT__%mdRL9jd->;%6I@aU=G%p?dUIq2rDiU46gCZ6THAHhMpFRu;ikT`4c<)54{EY~`&gWb$FtloaL<#$K zRbS+SzI#OE;^xx;yt!vs?r}Lm+*m+!N&H$D2RRJy(ZVkooCcf*oWtD%v&OnJ7-z@he0sZj9hns+d$& z7uaJI6VE@9Il>4r_@nHwtoKTH*LL_)4j#7EY{^Z3vr(NHjka@ z<#d}t-Nt*qvO^08vcWmLiL9>U%Wkj)aTjQ~3*3b_HtB#jHrauI5$uPt09@FY;3D8R zL4fr+G@6or0@$SmhCwt+P z9-T+zwBz5SJHzT%pCy5%>*$(^&fqOPx+9t15@NOe_S{`8^?um1?b>-_iz zVkg-dCW9pk0Xo3jpMwr?F;h8qicOKORwME-v<+t{^3 zhABQ=9qXB6XEs=bES?3hX*jk`)R*G9&*Fu<0Xg1*@n7XvsAz4|!afKWfo;#;`B@ux zVP8QIJTU(Syw-uHVsA(5Doyw8f|F@GER^8x(c28Tec8fZ-M8Fz{QWEn%%@CGy?`vBT z-JSRJE#<$Vn~>uYX?B3P55ljarJYu%cCM>3GhKI;|F+wPEuq7}n5OVA)|>rcCTrD} zT3w}D;=RCk?_c0w{OiK7KLQ9fwm{NUNmB#A*5*p_^MB_#ZJO~7c z5$}wi{0bEPjbReioiL~nn&#_jL++=70r-|`RtO3Xv69PS3BVP2}7hD^JHT42vu zzdwqFFo@H3Sy#ME2SuO?0eh!Y{ny^E@)z&7MZrSz9|rr4axo#LZ@b-$57uF5k~tOA z5LSfeYzK1{e;XPK{4FJq!%i}HlQ9*c^E~d3>d9|kSUveUj58F2a}le7y~;JLu<&&s=8P){%$OFAJhBM(Rb3%uzu2A zR~$pQ5_QA@A&LVahCwfdg`;n?2f%xKt%%|nTw9Kzes%4X2o&-}pDS?teS?gT#KzW; z^HpXRaWQ~gikwqAGLE>#dTEv}yiX<>n@8{ku#XuGfMwu|>b!c}1a~DA13}D;;}_!j z(moDZ$O!vbrGS_*0d$oD83v}nV8zt8r5n`oS`y=R|!kL|tx zt;pNG?S0<%-gos*?ftCRw72*FJydS*m(q9V_P)jur}N*#$G@_@e@!&b*WUAk_t@Sq zIC^h;Qz$s!`2M+P?9|?eU)|o`fA%xw_I^EmcW&?JIpTEwA$+vAx0veD&55gU(eomo z9awCi4<^ryfT4kz;RRA&_@b9cxf;QJ%j8sAHA1e|nYEkXvOtCg7Tw1*6I=z4UssI`gF|z5YNRozEjkd4FMt_k^35Fnw+z zl;=mh*S-txSXgpvBaF#fSL@m_4u#uvtlehJ_GbOs7Kwvy4R5Cyz#7AT-cE^QZvkj; z#;qDG>Ijevy`|T6`~;RoQEkPI!L!s@z^0JE_GWW-5rB!2F#+t28Z(G{f+HzepKZ{; zIP2|u-|-foA{BOH6|Mi`toOFp3_};!tCeE$oMXSR_OsrN0a0mG_=54VmR6KJ>LvbmME)b^>@#j|nWuTffgZ{3@5#cLEjN>k4eU<#M6 z@85FJn|pDsLdM42qKhT)y$^b?fI?v<5d`K;K;RyX(#nM-6|DFKS_i$|5dTjf^jE(+PEMfQi9xcjD%5xaS}HEO+kt z;GS5a-`@M*eb9TFIlJMh+~bdf-eE*$KF+P8Fd~x!=Bn6@f2^C+u30y^zKqTFrC_~q zTmKF=%-7#;BeG4Th3MMQe%Z34*xJ zzxAxztb@$hPZ(|$kz+hYR$sK=vK?22)dg@RnlRgpwHb?Hw%f2DL7DSUsigG@fEH6E zg3@=(4#h{B+l|FAfbD0nzY`o{9FYp97z8$zWG2vNEQa8nCO}YTill;WDFCZ|5vZxi zg|V1z#=h_4(6jd3I@XV@w5Ao2LesDcVp?$`aze)GW79F! zHx4ntA%EkiIIa8yV}T{Zbc~-#?fq!z$tM4Tg;O1s$|~ z=~yr5w3&|W93FeJ47g=F_7t-a?AcOuJ0jHQe<>SPMeJ zqNubiu}BA+s5F;xR}Khe9hB}%1EH)N=wyRbm-rWzNogt+{aDFr*8?voppjW%f!Z)Y z{+7#_VXK!@Yu1z7^5g}gfsASUAazW?znu3xAJwZ=kl*~574*v2T#p#sWr(*klYXrG zt&L`}v>2XBhK}biQTU@%`AK|OU%Au!fj2yw^iILduCAL3Qw`&tu6rEZn_B2^^V`_ zMZ(#x-ywj+!Vs;uDxqr^tpP;#`D&=?>4Li)&;6k$tN*9{xv?8V~R@w1brV?SM6)6)yr5Y2! zg3y?O1OXmYETo+vFzY}9NWz!^W<_HJWkJBl9d!>8Zia>@HUh3#>6%2+X{(F-aLwWb7$1M&TGj05DwIdVH@Rr?KyP|YA5J5h2K z4+iC;X8;pgVoS9?0MrjFh@VLPpnw|Zpa}X{IAjcC0v6&&>>>cpY0Ti_>g0My^@`=L z$+O(pjOtJOq4Po?tc@zH0d^CN2|#=>CIB^ssS0dA&pl&5g0g5brl4=y09?;+)ZiBpsB$l%cyZ?>2jRH1@lT@TB&>p`}f>;YqfAbh2`XjDJ)d8xQn9E zBVq{iHFUW^ed3y8(`m*-v2dbPsN8L5~IB7ra|Dw(-^v0@%1WCV+Q1jS1k@O=AMSRAt64 z0x)JYCV)%AjS1i-RAU0Tp@SIg`Q&H}$_qI*R_wQIKVi6IQ;vL7b4yc~So-Xr!FfYB4H4y)Zwg4(A1YQkUyqzJWc`DMOH#u!d$If5>W z0UIU;flPoix*Ya)?8DZ=zKM;y^*g&*Sb9H7-S)~4y+r-{mKUQZKlNe!zT|yy${+l5 zTS{&9vlc@;q6I@u0PM|}0N9%`0SsJWUXd0{QLjkbuGT6uv|)9s+ChNUY)b_7(!LBl z|BBRFxhf(R3@r%EI6DIWw&P1Aj;aDoK;<@>B$FnP%P^UgRK*=kXH6OgqHD1VYF1aRtE*vmixo*p z5!;Fi!A0oeg?MG+WsNsl_mA}dJn#FRnKNf5Ezl~%Wqxzcd!Fz5e)rq^eIG-TT6aml z3ay4(?}ApQLMzjemJqWo#|)()DWMgb4zbL*YqN~PAZ4V3F`rr&lD_k`N)#OxS{40m znIj2)HxfLzLi_5|6!U_tVkU_l*Y{*;*ruRr- z$wrJwds_xecJ?$5jH&}duqLEP2-;IcLbS@>qbfpdIb;<_hQuv7f;6ErHZ3N%kqjj{ zP+$`#AxZ67j_irWU_`2;y)A*g)aRjvP+8>jjhNxwfn`5uQhbp5Nx6Otwfg{9oM?=7f?i4bq*6-rKu^YKxVuxFV1F{A>LS~PopI{4Aj-DkE)h|on45k|Pf|F@ zUWVYtak+(P#Q=t#_@T$I+g~ zHwSJ=XVJY%km(?P1bqe5ztK+mP*v*uWfGC%rMZ<7T#sU%6wd`1XnFy$pg<89*F}|b zZPJYsV2I@a*>jp)qM_Dv$ia3B3aRMp=tISy(7n5xT?0pf+KuB<{NF4DJE5lVo=G z?@{dEna$;!z$NteGIYs=u!`1=NugyyJ~952jBaiYH}@bKV!AIA^x?G+z+(Db?y0xY z7Vajcb0!{=o5xvoCiSLm9CIzH&Mh+8*L~{-3i}f8u{~Pf3Nbr4hIAu`c3&n{UkI+g zRwM)|r$`7+Q&A*ja9asE#MRdlJVir_aCVC7m8qVfp)o;&?7mFUA^X4*b3}MF6tC5B|2dwTqYWYl+>PETT+@Zvq0Bl9G5!_bb&gzmChXv=hCx4zr$P~ zBI^9SwvK_VoLQh(YZWauqRs-9X`Bf`9VJL_ImE<-A|dFADK|RKWR4sxAa=q9~fvmCb?2=FkT{L&9Fbvrm+; z7??rM(+C5&Cqdj(f)2hSA?SHXg`C*WmJJhDg5>;a|5Jnea8=YaG_cWR7?!vx1o|s5 zlZ6glqpAqObg&|l6+>50e}F23pNOl55R3*yQPxe-i4>%(sOasDs@_(yphyp~UNLwA z3=0Q3V-D0OhGlLYs7;aZicTaDx)oj!P8vXKpb5a~A>$TT&9!qsKn=R0u|hiiDt*P$UG0 z;3;21Q1TQBL2@Y~agH7bv_yi7OerNmDVV+k7CG_J0bK?4Uz^ELS!MSMGccX;Y^%ESyWGnWRQAT zvNMQrpiB)3K{KOB2)e?GglGntt13dUX;P68tV5Hd^ttl&qPLC2kwF-eRLKQ)2EoZu ztRaWiqm#OAuSAR?Q0d<_Jwr~Zr7Iu)Meay@>-j9ZnxWhy?X_?ob(;_@u_+RQnQ=uz zFp?9rSqR2$iiF@eBt;}yBkdXn9+e;n>Rum6I;9AZQlqefI!Fl0o+2R_C@WGTDK!_C z?36MN45%R?m>f_f1oey}A(~Pnsv-pQ;EIIc99nXOYFruHBYN9N93yRpBz4UqjQp;a zVU1&??WNWtXdzT{@ICbvOfm1cQjFR4X6j_>C&zs1v7^nlR7D5|{fd;Z&7MmI`>wpC zSI}6EkYtq_7BsY35VDPpm0;zFWI^yD0Oq>La;*o&T?=#JV$ zO|DgwO4xMQs;Nd%PYTIxx}(T~9E2oHiu%+QCBAfJqRY-wOQe=V0MI>fZ-Q7PvUZ8) z2jkXUbxTRQnfZ>4Oq|Wkgk-IRl0!5zf0R*O#dHAm;fJ)DIUXax)Cw>l%?tvQH9~2V zN>$~lD4x+jo~a@op`xID&m;n-MK@h02dgDu7R&7A$Pn&PW;^xmiGv%^JdIUHvZ9g(A(4@7 z@bM~PdXfe|O@nrfL~I6AC%p;r^-uZrmYo8D=BvR87=&O2#UHMOXvFXY=9W62IB?*Tm>~KDrYC-fXEoHi z@NB~Kc|1Ai`PlLn*yoq}_F3Rge%4EV*94R zSGhG3``#x80uMnIpMgo}V8`T&6le-`8gzQa-~-FC`EL2(L#zbhQ)&)QMFO*!brgY_ z(jAvRl&Y_U8cr1e3-0|IK1|`WIhR^zn^qv(1le~NY}#o1Y4^WkvQZZb=?sw2=FxB z^B@3sTf|0E2nHdFgrH-fNQv}3;;>}*Jd6VoH6#QB0YyTvfT2i;_B;kuMF{#SiiE(l z#7RO3xSYh^bk=q$8ILq!HDXSIC{(5`ZUxLkhDNN0mQM8 zS22^$jpfdBQ*}c}8;neyjl2NQjqx~)#bJB^bO3Z1bl5qeK|3(6#4P<9|L;qkxo;q})llAn= zguxg`j&};~>x{oM473B*NHXOysn>lJYtF_B5~CWuJ&;|AlkmO@Rzz_~cE>N>B&sk{ zB__vwAWH`lXA=z>#jFhvgSrTVc}ulWFDA!GheZOBMMZI$8BPen7+sMN6c0h$gbZ#QQ6yw=+g3#+rFR{l zURSFj2ew!h0X*C*MxX6US`{qWrOY@mstyP_yp=RM&R8c&+9wF@r}G$$iZ2O4I8#p* zOC+xCb94mRry`fYp#e3uU!65kI-*@4At=f^MJoi%3)(CMHzp|(g5$gukvQtFgpZD+ z`wxV6-PZ=&}Qru?3mc2i==ClxK^YV{1ZUj)YSrdeUJp6SYK}hEE9U zKSe@N|0z-;<-Y`$>}gu#K)o6gf`&zr5DbD9kvMAvx`IFoN!q8&3y&V0bda1wTaZ5z z%XZ@$gD5Q9=IDPst#2eUGh z+@dKagkyyJvSxUGDgv4+u7c)8-H@D(bw@B%x}#)pPyf_aVABE|Og;^XtkMrIWQQO* zyBgga_(YwI)G3zXq_R=T7Rf~BE;X~NainfmOHfuUokK)$4C5yat`H=!A|Xg%MM|V} zPz22`9mavVYDfq&vmzmw)>kA%OGk~W2yyR3Q2Dlmkm7_c8 zLRRAE)EN@B(%{GmL6Z{gKg2urJuInC-G#n1v!hrWhc)?}nm&dpk|*kiWxn3*AxS7Q z7xF0?OpM6vpq>ScK9C|I7)UE3aW=)2I!s4B0r;3yqe8M5xlnJYo)U@r2}jf$2ezvr zAsEmr5`x)RMM5;{CsaiUsxd`EY)Q!k{irQY3{cM&rWBdVT~d~Ry+uVzBo4Pb;?OuytA>Q22c$>{N`oRH8i$Rl zA_Vh(iiDumB1gD7g`_w!hk9t(RFPzt)tV;`+m4@Dx zPek>kcqKZhI`EV;3Nyi}pc(gPaLQtp_3Bd%LA|(=sBd}yw0mgy7>X#6+Yz8SH zDk58}o)C@0X;l$|Wf?_6FuEW|LkS_pdRRDmSgJ^}%j(J#ho>DiLmcYblAa?B0aD?} z-2AEp^`H7L1Szga2BtRSvt5G4CPgNvDm)XGeh$u=N4#1KfhsJ?6 zwIT#9wIU(N){2B^9L7~e2u5v+gaDBv^twoj^(f^mq3c~%G*28(Vu~eieK^5MBYQQ+ z-i5taXOe{I`Ot83Qcb2rFDV2SnIaNr9QJ{`bQ}^O4!s%-A?V5}5`uO`krIi+2rSug zXdKwDehEQOPpt?+wpJuWroh9|w zz-034Y~_beT!opcT?vBTU5R@@e7h23Bp8uskKz2C2WHha?_0WQRyoEzJH(o;)&FE^ zM(bW`p`s_*E!P`zx(hKkA$F8tNm-)75NeNRX>V6;B_*}RJoYoWRgH*A96SzGy74;B z$fcV|YH|MM^Q7|5!~o6_({>ra?w+kelX$YV;q85Z87>a~suUD#oY&|U5nZ`KKr&ffZ%}^u+MM;qm%{0TR zBE-EuOUIFEgkXOTO+rvY6-hQF8IA$=WF?;a#}RMxIdzlvhD|4h%F?%4mX!j`Et|L~ zA;tzD)O)J7DKcw^$-O#(JV}BssUjivN@8+a^mGXkA~T9C4H!x3UDf?hdioC_;x8ii z7?p#hBc1?iKD8hOJv~K2vKJCH;(f4WN4#;MM6C!xL!(GY_ClgYyjS&wpmIe>NkB)6J0?dg_v0y478|Nz0*D9(dqR?oEXA9_XjwkoIEZCboal!5fj5ta1tEV zkXhMI*B4umTWg1^5x^4RK?!!tD-wb!Aw@#a0tnhH1cFJ5gg~f65s4#W$4WF3Es%7? z62RX^u_0@o2)d6{ME6nk#s$eD9eCfPNC>)iibxzy>b7b6fg%r*jtK&& zO{)zdU`CM;G?U7x63Jqduw-X3sT%9 zzYn1{alp;$+zcRxtnJ-3tK=-F9o^H86$IUAAQ!oqQ-O+&b39eqEX42>S98ob4-sE6 z0B47BN5(+uMy8xr6I9}2O8-t&YoOeK+6dj+WZ+ZKfF^Do<-yk6_Ao?>TRw5qXU!&m z%|I`Wj6ezLxndkjG*L6)*QJ!e_L@yT_&HKDKr)QgaRf9lJoQ0DxbfZ2($GVv*{m+< zLA9DKQ`a#fb)^IK7|RWJxVz5>IK1-YQ9tIAuPJAF`taYiJS7mg?D7P8gMBDKkEI@a z!{4ktL7L_d;)|B3Q&_|fR-!IP_C|^NHhM93iCP7v)LgJAsd@xU)UV$reg)~5DN(h@k8YCF%{hk1D4`Wt|s``$I5JD!u$$sEsEvkizl7W;xBk zGJ&n7^LE0-FieytzKqOUx@jxkkmwL?+DeJ-GDA<0)h_B!8&@slHe#hGm}M{}J5v`7 z?!>;iwBX9|r=xt#f)66NLT}L>PF?(Z3C;kZEx7V48Exs#=S_3pW5kaBrb=Xun*}T-7 z^Ey9!3`3FJO%5@yO)_N2W_9;IxXYuL@$}zUvnxTP%yxaOU-Wzxjvix zjT5DJ1$N7oR(neC`h6;?vH%(gN2Dj$3Oh>9+=Kf`85gCW%$o@Dw@R{ZLz>+Ej3&Y; zlv1yO<94#<`XK!oYy_l7@B#Bq8k3AdB-biQu7`zdNgGw0SGsX5qh^}U?P^m<_R;W| z5y{e%eg`IR>1qFX4Se0I`eF=oRrGx0W$J^dVNysSW8tAWaE=a*d`{s414v@fh-Z@!cEmpg@1Bx3`j;q@Eco|&QS+dW zkzF_Ot~odrA%Bp7LII@&)KYW4?M=U_VXtIxY_9OHWN^%*kiiALzLy+!U#2lg` zA(-`3L{fTJi7iAiRzhqGQ4oNk0WoTeEDS3Wg1I_HN@T=uLB$^N8wVQIkPsAiMM5xT zph$?0`1@2v2&N7d39;o25*8J4Km-ZqW`x+;QxMh#RB^i+8709qfRLo1LFQm}K@WF@ z?6uk%SHnW=(rE98-Ya@}F40DHmQpgclK(sFEO?io{epI+`+-CZ?q%bmm$O)2C#H5x0uaqO) z1=C9)1_Dn)67$KyNBe&Tfs5X~gO@&1`oT(JV7BnV>kl%gd!d7%W4irQ==>ufqA;5r zPGKcm6%4!ZYZAO1?0E?K_Cs^d?}OQUwt5ixJ-~T?4Vr5mqN7xtRGKW>bVtFD&qz__}HcP3KQmE6NXD#BUD~;8x)u~#~ z!|7QdGXta*CF`!kM{}x$9dyrTVSEktZ*=KTU1nK32Xjq_w=1dfe?Hj8km_swd+u;#D;>tVFYJktO@SxikM;^yFq@$m?XMN?t4ep= zxNmUm9`WKa>?7B&ZR}_)#pMP=oB4I%djpBD(6x7vOQcR<&)|ytg^xQV5P>~Iz}nFs zOx{7GrOb2W^gS@KNvxOd*i(yhrJy@_E5nOKkiZWR3_LKf`F3c*1hdkn1i@0VVy=qY zJSo}|l^D@Z0HtJ2T19Z}j9bD4HN#IVLrd~*-V{myP?L7w{L(dP3|5{Zlq7uKYr#wc722UC*aG{0TejB5EO@t zF+kX?CPaFv*eS6UXdv-MuA{F;&>ell*sEQwWm%X=Nn z8dF1?#fZiu9G-)CMZmT4Ga~HFQ;nz+L{G1IM06f_7(;McF5P$v-G#sgeQ|i9J@%|_2AfA26V?}x}91m4TND@ zs=gVjcbVkpt&2Dpmp2`nyXs2sI=X+-)4v~en%={K)mDi|tT{6k30Nhe2Z}6--l%hn zf{Gi|z=(Z4fiXS;Vvf`HL1qJ({#@7o4~tD!dSk=ku}? z%mp7GngwJ4c1p*p#Kdl}sbAZCqQGlo;0ysW1{TBcp%seB5mZ2i83rlcSj&LUlwo!! z8es?}(wf3@4eH#n2(T3PHhv^q@J>An15-=zQsd%_noNBis%WEODltBoI1RLPBPZvC zpCM7`PkE5#Mdx?aIf|}{8<1UrGJ+yy>5zp;Or;BvR?fOn00%ROL*~L=yAWBD16A)i z6rwtte?b1_a3}Zv zyTKcu9_-ye)cf?gz56p48leg`^C8<2q5~LD960?N)P@I6ZBJZfYWAa`v?>>3IN8ay~_xt4?m-h$cy;0u(B<}%vZ<6=T^1emh!}9*HygwrE z+vL4P-XE9u9rFI9yhr5yX?g#vygw)Jt@8diym87$^5kz}j%abifz!VRI^0IZsb3-@ z+AiYP5Bb&0ukZ2e+V0{PFRX2QkY7LL*RT1tmtP0? z^&r1q&(`^Veo0$@H@}urjpeOv6~8{muNHo>U(wdXuMhL<1N<7`*FW>?{rEyoyM3Uj zc%P>RF}eiLCFqt}pT)!0=xYSOz(bwy5?qX^rOrZ*gp&Z&Sw`?@qqB|RJj5}LeT!fY zBIrEO&mN1{H2{}@evM!uf(5xp)p-v9SMdLbI==<***dp_TRQL;=oJ41fX0q{6~Jr3 z)1`1xer-_UIPX`W)(Bp3L!JJM0ba~`&VB{}`!wyJ=SJ{~-$zOr7iSiqLGT6yMt=Oj zgssI6rS%TH+Etxqzj%{Xe1``oc8Et&P)`G1*V`uU$H!@SmeYeW*kxoi5-?ky4;mx5 z2rpWeACDs;+y99`74HD>s=8(U7=W&w4jt>%V$0(EZ8gD90qEKa0_N4vfXdHf!4iJs z$CKX^i;qJXtdM*Zo1XqE2R^!E)82Q=!+!c>o#n{$pS^s4_$B&Dlu#C=? z;KlF;3EqWQyKU^i?T^wflf>$SXY%96yYck{Dum!68?gka6+ujsl@EY=b|7C7On`cD zXQ;&?z5Hyxz5%BfQ{^UA*N@Wx_Rz@vc$g+XBuL;z9;XSufmc7M{L*>sE4B(hIMehe zD(t~?5kL4laQoH8SFXcH1)h%|gBLv^mtQ9MKHQ~`_Yu4aA!0xGQGz8VWWOS)fCcJ2 zL$Dg)HH#rHdgIp-c{H~cu6W*z zFc5qVG+i&6{P&1$pCA7WLmiCbD)7swx){{67{MWE0QJ=2l@o7wz{eWcXI1(>IvB<4 zLQwgo%jdJIgP%)%{XERJ*j(|0k@7_<+<>Q@AN<*IIBeqZEXLuJNJ|>u=#0bXQC9XM zHRZ>THHhmQSf9NUavx_I4fuc4O>rb*FY^+-vs#e~vsq2jDDRk`{cC`00O&w5&zJm! zt#diF-vc1OhXLroaY!_aJ;4i9-Ll>c6_&4sq^IKbHd_Z_y^%@Ccrf8R8#d*~4>I)` zDqM!wJ!%r!wk_;k*WKM>wRd=%db_*2MWrbu3eBNdtbM&I^{(sKAcms7v6w2ghI(RD ziiRTXE!|eArzaeZ1>3vUhobGFu9!7%-q}6T=CiNu>F%1>7KvD$!PtgK*n8?C@A^gwu_c~#$Q^jo9do!<7Y_E>wUqy0VMsJEjXF=CAN_JqCNk#IB= z>yBP|wfBbhSew^kn2hU|@N6j$R#SI(M>y1FwY2v%hoUXv78vbp3P&wQSuh-pc1Nu% zB__jt&EZI_9dT;Sv#$1{RjKqh`z!n{<>9J@t>u+fm7(g2icob^xO_p&e1B!De|}|o zQ){{3TRzV}&u?`j$)m=_>%tqnXIN)=8MM~!-mVrW^P$+>GY#9Fp~yMsgktA-y|j@r z-Q61tcDDwjp{{jd>%BKx?>*Lf>p5QQoyS@&y`2%O-K6!r?8H6m{N`|1%iJ?z76-^! zR~~E8MI?XPoJ7r8=Z8BZu?;X6i}r@C)=)=J*g7vALFbDf!7$=0g6N8L_jngA^7hQ@ zK}vw}cX)duq2{m}YVV4L*C8_dXv7D;&O`2E^ah)HTU*0X+tOLya_cPLJfFqq^m%Vz zr`B6Lxe1_9%~xib`Fo1 zLUH*Dt$B#4e$OQyYxop&JnKD{&x>9tsCBw!{p%`^)rh+79#HFS%X$QKgmJXmW5v-Q zT@E^m{?q`d_d?71IVk&r?`-f`Wy{chf(~DX#v8N-U67v>Uu{|5H6AO{gw_&tVjc8B zN6^chbt&A5VAKaXx*nZJ(BX}iwFR{7L-6-9(A(fXX#J-wtBv?x^oT(lzh_y$A$|-E zFwUGBeggi3#(#ol2Q>0C%Niy9IlAXm{{`CRwFvJY;UDQq9_uR5QOtx5fLgV1>)jrS z+gGDGew_#PJbpE=wmc)~VbmZ2%Fqf;zs~a1oM;_Yf1>4yppy|f$?}vTE~ihiJfm-c zxzjAq@M)lPEzkH|Yxcz1mdEF}toVGGL)>}K!KBAhxEZjDM(fZmLHvy@w~FGctXY$Y zyRxg`4&rS58rTbhhGDlG=6k@*`z?<*4mT2TBWZaWZ^8G!L+#7Zy9fO21Rvi7-ep;n zyJ6=+*nJ56On}daEl(VA*7)C`{|)*G)E@z#Keep-pF!=hW^3O zrwu&{Ws&hyX6PFYJcq1PH3GxYt24jB4jL+>#3bB2x@ zYWs_7Lw_^GudP3Em4;_Jq|~-%>wArTx4xYZ?EDkIRP)oQp|(4=y^$+af80>FIorRQ z8T_;J+oAlk^W&k++5EjiUx|TNV&Ih+cqIm2iGf#Q;FTD7B?exJfmdSSl^A%b#lQ(~ zEnBmsVey&^BV~;zU3krfkvXd`T)lV=6y$B4P!kMZv9ckSJ&m^1A6Rd#iu#6)?)+fT zzn~ctZ2`-vRME`2vbyF#d3%e|SyShB%ZkqtJJiWczFF}t#j1SSO201{Z0_p|HMOtz zm*cyqJ=olV8FQHu$DDad)Pn2!`hu9N#!NkCvSS;9>wQ+75=!dH@M{o9LCkwLUl(j{ zyDqq{7gN;mxMf2}3zU1Cf=%sREtp@8QzFUO!F&#*J$EkCoWS5@Z14p!9oZe^ z+;*_FD-fuVb$|epX0C?6F4v8$`OEm6k@&0`jb=|}#`mXVZX`9cCD(jwd^lrkVD?O2 zWfC4qNqHEJi2ocf(ma+g+iXoctgbd3Dy$PGN98f-3TfMnF1kZU`7-5oCM#J%+iOgl z?4;Q& z_YFbJlCO)k>B0p{?ywczp6yI!M|X3mBV3JzmtZTV{$qjF%w1Y;t>}2R>8keb9`UYn zT{s4t>f|0{2u04{(Y+4S{Ic}2q&!f+teJjJqz#aBe>oV2Azj3&YhKv^wdS@^G#HD9 z+G9PaHUdzq4t0<>1ds}W${-4SdyGzY(#^d`*iCVC@(4QGDM3#i2^T|=2$nHnkLXL^+sAZQ$H^)SH~w+550m|INI6Xg;i~-Gxv)+x+0;V zsg$s~Y_BNO(ReYlx)_i(O009nn%k%|kkLUA&ZrM()LWxrmvdV(ni6z6G*Zm|#HC#~ zpwkds9q#Gv3{!clCusDU^4560dO#SH5KCf^1_=tDHGcn6Mee7`2^y{EGT;lcBPt2M; zX7bq4V|~8yqT!;*@Nwg>WxuPKj{AI*uk%e8$B&&ZoA8V;iWg13seZ&W>KUIkinlL* zjBmVn+A~rdFB&iM`pV$F*LbU*%iBBY@r}M|xX3qtp?9)q!as4;#8Jbqnk<pGJv1UwtuB#K59n z^=G=*h=qURod(CxRD2^Ayr@57aP}F- z)Zlkxt(*GO2LIXMG6*1$pAWrW^?e2}f{>P&holzupD?(7p5j}ug-QG~H~;}f;t7K< zexqW_A<+I`4W2akM{iR78lURl2*C*2j~IM1(v_I<3$(vWux0rw6n_Qjb_z6XydG=! zPF(KI#@7e0g=2zI{w`JXA2Zlqw;eZ_@*{Nb&B#Y|&tB(U z?856@_!hxPZ?hhJr@{96@FNCK8~uF-Q%;5ceHQtY;iJ3?@%xYuB|e%I`;qU6DZfJf z9~f+PDqezoMtz^b6_EKLZZ!A|h@U&peCmDQmgX-IO z?i$67M*n_;;|8~)oJ;r(J`v@b*!zU~_Z@>94Sx7?)u$X6<7eYq#gR$HbKk9a!r=cg zxa=pY{~?rXx?gYbM^UbcDGx^XmR+T|@u!M^XYjbeBUh`w^)uC9dyV26gRcoHrd%1_ zdrhO_`kyQQqQR6mqyBXvqyG!Vj~GljH0s}rdWHN^9*uZzv*MaRDgLp+5ra3jsQ&1Q zDp%tTE1od;QGRUd=3s7%K{LEMU3xh{06%VzkKIFZGpUQT{!!?Sh4USx> z_{3|Cdyt<03Sa~WfiIaQG+S7 zNc;tZDZ5Dgb%QCxNc^B}-{7AZOqoXN|H)v=HWD9=dYb-G#*z4q22<9N_$-4d^GIA{ zFl8T!8w{olByp?3l!YYTVB0tNW`ijkN&Qb5Oc_by|FG>F{7r)?GfDlu22*yD_}2zg zhLZRhgDFc%d_tFok200Srx{GyO5%kEQ^u0`VuLAbNqn`zl({6n-eAgJ5+@9%3?}i% z45ln5@xL2PnM~rH22(bZ`1=M^Mw58TV9II||J7j1Y!aVS@?XC*IAZX}(9WLns<>s#7mgWh%N0Ihuq}5u>v}bB%O4iOk5agA%Qs#?4EJm~ zHWXjMww&33!L~fvmkqY%*(QkLo-H4TVlUX1U*oPs>f7>l7Z_}r@PvpxWw537N3USW zgF*?P)_d{7pKQx-NHMkK_iY#cg9~#Hkt(VAx4ZBX7an$D?$~nLKeIR+_qyPCP0buXN#SUHB#!zRiWXqX|;W z@IrrMF8zmG_%AM8dUQ5FrwY!vccu&b)I3s2f9JdGuXW+;T=;!1{9zaVlnZZDoc3?G z3qS6{lP>%l7ygqAKkdRtl}LP~{X5=;{~N)61kd;Jpvz{VTV|mvW<8DvT`mjVEDK#H z3*8wDT@-5)&rk6D6wlA_{2b4dcv5(#@catTukri_&mZuhnYaFo=P5k<@a)I)7d+@L z$}Z6V2l`h$&)_+LM=}>uzX(sU=7Jii#_eY}J|4sK3p~HXvj@+&@H~L$w|MZ2JD+KQ z-{ZkA{(pEqji->q?6nSSArqF@E?qjAPj{KN)rB0$@fpT|+#(mA^a|(@TCCy{xr9 zRnF9yvd9O;?A1LeoDc1@!i-cTyZK8(JC`1}SQ? zAk@-c!Q;WzBdsFK_oj*yNHmL8+J1a#$}chv}kMApD>PMFy{Q zY~l`b6L($1g~31*;@IrKU1)Z!2MQ@JCAF)V+EoRb+QOmVPZsem%p9pa`h-WHn@26c z9X%+@xur8e0kfdr-RP|e1_BM0LG2Euvw24kO-p@{V?%@>-Ew1aeHoq4u26hIQ=9l4 zwFyKhU4m#3fuW-=!85Hp7U!=!3My-thakPTtf9J~>aI{j0M&xj0Edw;C{o7us{@B7 zK!~QJBbb)ZaX9T3Bdk>MM-iMWpa#e8iL6|f&giZT`*&^u{gv)xNFWcz2+K6ZQkGri zC13G|n!EhW(9WZz!LZ%BqwGr?y#R^}^EA?w2wl(-?&+z6Ny>*V2$H1*kx+ZIZW+|r z_VAZ(D_@MfQw}W??+|%~^k7({9@4&g{tE@`5ps9sMJ0D_2(@RK7%*w7$V`eX)D)>e zid5xG5s}jMR|hfa5HZX_2ssjphU<`o4FM*vNwS4V8<9*#!c_$8pqHCcFZIyAtkQO2 zOBg->?hS!@$(NdugI(P%$QyxR{rZ45oLWXrA*RVATEpba)Q)N!TG8gUK>7wt%AdvP*p)NtIFeo;SZH=ANpqFmBX@RiDjd;U zk*5lPT5iKuk<+qC3||;TZpR=%#`A%|`qgWKm#o1tJZw}my)FBMD90A6=8i*fxxm5Y zf;?P8Etks$`MI2FF!9{kUckO^0dc-`O><{+hn?`P5%k!5qVs!VErErt-O(Gc1FjCq z5onDtm73SDXIdeN8tPCoI3R;;c{g@7YzW3e>*|(Yi@i*)5tco*@a(eId`F_q&y#3S z%T2WT`4g?0{h2&iCz~LpbF4G_ zg*E=o;;+2$#^;rKI%6HC4K9|+0~ERXHNmA3yWT9Ad0Ly(l?~-+EUP)YT7}cms zq&p&NW>_eO5Fq?6#*+?LVCAw)JL(%Q4hB#`1sdw5fhfR$&X?YmqnDL87#K}SFqRc& zFeGO!=6q(JBtWtW@5>95_ve|{P>_(fX3lV{3pKQPG91Dp!*zGzv2)#u7gJbfT(-!Q zSatwc7G?nRR@kOuvrCoM2s2E;IcE1zbN)RR+R6`S*(lR#Pap1!_P(8c6I~$4nL8-Y z*8wWe=>TQ-W^8YqLObcPt-DW`=WvXyu95ASW0zm9TdqO7b7=H;NR+Jpi60rEpjYoQ zeSmy5)w3$yOzwhd>YwXMs+4ck$e41GX0ul6CdbhBh0RP=?L9#`hT6IS751EEeCIRg*wBAv7BnyQNQBUk-t@&kP+TFGM+Rn z*3z33oUFFTH&dXq1tw=S=7^i6wxLXxGc8mmXbWy=!!ojS87k6||M&_t6*;R*IaBy+ zf|w6k<40$>o~v{+!pz^`I{D0*1GO8{teBb6RCdhFWGoZr+%`EUTGCPHPArEjIUc>R zm0niMUjE`~*$;bkpPj#6*hDyJhs8PQC`6|7kKD64%~vvyV82}Z_Al|ZVV$?E)7IeS z5n`DMH6eCaseWfIe`K0k=Gm7*rysI*n6EUwe$6SA+mMxlobhkI;csq}osRviic%Mi z?RC#1CY`eox6^dsn7O)(S8H!T){a+VhDn03K!r(gtd-mVBx)5#cpyu$JWCwL7XQg3 zY_Lj%4LVTM2AUuogjsE8DSJ`HogcMedTmH$DbkV1*=;UQ`FgwBug5J>xB^Nd&+nAH zpn(}84gwj_Er)Rm^H5GBC=wjHMkUf#4Go;p#1B!(wE@PZ-x--N@k*M9DoVuYww9pA z^dSUX@)xeZ|IC{ldS}Ei3xy|X0t$<#tS&R-?D@_AXZMh@>eat{(&q_M?$lWUGu%|b zylp68-ueodx0ZtCjdjwCzszR_ozH75owIr=$@TIWmqbukQ_Y=pL2T5<4c~I{M>b`*qyF@I0cmVI?vqYJi4SloNd!jh$Hwm?~FF zELNBf60@rtE(zlQ^67Ov*Q}SDf83J33(Za}ZhgeI)~v%k?4~Oc4X>{|NHPQhYl1u| zpAs$F70R@F5M_RDR?8Qf8oMckJ)F(qV5k+7d?r&I+^O|BY*ZPyI>N27V6+=oOQT4l zJTjYQnuT;pg%ERAVdI`PF&1rK*Y?~TD3H06tUjMiYy%N9<7yO;|a6#pq{J}o_P}c^rSY^bV z3NW>+IzxT9RSXx*7}>A_yQ9I4`HdPMhdeBR=BGa<$=Oor6>;+?kYujprBSMWE=LOj;+eshK(bxUf6Be zhV_?YDQ14xGdK&l7|(F-VWw-ETZ|#N`2rVXGW}vK#?O`S0xHJJ`pdEyrw;|q zF2AhJ=59AE9NswDc8GK@lsep*UH#=u`Cz!GITQ)EFb5RcDNvcBWq8FU+TnHK zXkDM_?iSXO^LZlIW_Uy4GwhSx(Av=rL6d@-#>s3b*u`yg_`{S@b!=w ziUnmwy}kV)M(2mSThX(>XvvabxUV@JfgP^f`*KvOa#X5wRQ$Qt{0nkS&CfM8KaZ)( z9E%HbEiTBlxL{$f#a!lbYqdMf6}diE! z`Knwds&jqx`*P#Z@5^PwU!G4b*H{1iJm&JGrhk4uySY*6Ur><~et$)7BKs>Un5ms- z)fP^bldq2zsB8=OwYSO=( JywMTA{}1@W;O+nb diff --git a/doc/examples/at__object_t_key_type.cpp b/doc/examples/at__object_t_key_type.cpp index a66bd9a3..7797418f 100644 --- a/doc/examples/at__object_t_key_type.cpp +++ b/doc/examples/at__object_t_key_type.cpp @@ -21,9 +21,23 @@ int main() // output changed array std::cout << object << '\n'; - // try to write at a nonexisting key + + // exception type_error.304 try { + // use at() on a non-object type + json str = "I am a string"; + str.at("the good") = "Another string"; + } + catch (json::type_error& e) + { + std::cout << e.what() << '\n'; + } + + // exception out_of_range.401 + try + { + // try to write at a nonexisting key object.at("the fast") = "il rapido"; } catch (json::out_of_range& e) diff --git a/doc/examples/at__object_t_key_type.link b/doc/examples/at__object_t_key_type.link index 1cb99d46..8874783b 100644 --- a/doc/examples/at__object_t_key_type.link +++ b/doc/examples/at__object_t_key_type.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/at__object_t_key_type.output b/doc/examples/at__object_t_key_type.output index 654b9eb6..b544b729 100644 --- a/doc/examples/at__object_t_key_type.output +++ b/doc/examples/at__object_t_key_type.output @@ -1,3 +1,4 @@ "il brutto" {"the bad":"il cattivo","the good":"il buono","the ugly":"il brutto"} +[json.exception.type_error.304] cannot use at() with string [json.exception.out_of_range.403] key 'the fast' not found diff --git a/doc/examples/at__object_t_key_type_const.cpp b/doc/examples/at__object_t_key_type_const.cpp index 99eb6f01..f6042404 100644 --- a/doc/examples/at__object_t_key_type_const.cpp +++ b/doc/examples/at__object_t_key_type_const.cpp @@ -15,9 +15,23 @@ int main() // output element with key "the ugly" std::cout << object.at("the ugly") << '\n'; - // try to read from a nonexisting key + + // exception type_error.304 try { + // use at() on a non-object type + const json str = "I am a string"; + std::cout << str.at("the good") << '\n'; + } + catch (json::type_error& e) + { + std::cout << e.what() << '\n'; + } + + // exception out_of_range.401 + try + { + // try to read from a nonexisting key std::cout << object.at("the fast") << '\n'; } catch (json::out_of_range) diff --git a/doc/examples/at__object_t_key_type_const.link b/doc/examples/at__object_t_key_type_const.link index a07dbd59..cd8594e6 100644 --- a/doc/examples/at__object_t_key_type_const.link +++ b/doc/examples/at__object_t_key_type_const.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/at__object_t_key_type_const.output b/doc/examples/at__object_t_key_type_const.output index b3dd11d3..40ca3f09 100644 --- a/doc/examples/at__object_t_key_type_const.output +++ b/doc/examples/at__object_t_key_type_const.output @@ -1,2 +1,3 @@ "il brutto" +[json.exception.type_error.304] cannot use at() with string out of range diff --git a/doc/examples/at__size_type.cpp b/doc/examples/at__size_type.cpp index 07e363ab..28def1dd 100644 --- a/doc/examples/at__size_type.cpp +++ b/doc/examples/at__size_type.cpp @@ -16,9 +16,23 @@ int main() // output changed array std::cout << array << '\n'; - // try to write beyond the array limit + + // exception type_error.304 try { + // use at() on a non-array type + json str = "I am a string"; + str.at(0) = "Another string"; + } + catch (json::type_error& e) + { + std::cout << e.what() << '\n'; + } + + // exception out_of_range.401 + try + { + // try to write beyond the array limit array.at(5) = "sixth"; } catch (json::out_of_range& e) diff --git a/doc/examples/at__size_type.link b/doc/examples/at__size_type.link index 78ec0ca0..cba0fa00 100644 --- a/doc/examples/at__size_type.link +++ b/doc/examples/at__size_type.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/at__size_type.output b/doc/examples/at__size_type.output index 4f0c9e54..54026436 100644 --- a/doc/examples/at__size_type.output +++ b/doc/examples/at__size_type.output @@ -1,3 +1,4 @@ "third" ["first","second","third","fourth"] +[json.exception.type_error.304] cannot use at() with string [json.exception.out_of_range.401] array index 5 is out of range diff --git a/doc/examples/at__size_type_const.cpp b/doc/examples/at__size_type_const.cpp index 88d28be6..213d22fd 100644 --- a/doc/examples/at__size_type_const.cpp +++ b/doc/examples/at__size_type_const.cpp @@ -5,17 +5,31 @@ using json = nlohmann::json; int main() { // create JSON array - json array = {"first", "2nd", "third", "fourth"}; + const json array = {"first", "2nd", "third", "fourth"}; // output element at index 2 (third element) std::cout << array.at(2) << '\n'; - // try to read beyond the array limit + + // exception type_error.304 try { + // use at() on a non-array type + const json str = "I am a string"; + std::cout << str.at(0) << '\n'; + } + catch (json::type_error& e) + { + std::cout << e.what() << '\n'; + } + + // exception out_of_range.401 + try + { + // try to read beyond the array limit std::cout << array.at(5) << '\n'; } - catch (const json::out_of_range& e) + catch (json::out_of_range& e) { std::cout << e.what() << '\n'; } diff --git a/doc/examples/at__size_type_const.link b/doc/examples/at__size_type_const.link index c703c5d9..ce3647ac 100644 --- a/doc/examples/at__size_type_const.link +++ b/doc/examples/at__size_type_const.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/at__size_type_const.output b/doc/examples/at__size_type_const.output index e52b6cc0..8135a27a 100644 --- a/doc/examples/at__size_type_const.output +++ b/doc/examples/at__size_type_const.output @@ -1,2 +1,3 @@ "third" +[json.exception.type_error.304] cannot use at() with string [json.exception.out_of_range.401] array index 5 is out of range diff --git a/doc/examples/at_json_pointer.cpp b/doc/examples/at_json_pointer.cpp index 6d1617e6..3ef91282 100644 --- a/doc/examples/at_json_pointer.cpp +++ b/doc/examples/at_json_pointer.cpp @@ -33,10 +33,56 @@ int main() // output the changed array std::cout << j["array"] << '\n'; - // try to use an invalid JSON pointer + + // out_of_range.106 try { - auto ref = j.at("/number/foo"_json_pointer); + // try to use an array index with leading '0' + json::reference ref = j.at("/array/01"_json_pointer); + } + catch (json::parse_error& e) + { + std::cout << e.what() << '\n'; + } + + // out_of_range.109 + try + { + // try to use an array index that is not a number + json::reference ref = j.at("/array/one"_json_pointer); + } + catch (json::parse_error& e) + { + std::cout << e.what() << '\n'; + } + + // out_of_range.401 + try + { + // try to use a an invalid array index + json::reference ref = j.at("/array/4"_json_pointer); + } + catch (json::out_of_range& e) + { + std::cout << e.what() << '\n'; + } + + // out_of_range.402 + try + { + // try to use the array index '-' + json::reference ref = j.at("/array/-"_json_pointer); + } + catch (json::out_of_range& e) + { + std::cout << e.what() << '\n'; + } + + // out_of_range.404 + try + { + // try to use a JSON pointer that cannot be resolved + json::reference ref = j.at("/number/foo"_json_pointer); } catch (json::out_of_range& e) { diff --git a/doc/examples/at_json_pointer.link b/doc/examples/at_json_pointer.link index 45402303..c8563ec2 100644 --- a/doc/examples/at_json_pointer.link +++ b/doc/examples/at_json_pointer.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/at_json_pointer.output b/doc/examples/at_json_pointer.output index a45737ad..505792f2 100644 --- a/doc/examples/at_json_pointer.output +++ b/doc/examples/at_json_pointer.output @@ -4,4 +4,8 @@ 2 "bar" [1,21] +[json.exception.parse_error.106] parse error: array index '01' must not begin with '0' +[json.exception.parse_error.109] parse error: array index 'one' is not a number +[json.exception.out_of_range.401] array index 4 is out of range +[json.exception.out_of_range.402] array index '-' (2) is out of range [json.exception.out_of_range.404] unresolved reference token 'foo' diff --git a/doc/examples/at_json_pointer_const.cpp b/doc/examples/at_json_pointer_const.cpp index dab1b39c..a1d065f2 100644 --- a/doc/examples/at_json_pointer_const.cpp +++ b/doc/examples/at_json_pointer_const.cpp @@ -5,7 +5,7 @@ using json = nlohmann::json; int main() { // create a JSON value - json j = + const json j = { {"number", 1}, {"string", "foo"}, {"array", {1, 2}} }; @@ -21,10 +21,44 @@ int main() // output element with JSON pointer "/array/1" std::cout << j.at("/array/1"_json_pointer) << '\n'; - // try to use an invalid JSON pointer + // out_of_range.109 try { - auto ref = j.at("/number/foo"_json_pointer); + // try to use an array index that is not a number + json::const_reference ref = j.at("/array/one"_json_pointer); + } + catch (json::parse_error& e) + { + std::cout << e.what() << '\n'; + } + + // out_of_range.401 + try + { + // try to use a an invalid array index + json::const_reference ref = j.at("/array/4"_json_pointer); + } + catch (json::out_of_range& e) + { + std::cout << e.what() << '\n'; + } + + // out_of_range.402 + try + { + // try to use the array index '-' + json::const_reference ref = j.at("/array/-"_json_pointer); + } + catch (json::out_of_range& e) + { + std::cout << e.what() << '\n'; + } + + // out_of_range.404 + try + { + // try to use a JSON pointer that cannot be resolved + json::const_reference ref = j.at("/number/foo"_json_pointer); } catch (json::out_of_range& e) { diff --git a/doc/examples/at_json_pointer_const.link b/doc/examples/at_json_pointer_const.link index 70e7cf86..f421faf4 100644 --- a/doc/examples/at_json_pointer_const.link +++ b/doc/examples/at_json_pointer_const.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/at_json_pointer_const.output b/doc/examples/at_json_pointer_const.output index 21712e86..b3361f04 100644 --- a/doc/examples/at_json_pointer_const.output +++ b/doc/examples/at_json_pointer_const.output @@ -2,4 +2,7 @@ "foo" [1,2] 2 +[json.exception.parse_error.109] parse error: array index 'one' is not a number +[json.exception.out_of_range.401] array index 4 is out of range +[json.exception.out_of_range.402] array index '-' (2) is out of range [json.exception.out_of_range.404] unresolved reference token 'foo' diff --git a/src/json.hpp b/src/json.hpp index 605d6aa5..7f31377f 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -3723,16 +3723,20 @@ class basic_json @return reference to the element at index @a idx @throw type_error.304 if the JSON value is not an array; in this case, - calling `at` with an index makes no sense. + calling `at` with an index makes no sense. See example below. @throw out_of_range.401 if the index @a idx is out of range of the array; - that is, `idx >= size()`; see example below. + that is, `idx >= size()`. See example below. + + @exceptionsafety Strong guarantee: if an exception is thrown, there are no + changes in the JSON value. @complexity Constant. - @liveexample{The example below shows how array elements can be read and - written using `at()`.,at__size_type} - @since version 1.0.0 + + @liveexample{The example below shows how array elements can be read and + written using `at()`. It also demonstrates the different exceptions that + can be thrown.,at__size_type} */ reference at(size_type idx) { @@ -3766,16 +3770,20 @@ class basic_json @return const reference to the element at index @a idx @throw type_error.304 if the JSON value is not an array; in this case, - calling `at` with an index makes no sense. + calling `at` with an index makes no sense. See example below. @throw out_of_range.401 if the index @a idx is out of range of the array; - that is, `idx >= size()`; see example below. + that is, `idx >= size()`. See example below. + + @exceptionsafety Strong guarantee: if an exception is thrown, there are no + changes in the JSON value. @complexity Constant. - @liveexample{The example below shows how array elements can be read using - `at()`.,at__size_type_const} - @since version 1.0.0 + + @liveexample{The example below shows how array elements can be read using + `at()`. It also demonstrates the different exceptions that can be thrown., + at__size_type_const} */ const_reference at(size_type idx) const { @@ -3809,20 +3817,24 @@ class basic_json @return reference to the element at key @a key @throw type_error.304 if the JSON value is not an object; in this case, - calling `at` with a key makes no sense. + calling `at` with a key makes no sense. See example below. @throw out_of_range.403 if the key @a key is is not stored in the object; - that is, `find(key) == end()`; see example below. + that is, `find(key) == end()`. See example below. + + @exceptionsafety Strong guarantee: if an exception is thrown, there are no + changes in the JSON value. @complexity Logarithmic in the size of the container. - @liveexample{The example below shows how object elements can be read and - written using `at()`.,at__object_t_key_type} - @sa @ref operator[](const typename object_t::key_type&) for unchecked access by reference @sa @ref value() for access by value with a default value @since version 1.0.0 + + @liveexample{The example below shows how object elements can be read and + written using `at()`. It also demonstrates the different exceptions that + can be thrown.,at__object_t_key_type} */ reference at(const typename object_t::key_type& key) { @@ -3856,20 +3868,24 @@ class basic_json @return const reference to the element at key @a key @throw type_error.304 if the JSON value is not an object; in this case, - calling `at` with a key makes no sense. + calling `at` with a key makes no sense. See example below. @throw out_of_range.403 if the key @a key is is not stored in the object; - that is, `find(key) == end()`; see example below. + that is, `find(key) == end()`. See example below. + + @exceptionsafety Strong guarantee: if an exception is thrown, there are no + changes in the JSON value. @complexity Logarithmic in the size of the container. - @liveexample{The example below shows how object elements can be read using - `at()`.,at__object_t_key_type_const} - @sa @ref operator[](const typename object_t::key_type&) for unchecked access by reference @sa @ref value() for access by value with a default value @since version 1.0.0 + + @liveexample{The example below shows how object elements can be read using + `at()`. It also demonstrates the different exceptions that can be thrown., + at__object_t_key_type_const} */ const_reference at(const typename object_t::key_type& key) const { @@ -12791,24 +12807,30 @@ basic_json_parser_74: @return reference to the element pointed to by @a ptr - @complexity Constant. - @throw parse_error.106 if an array index in the passed JSON pointer @a ptr - begins with '0' + begins with '0'. See example below. @throw parse_error.109 if an array index in the passed JSON pointer @a ptr - is not a number + is not a number. See example below. - @throw out_of_range.402 if the array index `-` is used in the passed JSON + @throw out_of_range.401 if an array index in the passed JSON pointer @a ptr + is out of range. See example below. + + @throw out_of_range.402 if the array index '-' is used in the passed JSON pointer @a ptr. As `at` provides checked access (and no elements are - implicitly inserted), the index `-` is always invalid. + implicitly inserted), the index '-' is always invalid. See example below. - @throw out_of_range.404 if the JSON pointer @a ptr can not be resolved; - see example below. + @throw out_of_range.404 if the JSON pointer @a ptr can not be resolved. + See example below. - @liveexample{The behavior is shown in the example.,at_json_pointer} + @exceptionsafety Strong guarantee: if an exception is thrown, there are no + changes in the JSON value. + + @complexity Constant. @since version 2.0.0 + + @liveexample{The behavior is shown in the example.,at_json_pointer} */ reference at(const json_pointer& ptr) { @@ -12825,24 +12847,30 @@ basic_json_parser_74: @return reference to the element pointed to by @a ptr - @complexity Constant. - @throw parse_error.106 if an array index in the passed JSON pointer @a ptr - begins with '0' + begins with '0'. See example below. @throw parse_error.109 if an array index in the passed JSON pointer @a ptr - is not a number + is not a number. See example below. - @throw out_of_range.402 if the array index `-` is used in the passed JSON + @throw out_of_range.401 if an array index in the passed JSON pointer @a ptr + is out of range. See example below. + + @throw out_of_range.402 if the array index '-' is used in the passed JSON pointer @a ptr. As `at` provides checked access (and no elements are - implicitly inserted), the index `-` is always invalid. + implicitly inserted), the index '-' is always invalid. See example below. - @throw out_of_range.404 if the JSON pointer @a ptr can not be resolved; - see example below. + @throw out_of_range.404 if the JSON pointer @a ptr can not be resolved. + See example below. - @liveexample{The behavior is shown in the example.,at_json_pointer_const} + @exceptionsafety Strong guarantee: if an exception is thrown, there are no + changes in the JSON value. + + @complexity Constant. @since version 2.0.0 + + @liveexample{The behavior is shown in the example.,at_json_pointer_const} */ const_reference at(const json_pointer& ptr) const { diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 2bd2fd76..48111521 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -3723,16 +3723,20 @@ class basic_json @return reference to the element at index @a idx @throw type_error.304 if the JSON value is not an array; in this case, - calling `at` with an index makes no sense. + calling `at` with an index makes no sense. See example below. @throw out_of_range.401 if the index @a idx is out of range of the array; - that is, `idx >= size()`; see example below. + that is, `idx >= size()`. See example below. + + @exceptionsafety Strong guarantee: if an exception is thrown, there are no + changes in the JSON value. @complexity Constant. - @liveexample{The example below shows how array elements can be read and - written using `at()`.,at__size_type} - @since version 1.0.0 + + @liveexample{The example below shows how array elements can be read and + written using `at()`. It also demonstrates the different exceptions that + can be thrown.,at__size_type} */ reference at(size_type idx) { @@ -3766,16 +3770,20 @@ class basic_json @return const reference to the element at index @a idx @throw type_error.304 if the JSON value is not an array; in this case, - calling `at` with an index makes no sense. + calling `at` with an index makes no sense. See example below. @throw out_of_range.401 if the index @a idx is out of range of the array; - that is, `idx >= size()`; see example below. + that is, `idx >= size()`. See example below. + + @exceptionsafety Strong guarantee: if an exception is thrown, there are no + changes in the JSON value. @complexity Constant. - @liveexample{The example below shows how array elements can be read using - `at()`.,at__size_type_const} - @since version 1.0.0 + + @liveexample{The example below shows how array elements can be read using + `at()`. It also demonstrates the different exceptions that can be thrown., + at__size_type_const} */ const_reference at(size_type idx) const { @@ -3809,20 +3817,24 @@ class basic_json @return reference to the element at key @a key @throw type_error.304 if the JSON value is not an object; in this case, - calling `at` with a key makes no sense. + calling `at` with a key makes no sense. See example below. @throw out_of_range.403 if the key @a key is is not stored in the object; - that is, `find(key) == end()`; see example below. + that is, `find(key) == end()`. See example below. + + @exceptionsafety Strong guarantee: if an exception is thrown, there are no + changes in the JSON value. @complexity Logarithmic in the size of the container. - @liveexample{The example below shows how object elements can be read and - written using `at()`.,at__object_t_key_type} - @sa @ref operator[](const typename object_t::key_type&) for unchecked access by reference @sa @ref value() for access by value with a default value @since version 1.0.0 + + @liveexample{The example below shows how object elements can be read and + written using `at()`. It also demonstrates the different exceptions that + can be thrown.,at__object_t_key_type} */ reference at(const typename object_t::key_type& key) { @@ -3856,20 +3868,24 @@ class basic_json @return const reference to the element at key @a key @throw type_error.304 if the JSON value is not an object; in this case, - calling `at` with a key makes no sense. + calling `at` with a key makes no sense. See example below. @throw out_of_range.403 if the key @a key is is not stored in the object; - that is, `find(key) == end()`; see example below. + that is, `find(key) == end()`. See example below. + + @exceptionsafety Strong guarantee: if an exception is thrown, there are no + changes in the JSON value. @complexity Logarithmic in the size of the container. - @liveexample{The example below shows how object elements can be read using - `at()`.,at__object_t_key_type_const} - @sa @ref operator[](const typename object_t::key_type&) for unchecked access by reference @sa @ref value() for access by value with a default value @since version 1.0.0 + + @liveexample{The example below shows how object elements can be read using + `at()`. It also demonstrates the different exceptions that can be thrown., + at__object_t_key_type_const} */ const_reference at(const typename object_t::key_type& key) const { @@ -11824,24 +11840,30 @@ class basic_json @return reference to the element pointed to by @a ptr - @complexity Constant. - @throw parse_error.106 if an array index in the passed JSON pointer @a ptr - begins with '0' + begins with '0'. See example below. @throw parse_error.109 if an array index in the passed JSON pointer @a ptr - is not a number + is not a number. See example below. - @throw out_of_range.402 if the array index `-` is used in the passed JSON + @throw out_of_range.401 if an array index in the passed JSON pointer @a ptr + is out of range. See example below. + + @throw out_of_range.402 if the array index '-' is used in the passed JSON pointer @a ptr. As `at` provides checked access (and no elements are - implicitly inserted), the index `-` is always invalid. + implicitly inserted), the index '-' is always invalid. See example below. - @throw out_of_range.404 if the JSON pointer @a ptr can not be resolved; - see example below. + @throw out_of_range.404 if the JSON pointer @a ptr can not be resolved. + See example below. - @liveexample{The behavior is shown in the example.,at_json_pointer} + @exceptionsafety Strong guarantee: if an exception is thrown, there are no + changes in the JSON value. + + @complexity Constant. @since version 2.0.0 + + @liveexample{The behavior is shown in the example.,at_json_pointer} */ reference at(const json_pointer& ptr) { @@ -11858,24 +11880,30 @@ class basic_json @return reference to the element pointed to by @a ptr - @complexity Constant. - @throw parse_error.106 if an array index in the passed JSON pointer @a ptr - begins with '0' + begins with '0'. See example below. @throw parse_error.109 if an array index in the passed JSON pointer @a ptr - is not a number + is not a number. See example below. - @throw out_of_range.402 if the array index `-` is used in the passed JSON + @throw out_of_range.401 if an array index in the passed JSON pointer @a ptr + is out of range. See example below. + + @throw out_of_range.402 if the array index '-' is used in the passed JSON pointer @a ptr. As `at` provides checked access (and no elements are - implicitly inserted), the index `-` is always invalid. + implicitly inserted), the index '-' is always invalid. See example below. - @throw out_of_range.404 if the JSON pointer @a ptr can not be resolved; - see example below. + @throw out_of_range.404 if the JSON pointer @a ptr can not be resolved. + See example below. - @liveexample{The behavior is shown in the example.,at_json_pointer_const} + @exceptionsafety Strong guarantee: if an exception is thrown, there are no + changes in the JSON value. + + @complexity Constant. @since version 2.0.0 + + @liveexample{The behavior is shown in the example.,at_json_pointer_const} */ const_reference at(const json_pointer& ptr) const { From 80dcf22fc33897ed027d7bd4e687c48bf9148280 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 12 Mar 2017 14:39:20 +0100 Subject: [PATCH 073/530] :hammer: added a check function for MsgPack strings #497 To avoid the error described in #497, I added a function msgpack_expect_string that is executed every time a string is expected during the parsing of a map. In case the current byte does not belong to a MsgPack string, an exception is thrown. --- src/json.hpp | 32 ++++++++++++++++++++++++++++++++ src/json.hpp.re2c | 32 ++++++++++++++++++++++++++++++++ 2 files changed, 64 insertions(+) diff --git a/src/json.hpp b/src/json.hpp index f7b832b5..5fdf4852 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -7748,6 +7748,35 @@ class basic_json } } + /*! + @brief check if the next byte belongs to a string + + While parsing a map, the keys must be strings. This function checks if the + current byte is one of the start bytes for a string in MessagePack: + + - 0xa0 - 0xbf: fixstr + - 0xd9: str 8 + - 0xda: str 16 + - 0xdb: str 32 + + @param[in] v MessagePack serialization + @param[in] idx byte index in @a v to check for a string + + @throw std::invalid_argument if `v[idx]` does not belong to a string + */ + static void msgpack_expect_string(const std::vector& v, size_t idx) + { + check_length(v.size(), 1, idx); + + const auto byte = v[idx]; + if ((byte >= 0xa0 and byte <= 0xbf) or (byte >= 0xd9 and byte <= 0xdb)) + { + return; + } + + JSON_THROW(std::invalid_argument("error parsing a msgpack string @ " + std::to_string(idx) + ": " + std::to_string(static_cast(v[idx])))); + } + /*! @brief create a JSON value from a given MessagePack vector @@ -7782,6 +7811,7 @@ class basic_json const size_t len = v[current_idx] & 0x0f; for (size_t i = 0; i < len; ++i) { + msgpack_expect_string(v, idx); std::string key = from_msgpack_internal(v, idx); result[key] = from_msgpack_internal(v, idx); } @@ -7959,6 +7989,7 @@ class basic_json idx += 2; // skip 2 size bytes for (size_t i = 0; i < len; ++i) { + msgpack_expect_string(v, idx); std::string key = from_msgpack_internal(v, idx); result[key] = from_msgpack_internal(v, idx); } @@ -7972,6 +8003,7 @@ class basic_json idx += 4; // skip 4 size bytes for (size_t i = 0; i < len; ++i) { + msgpack_expect_string(v, idx); std::string key = from_msgpack_internal(v, idx); result[key] = from_msgpack_internal(v, idx); } diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 04de67cf..b3a39d64 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -7748,6 +7748,35 @@ class basic_json } } + /*! + @brief check if the next byte belongs to a string + + While parsing a map, the keys must be strings. This function checks if the + current byte is one of the start bytes for a string in MessagePack: + + - 0xa0 - 0xbf: fixstr + - 0xd9: str 8 + - 0xda: str 16 + - 0xdb: str 32 + + @param[in] v MessagePack serialization + @param[in] idx byte index in @a v to check for a string + + @throw std::invalid_argument if `v[idx]` does not belong to a string + */ + static void msgpack_expect_string(const std::vector& v, size_t idx) + { + check_length(v.size(), 1, idx); + + const auto byte = v[idx]; + if ((byte >= 0xa0 and byte <= 0xbf) or (byte >= 0xd9 and byte <= 0xdb)) + { + return; + } + + JSON_THROW(std::invalid_argument("error parsing a msgpack string @ " + std::to_string(idx) + ": " + std::to_string(static_cast(v[idx])))); + } + /*! @brief create a JSON value from a given MessagePack vector @@ -7782,6 +7811,7 @@ class basic_json const size_t len = v[current_idx] & 0x0f; for (size_t i = 0; i < len; ++i) { + msgpack_expect_string(v, idx); std::string key = from_msgpack_internal(v, idx); result[key] = from_msgpack_internal(v, idx); } @@ -7959,6 +7989,7 @@ class basic_json idx += 2; // skip 2 size bytes for (size_t i = 0; i < len; ++i) { + msgpack_expect_string(v, idx); std::string key = from_msgpack_internal(v, idx); result[key] = from_msgpack_internal(v, idx); } @@ -7972,6 +8003,7 @@ class basic_json idx += 4; // skip 4 size bytes for (size_t i = 0; i < len; ++i) { + msgpack_expect_string(v, idx); std::string key = from_msgpack_internal(v, idx); result[key] = from_msgpack_internal(v, idx); } From 87eafd8d6af591254646b53265c21f8d7b8d28e8 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 12 Mar 2017 15:20:17 +0100 Subject: [PATCH 074/530] :white_check_mark: added regression tests for #473 These tests currently pass without any adjustments to the source code. --- test/src/unit-regression.cpp | 50 ++++++++++++++++++++++++++++++++++++ 1 file changed, 50 insertions(+) diff --git a/test/src/unit-regression.cpp b/test/src/unit-regression.cpp index 957d5a25..7b4086b1 100644 --- a/test/src/unit-regression.cpp +++ b/test/src/unit-regression.cpp @@ -32,6 +32,7 @@ SOFTWARE. using nlohmann::json; #include +#include TEST_CASE("regression tests") { @@ -796,6 +797,55 @@ TEST_CASE("regression tests") CHECK(s1 == s2); } + SECTION("issue #473 - inconsistent behavior in conversion to array type") + { + json j_array = {1, 2, 3, 4}; + json j_number = 42; + json j_null = nullptr; + + SECTION("std::vector") + { + auto create = [](const json & j) + { + std::vector v = j; + }; + + CHECK_NOTHROW(create(j_array)); + CHECK_THROWS_AS(create(j_number), std::domain_error); + CHECK_THROWS_WITH(create(j_number), "type must be array, but is number"); + CHECK_THROWS_AS(create(j_null), std::domain_error); + CHECK_THROWS_WITH(create(j_null), "type must be array, but is null"); + } + + SECTION("std::list") + { + auto create = [](const json & j) + { + std::list v = j; + }; + + CHECK_NOTHROW(create(j_array)); + CHECK_THROWS_AS(create(j_number), std::domain_error); + CHECK_THROWS_WITH(create(j_number), "type must be array, but is number"); + CHECK_THROWS_AS(create(j_null), std::domain_error); + CHECK_THROWS_WITH(create(j_null), "type must be array, but is null"); + } + + SECTION("std::forward_list") + { + auto create = [](const json & j) + { + std::forward_list v = j; + }; + + CHECK_NOTHROW(create(j_array)); + CHECK_THROWS_AS(create(j_number), std::domain_error); + CHECK_THROWS_WITH(create(j_number), "type must be array, but is number"); + CHECK_THROWS_AS(create(j_null), std::domain_error); + CHECK_THROWS_WITH(create(j_null), "type must be array, but is null"); + } + } + SECTION("issue #486 - json::value_t can't be a map's key type in VC++ 2015") { // the code below must compile with MSVC From 9355f0588817683d918b35c6366a9193a3e9e331 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 12 Mar 2017 17:05:02 +0100 Subject: [PATCH 075/530] :hammer: cleaned up array from_json methods #473 Removed some code that is not needed any more. Thus, streamlining the array from_json methods. --- src/json.hpp | 30 ++++++++---------------------- src/json.hpp.re2c | 30 ++++++++---------------------- 2 files changed, 16 insertions(+), 44 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 5fdf4852..ebca577f 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -700,22 +700,15 @@ void from_json(const BasicJsonType& j, typename BasicJsonType::array_t& arr) } // forward_list doesn't have an insert method -template +template::value, int> = 0> void from_json(const BasicJsonType& j, std::forward_list& l) { - // do not perform the check when user wants to retrieve jsons - // (except when it's null.. ?) - if (j.is_null()) + if (not j.is_array()) { JSON_THROW(std::domain_error("type must be array, but is " + j.type_name())); } - if (not std::is_same::value) - { - if (not j.is_array()) - { - JSON_THROW(std::domain_error("type must be array, but is " + j.type_name())); - } - } + for (auto it = j.rbegin(), end = j.rend(); it != end; ++it) { l.push_front(it->template get()); @@ -747,8 +740,8 @@ auto from_json_array_impl(const BasicJsonType& j, CompatibleArrayType& arr, prio using std::end; arr.reserve(j.size()); - std::transform( - j.begin(), j.end(), std::inserter(arr, end(arr)), [](const BasicJsonType & i) + std::transform(j.begin(), j.end(), + std::inserter(arr, end(arr)), [](const BasicJsonType & i) { // get() returns *this, this won't call a from_json // method when value_type is BasicJsonType @@ -758,22 +751,15 @@ auto from_json_array_impl(const BasicJsonType& j, CompatibleArrayType& arr, prio template::value and + std::is_convertible::value and not std::is_same::value, int> = 0> void from_json(const BasicJsonType& j, CompatibleArrayType& arr) { - if (j.is_null()) + if (not j.is_array()) { JSON_THROW(std::domain_error("type must be array, but is " + j.type_name())); } - // when T == BasicJsonType, do not check if value_t is correct - if (not std::is_same::value) - { - if (not j.is_array()) - { - JSON_THROW(std::domain_error("type must be array, but is " + j.type_name())); - } - } from_json_array_impl(j, arr, priority_tag<1> {}); } diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index b3a39d64..bdbd806d 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -700,22 +700,15 @@ void from_json(const BasicJsonType& j, typename BasicJsonType::array_t& arr) } // forward_list doesn't have an insert method -template +template::value, int> = 0> void from_json(const BasicJsonType& j, std::forward_list& l) { - // do not perform the check when user wants to retrieve jsons - // (except when it's null.. ?) - if (j.is_null()) + if (not j.is_array()) { JSON_THROW(std::domain_error("type must be array, but is " + j.type_name())); } - if (not std::is_same::value) - { - if (not j.is_array()) - { - JSON_THROW(std::domain_error("type must be array, but is " + j.type_name())); - } - } + for (auto it = j.rbegin(), end = j.rend(); it != end; ++it) { l.push_front(it->template get()); @@ -747,8 +740,8 @@ auto from_json_array_impl(const BasicJsonType& j, CompatibleArrayType& arr, prio using std::end; arr.reserve(j.size()); - std::transform( - j.begin(), j.end(), std::inserter(arr, end(arr)), [](const BasicJsonType & i) + std::transform(j.begin(), j.end(), + std::inserter(arr, end(arr)), [](const BasicJsonType & i) { // get() returns *this, this won't call a from_json // method when value_type is BasicJsonType @@ -758,22 +751,15 @@ auto from_json_array_impl(const BasicJsonType& j, CompatibleArrayType& arr, prio template::value and + std::is_convertible::value and not std::is_same::value, int> = 0> void from_json(const BasicJsonType& j, CompatibleArrayType& arr) { - if (j.is_null()) + if (not j.is_array()) { JSON_THROW(std::domain_error("type must be array, but is " + j.type_name())); } - // when T == BasicJsonType, do not check if value_t is correct - if (not std::is_same::value) - { - if (not j.is_array()) - { - JSON_THROW(std::domain_error("type must be array, but is " + j.type_name())); - } - } from_json_array_impl(j, arr, priority_tag<1> {}); } From 8feaf8dc94a88393510637af36a24163c95a1b45 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 12 Mar 2017 18:38:05 +0100 Subject: [PATCH 076/530] :boom: implemented new handling of NaN and INF #70 #329 #388 MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit - If an overflow occurs during parsing a number from a JSON text, an exception (std::out_of_range for the moment, to be replaced by a user-defined exception #244) is thrown so that the overflow is detected early and roundtripping is guaranteed. - NaN and INF floating-point values can be stored in a JSON value and are not replaced by null. That is, the basic_json class behaves like double in this regard (no exception occurs). However, NaN and INF are serialized to “null”. - Adjusted test cases appropriately. --- src/json.hpp | 24 +++++++++++------------ src/json.hpp.re2c | 24 +++++++++++------------ test/src/unit-cbor.cpp | 15 ++++++++++----- test/src/unit-class_parser.cpp | 4 +++- test/src/unit-constructor1.cpp | 12 +++++++++--- test/src/unit-msgpack.cpp | 6 +++--- test/src/unit-regression.cpp | 35 ++++++++++++++++++++++++++++++++-- test/src/unit-testsuites.cpp | 26 +++++++++++++++++++------ 8 files changed, 100 insertions(+), 46 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index ebca577f..42a07f87 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -263,16 +263,8 @@ struct external_constructor template static void construct(BasicJsonType& j, typename BasicJsonType::number_float_t val) noexcept { - // replace infinity and NAN by null - if (not std::isfinite(val)) - { - j = BasicJsonType{}; - } - else - { - j.m_type = value_t::number_float; - j.m_value = val; - } + j.m_type = value_t::number_float; + j.m_value = val; j.assert_invariant(); } }; @@ -6653,6 +6645,13 @@ class basic_json */ void dump_float(number_float_t x) { + // NaN / inf + if (not std::isfinite(x) or std::isnan(x)) + { + o.write("null", 4); + return; + } + // special case for 0.0 and -0.0 if (x == 0) { @@ -11425,11 +11424,10 @@ basic_json_parser_74: result.m_type = value_t::number_float; result.m_value = val; - // replace infinity and NAN by null + // throw in case of infinity or NAN if (not std::isfinite(result.m_value.number_float)) { - result.m_type = value_t::null; - result.m_value = basic_json::json_value(); + JSON_THROW(std::out_of_range("number overflow: " + get_token_string())); } return true; diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index bdbd806d..042e3ab7 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -263,16 +263,8 @@ struct external_constructor template static void construct(BasicJsonType& j, typename BasicJsonType::number_float_t val) noexcept { - // replace infinity and NAN by null - if (not std::isfinite(val)) - { - j = BasicJsonType{}; - } - else - { - j.m_type = value_t::number_float; - j.m_value = val; - } + j.m_type = value_t::number_float; + j.m_value = val; j.assert_invariant(); } }; @@ -6653,6 +6645,13 @@ class basic_json */ void dump_float(number_float_t x) { + // NaN / inf + if (not std::isfinite(x) or std::isnan(x)) + { + o.write("null", 4); + return; + } + // special case for 0.0 and -0.0 if (x == 0) { @@ -10459,11 +10458,10 @@ class basic_json result.m_type = value_t::number_float; result.m_value = val; - // replace infinity and NAN by null + // throw in case of infinity or NAN if (not std::isfinite(result.m_value.number_float)) { - result.m_type = value_t::null; - result.m_value = basic_json::json_value(); + JSON_THROW(std::out_of_range("number overflow: " + get_token_string())); } return true; diff --git a/test/src/unit-cbor.cpp b/test/src/unit-cbor.cpp index 84b280bc..d313727d 100644 --- a/test/src/unit-cbor.cpp +++ b/test/src/unit-cbor.cpp @@ -33,6 +33,7 @@ SOFTWARE. using nlohmann::json; #include +#include TEST_CASE("CBOR") { @@ -744,13 +745,17 @@ TEST_CASE("CBOR") SECTION("infinity") { json j = json::from_cbor(std::vector({0xf9, 0x7c, 0x00})); - CHECK(j == nullptr); + json::number_float_t d = j; + CHECK(not std::isfinite(d)); + CHECK(j.dump() == "null"); } SECTION("NaN") { json j = json::from_cbor(std::vector({0xf9, 0x7c, 0x01})); - CHECK(j == nullptr); + json::number_float_t d = j; + CHECK(std::isnan(d)); + CHECK(j.dump() == "null"); } } } @@ -1344,7 +1349,7 @@ TEST_CASE("CBOR roundtrips", "[hide]") "test/data/nst_json_testsuite/test_parsing/y_number_after_space.json", "test/data/nst_json_testsuite/test_parsing/y_number_double_close_to_zero.json", "test/data/nst_json_testsuite/test_parsing/y_number_double_huge_neg_exp.json", - "test/data/nst_json_testsuite/test_parsing/y_number_huge_exp.json", + //"test/data/nst_json_testsuite/test_parsing/y_number_huge_exp.json", "test/data/nst_json_testsuite/test_parsing/y_number_int_with_exp.json", "test/data/nst_json_testsuite/test_parsing/y_number_minus_zero.json", "test/data/nst_json_testsuite/test_parsing/y_number_negative_int.json", @@ -1356,9 +1361,9 @@ TEST_CASE("CBOR roundtrips", "[hide]") "test/data/nst_json_testsuite/test_parsing/y_number_real_exponent.json", "test/data/nst_json_testsuite/test_parsing/y_number_real_fraction_exponent.json", "test/data/nst_json_testsuite/test_parsing/y_number_real_neg_exp.json", - "test/data/nst_json_testsuite/test_parsing/y_number_real_neg_overflow.json", + //"test/data/nst_json_testsuite/test_parsing/y_number_real_neg_overflow.json", "test/data/nst_json_testsuite/test_parsing/y_number_real_pos_exponent.json", - "test/data/nst_json_testsuite/test_parsing/y_number_real_pos_overflow.json", + //"test/data/nst_json_testsuite/test_parsing/y_number_real_pos_overflow.json", "test/data/nst_json_testsuite/test_parsing/y_number_real_underflow.json", "test/data/nst_json_testsuite/test_parsing/y_number_simple_int.json", "test/data/nst_json_testsuite/test_parsing/y_number_simple_real.json", diff --git a/test/src/unit-class_parser.cpp b/test/src/unit-class_parser.cpp index e3ad3723..35aafe05 100644 --- a/test/src/unit-class_parser.cpp +++ b/test/src/unit-class_parser.cpp @@ -272,7 +272,9 @@ TEST_CASE("parser class") SECTION("overflow") { - CHECK(json::parser("1.18973e+4932").parse() == json()); + // overflows during parsing yield an exception + CHECK_THROWS_AS(json::parser("1.18973e+4932").parse() == json(), std::out_of_range); + CHECK_THROWS_WITH(json::parser("1.18973e+4932").parse() == json(), "number overflow: 1.18973e+4932"); } SECTION("invalid numbers") diff --git a/test/src/unit-constructor1.cpp b/test/src/unit-constructor1.cpp index 9363f0ba..3b23d78d 100644 --- a/test/src/unit-constructor1.cpp +++ b/test/src/unit-constructor1.cpp @@ -702,11 +702,17 @@ TEST_CASE("constructors") SECTION("infinity") { - // infinity is stored as null - // should change in the future: https://github.com/nlohmann/json/issues/388 + // infinity is stored properly, but serialized to null json::number_float_t n(std::numeric_limits::infinity()); json j(n); - CHECK(j.type() == json::value_t::null); + CHECK(j.type() == json::value_t::number_float); + + // check round trip of infinity + json::number_float_t d = j; + CHECK(d == n); + + // check that inf is serialized to null + CHECK(j.dump() == "null"); } } diff --git a/test/src/unit-msgpack.cpp b/test/src/unit-msgpack.cpp index 3a2ae5b1..90b4eb2a 100644 --- a/test/src/unit-msgpack.cpp +++ b/test/src/unit-msgpack.cpp @@ -1116,7 +1116,7 @@ TEST_CASE("MessagePack roundtrips", "[hide]") "test/data/nst_json_testsuite/test_parsing/y_number_after_space.json", "test/data/nst_json_testsuite/test_parsing/y_number_double_close_to_zero.json", "test/data/nst_json_testsuite/test_parsing/y_number_double_huge_neg_exp.json", - "test/data/nst_json_testsuite/test_parsing/y_number_huge_exp.json", + //"test/data/nst_json_testsuite/test_parsing/y_number_huge_exp.json", "test/data/nst_json_testsuite/test_parsing/y_number_int_with_exp.json", "test/data/nst_json_testsuite/test_parsing/y_number_minus_zero.json", "test/data/nst_json_testsuite/test_parsing/y_number_negative_int.json", @@ -1128,9 +1128,9 @@ TEST_CASE("MessagePack roundtrips", "[hide]") "test/data/nst_json_testsuite/test_parsing/y_number_real_exponent.json", "test/data/nst_json_testsuite/test_parsing/y_number_real_fraction_exponent.json", "test/data/nst_json_testsuite/test_parsing/y_number_real_neg_exp.json", - "test/data/nst_json_testsuite/test_parsing/y_number_real_neg_overflow.json", + //"test/data/nst_json_testsuite/test_parsing/y_number_real_neg_overflow.json", "test/data/nst_json_testsuite/test_parsing/y_number_real_pos_exponent.json", - "test/data/nst_json_testsuite/test_parsing/y_number_real_pos_overflow.json", + //"test/data/nst_json_testsuite/test_parsing/y_number_real_pos_overflow.json", "test/data/nst_json_testsuite/test_parsing/y_number_real_underflow.json", "test/data/nst_json_testsuite/test_parsing/y_number_simple_int.json", "test/data/nst_json_testsuite/test_parsing/y_number_simple_real.json", diff --git a/test/src/unit-regression.cpp b/test/src/unit-regression.cpp index 7b4086b1..b1c1f590 100644 --- a/test/src/unit-regression.cpp +++ b/test/src/unit-regression.cpp @@ -49,6 +49,7 @@ TEST_CASE("regression tests") SECTION("issue #70 - Handle infinity and NaN cases") { + /* SECTION("NAN value") { CHECK(json(NAN) == json()); @@ -60,6 +61,36 @@ TEST_CASE("regression tests") CHECK(json(INFINITY) == json()); CHECK(json(json::number_float_t(INFINITY)) == json()); } + */ + + // With 3.0.0, the semantics of this changed: NAN and infinity are + // stored properly inside the JSON value (no exception or conversion + // to null), but are serialized as null. + SECTION("NAN value") + { + json j1 = NAN; + CHECK(j1.is_number_float()); + json::number_float_t f1 = j1; + CHECK(std::isnan(f1)); + + json j2 = json::number_float_t(NAN); + CHECK(j2.is_number_float()); + json::number_float_t f2 = j2; + CHECK(std::isnan(f2)); + } + + SECTION("infinity") + { + json j1 = INFINITY; + CHECK(j1.is_number_float()); + json::number_float_t f1 = j1; + CHECK(not std::isfinite(f1)); + + json j2 = json::number_float_t(INFINITY); + CHECK(j2.is_number_float()); + json::number_float_t f2 = j2; + CHECK(not std::isfinite(f2)); + } } SECTION("pull request #71 - handle enum type") @@ -559,8 +590,8 @@ TEST_CASE("regression tests") SECTION("issue #329 - serialized value not always can be parsed") { - json j = json::parse("22e2222"); - CHECK(j == json()); + CHECK_THROWS_AS(json::parse("22e2222"), std::out_of_range); + CHECK_THROWS_WITH(json::parse("22e2222"), "number overflow: 22e2222"); } SECTION("issue #366 - json::parse on failed stream gets stuck") diff --git a/test/src/unit-testsuites.cpp b/test/src/unit-testsuites.cpp index e83d5b74..8eb4d211 100644 --- a/test/src/unit-testsuites.cpp +++ b/test/src/unit-testsuites.cpp @@ -460,7 +460,6 @@ TEST_CASE("nst's JSONTestSuite") "test/data/nst_json_testsuite/test_parsing/y_number_after_space.json", "test/data/nst_json_testsuite/test_parsing/y_number_double_close_to_zero.json", "test/data/nst_json_testsuite/test_parsing/y_number_double_huge_neg_exp.json", - "test/data/nst_json_testsuite/test_parsing/y_number_huge_exp.json", "test/data/nst_json_testsuite/test_parsing/y_number_int_with_exp.json", "test/data/nst_json_testsuite/test_parsing/y_number_minus_zero.json", "test/data/nst_json_testsuite/test_parsing/y_number_negative_int.json", @@ -472,9 +471,7 @@ TEST_CASE("nst's JSONTestSuite") "test/data/nst_json_testsuite/test_parsing/y_number_real_exponent.json", "test/data/nst_json_testsuite/test_parsing/y_number_real_fraction_exponent.json", "test/data/nst_json_testsuite/test_parsing/y_number_real_neg_exp.json", - "test/data/nst_json_testsuite/test_parsing/y_number_real_neg_overflow.json", "test/data/nst_json_testsuite/test_parsing/y_number_real_pos_exponent.json", - "test/data/nst_json_testsuite/test_parsing/y_number_real_pos_overflow.json", "test/data/nst_json_testsuite/test_parsing/y_number_real_underflow.json", "test/data/nst_json_testsuite/test_parsing/y_number_simple_int.json", "test/data/nst_json_testsuite/test_parsing/y_number_simple_real.json", @@ -765,9 +762,6 @@ TEST_CASE("nst's JSONTestSuite") { for (auto filename : { - // we currently do not limit exponents - "test/data/nst_json_testsuite/test_parsing/i_number_neg_int_huge_exp.json", - "test/data/nst_json_testsuite/test_parsing/i_number_pos_double_huge_exp.json", // we do not pose a limit on nesting "test/data/nst_json_testsuite/test_parsing/i_structure_500_nested_arrays.json", // we silently ignore BOMs @@ -787,6 +781,26 @@ TEST_CASE("nst's JSONTestSuite") } } + // numbers that overflow during parsing + SECTION("i/y -> n (out of range)") + { + for (auto filename : + { + "test/data/nst_json_testsuite/test_parsing/i_number_neg_int_huge_exp.json", + "test/data/nst_json_testsuite/test_parsing/i_number_pos_double_huge_exp.json", + "test/data/nst_json_testsuite/test_parsing/y_number_huge_exp.json", + "test/data/nst_json_testsuite/test_parsing/y_number_real_neg_overflow.json", + "test/data/nst_json_testsuite/test_parsing/y_number_real_pos_overflow.json" + } + ) + { + CAPTURE(filename); + std::ifstream f(filename); + json j; + CHECK_THROWS_AS(j << f, std::out_of_range); + } + } + SECTION("i -> n") { for (auto filename : From c5cf32e34d19573e2fefde373ca7a48bdae52be3 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 12 Mar 2017 20:59:33 +0100 Subject: [PATCH 077/530] :hammer: added user-defined exception 406 --- src/json.hpp | 3 ++- src/json.hpp.re2c | 3 ++- test/src/unit-class_parser.cpp | 5 +++-- test/src/unit-regression.cpp | 5 +++-- test/src/unit-testsuites.cpp | 2 +- 5 files changed, 11 insertions(+), 7 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 41a85eaa..5511dcd5 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -283,6 +283,7 @@ json.exception.out_of_range.402 | array index '-' (3) is out of range | The spec json.exception.out_of_range.403 | key 'foo' not found | The provided key was not found in the JSON object. json.exception.out_of_range.404 | unresolved reference token 'foo' | A reference token in a JSON Pointer could not be resolved. json.exception.out_of_range.405 | JSON pointer has no parent | The JSON Patch operations 'remove' and 'add' can not be applied to the root element of the JSON value. +json.exception.out_of_range.406 | number overflow parsing '10E1000' | A parsed number could not be stored as without changing it to NaN or INF. @since version 3.0.0 */ @@ -11717,7 +11718,7 @@ basic_json_parser_74: // throw in case of infinity or NAN if (not std::isfinite(result.m_value.number_float)) { - JSON_THROW(std::out_of_range("number overflow: " + get_token_string())); + JSON_THROW(out_of_range(406, "number overflow parsing '" + get_token_string() + "'")); } return true; diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 711dd688..f1173688 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -283,6 +283,7 @@ json.exception.out_of_range.402 | array index '-' (3) is out of range | The spec json.exception.out_of_range.403 | key 'foo' not found | The provided key was not found in the JSON object. json.exception.out_of_range.404 | unresolved reference token 'foo' | A reference token in a JSON Pointer could not be resolved. json.exception.out_of_range.405 | JSON pointer has no parent | The JSON Patch operations 'remove' and 'add' can not be applied to the root element of the JSON value. +json.exception.out_of_range.406 | number overflow parsing '10E1000' | A parsed number could not be stored as without changing it to NaN or INF. @since version 3.0.0 */ @@ -10750,7 +10751,7 @@ class basic_json // throw in case of infinity or NAN if (not std::isfinite(result.m_value.number_float)) { - JSON_THROW(std::out_of_range("number overflow: " + get_token_string())); + JSON_THROW(out_of_range(406, "number overflow parsing '" + get_token_string() + "'")); } return true; diff --git a/test/src/unit-class_parser.cpp b/test/src/unit-class_parser.cpp index c9240c9f..61aa1c37 100644 --- a/test/src/unit-class_parser.cpp +++ b/test/src/unit-class_parser.cpp @@ -277,8 +277,9 @@ TEST_CASE("parser class") SECTION("overflow") { // overflows during parsing yield an exception - CHECK_THROWS_AS(json::parser("1.18973e+4932").parse() == json(), std::out_of_range); - CHECK_THROWS_WITH(json::parser("1.18973e+4932").parse() == json(), "number overflow: 1.18973e+4932"); + CHECK_THROWS_AS(json::parser("1.18973e+4932").parse() == json(), json::out_of_range); + CHECK_THROWS_WITH(json::parser("1.18973e+4932").parse() == json(), + "[json.exception.out_of_range.406] number overflow parsing '1.18973e+4932'"); } SECTION("invalid numbers") diff --git a/test/src/unit-regression.cpp b/test/src/unit-regression.cpp index a8287aeb..33f7a9ef 100644 --- a/test/src/unit-regression.cpp +++ b/test/src/unit-regression.cpp @@ -590,8 +590,9 @@ TEST_CASE("regression tests") SECTION("issue #329 - serialized value not always can be parsed") { - CHECK_THROWS_AS(json::parse("22e2222"), std::out_of_range); - CHECK_THROWS_WITH(json::parse("22e2222"), "number overflow: 22e2222"); + CHECK_THROWS_AS(json::parse("22e2222"), json::out_of_range); + CHECK_THROWS_WITH(json::parse("22e2222"), + "[json.exception.out_of_range.406] number overflow parsing '22e2222'"); } SECTION("issue #366 - json::parse on failed stream gets stuck") diff --git a/test/src/unit-testsuites.cpp b/test/src/unit-testsuites.cpp index 71b45b91..8d6a8162 100644 --- a/test/src/unit-testsuites.cpp +++ b/test/src/unit-testsuites.cpp @@ -797,7 +797,7 @@ TEST_CASE("nst's JSONTestSuite") CAPTURE(filename); std::ifstream f(filename); json j; - CHECK_THROWS_AS(j << f, std::out_of_range); + CHECK_THROWS_AS(j << f, json::out_of_range); } } From b8b4362ca49698c677d04a8c2088d30925657bf5 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Mon, 13 Mar 2017 19:04:19 +0100 Subject: [PATCH 078/530] :fire: removed accidentally committed binaries --- doc/examples/get_ref | Bin 38132 -> 0 bytes doc/examples/json_pointer | Bin 47612 -> 0 bytes 2 files changed, 0 insertions(+), 0 deletions(-) delete mode 100755 doc/examples/get_ref delete mode 100755 doc/examples/json_pointer diff --git a/doc/examples/get_ref b/doc/examples/get_ref deleted file mode 100755 index d2dda9ba81796fd46a0b3d8cb861339285e07f15..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 38132 zcmeHw3wT^tb?%WLi3kZJr{v-`AxzSQI6$3|{E`zw%Jjd}&Gq4fF^cqS7=G5Tg74 zYwa^{C%(oP++Ll}=d8WWwI3S)!>^us=|amYS!h{_1(s!%;SFA3S;@pz z2wB!2;uHE}#3S)&eLla&w=2IF4HM!y z{L+jy5Q#Ljw{>L{n9r}SM)|$X~9zo z>AFWEZ7q?my{+}_Es>6R=V|%*?8uM#{fGVI3LgQCM0U4lTA@(BGUTUx9D96-=kvQ# zljIwTv_xM{z8!|&l*vOpn;+|Yok=5zL>i;P$OS8AibQJhxII<&t(%)VkWT5+R1`ENdyiQa}>Hs`|PscTu^=gEn0(ox3moLj=_xxkLr+a$PDc2WCd=27)crV6#Lw8r_4K2;} zH#F{TX#~#aa=et|^|jWq8=fxj7{2_Y_g(kBzBoh?n`>Qs5i(z7y~)P``7A?ynU8iX z`(ZiM1Jk7cFIs`xmt@LD`3-M=^YZfL5TdzWP=DuQxdLIseRSU|s|B8kpC>yawhqFt34m4a{p`UIX(QnAgDBYhb(W{Mg=q zY8V{G!!MH0c=+ih0Cp&8F0db(`n(|} zPAD;ScvuKPN?<=KX(0jaW=R>>j6sHHRcNjWX zy1*MY>->B)zI{2EFG6(W&z=OYaN6z{0(^V~Ou)sLGX($$x)CCOK1HFkgOj zycxuK(mlJUyaSgan+x3Sp0b@|-qY6#o1j)NQ@>j8!xczz7G2}rgFt_depo&?YJBnh z^7o&@&IK}!RzkzJ14Sm6QrAx*R@={&h`^E3zK46KOZo<0hh!&JhHv1&E#N3r2a}8r z9GU2yUf`rImPDsW;zB1?Dv94^;=5UNDj7lyd#>%)Ov-R`Q;FXljo3(f#j$?9_A>4s=`N+Z30;)ytZ-_^8#a2$z@v1Ac`48F++9^57c}+yU>gm zUz8!wA?XK6ABstsV){@FtiZlQ-i0itUyBJ!F)}<+0!OLsq2xP`%)zVNewWI$CpWB2xyT7iKF`)Zy&7=Cu;+)zm%DeT`M)xG(% zG$g52Uj+F2W7zhyQk8giYH@;pu zy$ULq&I*VjfMMW-F`?k?9b^AT-^a^w;BqX$Hi&%)!`AV&XrN_Q>pfmYlaOog_MaMK zJ2}$BY-fah#CAud+x1G1LE;GeQQ*KL^c0wXB|&IZyVnWt%aX|?RF!6>9MTje3}Z-j z581sZAa?gK!oz?^0Wls&qz}6WNWpgRco22Qc*N^q5|#P`!uA74UH75xw0X4cc%5$4 z7q9bz*TG7;IxyUnNbJaziLoX_Zd?YK874g(ksH^6X z-TPn}vJPfUD$Qzbsoi@Fbp~{{qaWaPXxX<)nxMQp=H$B##Q{9Y?y(cCMN9MyoVk!7aAC z+P?k;`@y+I(1doafm>1H$w-S0h~+Jbt%xOz+YG=BBi-8S_8Ne(3!p&Lz;p*i6#H za>r}$sgQOU^ndjE6Bk(a_-*iKkcTdSxH=dX4jc|z1&*2%-qo6X(z_mgOznQ}@sMxy zYaa7{_`=+r?+*lRcX2c3tL$RZyxr|+hjHJYz!K)$lWh5PUgp?@teB%6WCjE84=9cZ zP|W{g{ZJ8(YjyyRadrUB>sEM|<>W3bAI-OK*6Dr@S!eBr?&rMMY6j!(6^tjAhf0YJ znLTmAVlg2Yhq4NwiJ{M%(dBBOh>9KM?)LrZ+B?L%O26zz$D4=95l&X}`(4EO}_F@$LZ@K)GP$aXtRZKnfz-h!5` zEdv!OPBYtKHXIK>i7HyX`<`4F=%o+hhM#mE1ivYJe9^;9LPYOai4%Tu`g^FK2>QSc z)LUwx0}+_Lq~14QVCt<(f*x1hy&P2TW$Oiqp1h5y^7IG6l?xG>Z=BR7vc=Mk-QxBx zeu=i6)d2kcFW=3$4o+?AgdxQ%gxDW_KSpuAqQl>A%uid zNd*E%QT~tN^ndy=x?{Hde|~+o{>4jiQWp9-`VTS4FPI~9JLE6Zw50?((}N^PN}c9` zaheAfB5(z${2b>h{}iR9JP=PAjtX!_PBP^xm7i1~U=-zl1oP_jKf-ck%l|D})^z`| zQ39mmz-9ke0R&Um0aDd!+76|xQxXB_|B4Af|5uE_^nXT({wFCV<$-w0P;3JR`2Hn0UnTmI>s(MJ9u@78w5TCI0QHh zI4pwI995~tSoTdK6($(WPmt#*;~t^8bw`;Fc^E}`_Fn~gTo{sUd0fsZzCN)Ka%#p< z{shXxG*ELA}R9rKKxGCTspa(buIHU5+syxFi`wJ)q6CQGL88I@%xOFNI(;*L| zD9^tJNvSq*|8d$)&b`*9zn+_W6z|7hoSWMM(n~Q4>d_MZ`zY`%dmQxr5obG`hJCXx zU6bve_Y7^iSxqJo_Sf5(<+x7;+xBP@CmchzosN`WUQF~my!SWV4?iF z-wMu0o$jw$MXF4wx8`MJh*u(lnqDnrHK z09TbN?e;n=jD%JfDit0kM6eNN2*@3_!oYxiZOE^o}wFzY#u+sNLs2xB1Z*7pOQ%RI(+9N z*(W3l(mqJPWjqW zm;iN0V4po<;p=5aF#+hCg6XbA+HbIAREqlm)}_F*G%+%%wVnV-Yk~+gv5XnrlmI6c zQcQr8s#HutH3nU344I~lDh-1&B@&<>3)n4XWX`4p#Ch;nC3XW&3xo@*OlBCt(oFEF zM4n^bo5Q%nF> zUzEsA3UHFPq>U)>^-?R5NdYkYnq2_=0L28rCM!merHPS#t#uzjS`$Q|i9XFCz)AHh zCIF%-ra&x10-1cvU>I~LkpQe@6%*j35{eOITLyALl@(|+VYMjo9P{4uUt}2cuq3ez zZjxb$I;h+fn8|)%n2}CuNYe!K9pBWLq^Vnu4k#qXN!PW5SwgsPvz@iJ`$5>gy`{Fh z2eZLF@H$dQmG_kL7T}~j#RNF18N~!(3IR5rd<cr0$?)|I?>|mwO!q~TAw)Z=x8OT{7-GqWO;+5W z7Kyv7NZd_D;x0!VgrlH&Y`wq2Zly2xfQChGN5dlbJ`Ic9_b}`xXQipfy^jkzN;$nF z2*2F7%Be$<{Yn`5eF<@K9V`iPIRSZL=*4BHAwC3r5T+FdnTI@h zu^a>(#<6hk4@-J~2@5SAdEfvJ(kQSPemT72PJfLDkJ|&B0mLijql1&cjXC5n-8C@d zvIHChoB%$79e=nHJUF1^kM)8q2hC}Y@yN|-SV8@^;sVH0E{>hq@^7t^U92#+mQvft zHl=nB*{S>M#19$i5HQdLa3Z-IU(`V`c(OfA!}>r;`V4|z!^9sB<3c?{W1aj zldK*H@U+Pp5#6{9^1!7V%I=18yRp{ZIgBur(2WH)Pf12Y!tn)e;SDT9coQ!ih#TxZ zB-a9rOO2J2?!ZIQScb|`3XW5-!#*B<3@XNc>0=mim|C$t`WS~B_S}7sH~g6Fmp&%@ zrCb)FAn!H5paBv8-M2SOMHqiWkGg_|DdqtUEakNC3KM{S;8tDki|3$oOY4Vtq@I$+y0SK}fR*$aUp>7fL*-Z0k!d=(qx~ zH7=O;3pZ6OVm*w_P^c#)$be7A(hU z&T*KtW-D2ir4=C4mCt)VOqOSM-t+js80Fik#b3bD6xa{i#m{?wj*TTW7e|qH?bLtZ^PZYh=}yOu zA+*M-_seX0*;WC1`f?Yld@jy=zJ?8pEOIJwHH1dVg{MDc&~1 zOduxF8L47_bOtJ=Tk6qsYIGSeuO`jMzv^VRo&_8%?MqzR7lvrI;aU$ehvV@O6{?UOSAT+B+KzB(Xl`fEOMMCJMv0v;_&{sknGdc)gw{|pf?`BN|$8$)*z{GoI;iz5@jyk~iNNVhTE z|Iv@4F~+}vxU^<@N(n(pjJi#o2tO6Xygp^8aIvv7vjZpBeK2XS`Uy9K!K4;xD0z%k z0t@S4_$ib_6uyt%G~`h^rLPP;uKZoXWbuv5o+dle&+j0rojcdJ!lNO7_456R@kSj`LC0*VN7JBhvS9 z-@sucfuam09!wJ1AX%g>A&RI{x0!H8Lsfe3lb$UHY`%FZm421#CojD!2<3^VY4=uI zf#g$KvNl*&-{~I@e|WK^adRR(I3B*wpo4C0u=io?b&oqwc(=dGj5R55`eVo;Gffc1 zI;ku06^B&fveW@GY!^~9XbQ$-WG|zny2;}tnbUg|yVOtsd3g&^5W*^FK>IjP-1}JX zZ-7t)@O~&+bSUgU&XKcpoI`_rWnwOz(3(4<$Fneg?6m#7(~gENS}j{QC*} zPfeke4Sg7J(wG=*W2}%fa8!VJBsjGi0@j!O9OHV05XzT(41SlECY=5>)>G&S>}P&2 zy&kNSha|#xc4Yiumro+AmLgpm*i~?oPzOGu^Sa5ZUEq2gX-H`L)ok2x$-ve?nIf+l z8B!*73t5sskYK!#B0WSpk3pgcN2VVZ>FEy~?plaB7sBrgD+=jm74I+^- zR~mUxBaJhx(*Y=NSISxu(^zah@Gv+2^lB*@yGbYoR6D=K;d+P313A2VFQ7_|9UR?( zsK5gQ08PL=e|!lvoLy(6~-((6PUWnsX=wqT)` zzsOUswF)HuI?<31f_xC54otU?VG<8n@b5ot}o ziFCL;0yQRdUQpUIQ37z*U}z;z~rXhqH{J zq&&-%3&6;ibQsa7ij1XK97ICp`h%!c`;R4HU{)9zx4#qy4>}x?G*{D{Gap3NzWh-9 z5c!&Q&bUnHQhQe}E7-0LV`2^~bUET6CdWhpnFW%HW8~DVSe+ZgBpFN7&&Sl>Z&hyvL^nTWpvL|sy_p`2j65U@aom%{BdL5_t z*WV^w|C9LZ_fMI6XYOY`jOopGFXJqS$+Zei;eCOf`GZi8j2YU&v%8=5D)7PR+d-bc z0lU-t>-=#DGknxvUyg{hzmD}~&iyPjm|5#(?q6o#`8&tw`~Ctnam$67?%O|i?NP`v zckS2Z`%U>iAm1_h9+U5J`JRyPgnS>C@3-aq9r;em_fh%sw*TCeJ6qGgF?1V40UGJIGISk76%5%7En}#jp(`1BKSQr& z=#vax#LyQQqP1)~#?bRHP)*-s=oCXgVdzH)LDPqSmarDyMO}OZ@4w=;#__%iFa42) zllZjbmFE^z`z;6^2DI+Qhd#07`K&>B`E&8-@gbRcO7=k?cCn;rjl;4JCR?D&Tu_c}yaoB83x0DZoXkmW(VSio9;O=KfJWPXH* zgHJD@JadPyUw9ouUs7BH{6(DS7%_qOb_V#&9p1U{uH6W%!uu%-dK@oohxJ7wkAp8u zJVu1K1S!iCMAk!ilKqs(DukHw5|J%HE>-Q2zmer;ipht8JCXcNbPDT}__RqhK68i9 zRsHwxPe7C1R05^ng;Ex?DirViM1BV3y+F1>kcD5z=YBusT}XaE5P2Q}LI#(h=GH#} zxkcmg+e(OG^Q|Gg3Ljj5@t=}?TGw>lZ$M6Y7@$y}VZ=M}`KD5yySi_#Y-?$6YK^wF zt!|9Pqs=Ylq4?g8SR~fj+1?pOf67l^tH1Tt+}?)NXjfOPGmhWF+7s<;j<&_E<;!pA z>TI~-?ymN>+%f+!Jl-jz!`(-q>D$cdQ{EZ0`)VMkI2%i2+etcS{R@ zy{bsIXlG}1Z+0$?c==5F(_h)SOiWg`=C+3R){bbrxxOV9+!EUr>x{KE#I{lFT@Z42 zEFS5M?RrZv+7R!Kwgja<&0Rr@vRW#-C9+uNhsNsL+goDMHmk9@t0CIi7;6M+YdwB# zEu&cLF40h0mCLQ=*3KZCob7uBUXF)r48F|Z0JPF2wYBGqK)`XsUY|R@`sWd0n6TmG{An0=NZ5%oE=@U33P8oI^bXx z@&nGGwU^eQ^CRoU*6XY$k|CP)vK=eYZB#6Y|Vyk5MBJh1Rs)C`=aj{i01~~Z| ztE2-qF9Eq`f=IjCT2S^z&|(~np@L%i9a5LFYhQQt5^b^vm({4X)MnF$44IeENAfuYq|D%xhp?1M?b~*TB36<~1;{fq4zgYhYdj z^BTz3z>=%Ww%oe)*1AjAz5CXVrCabD>DwfENmV3r=jN^P6_H5!%7*stc-Xq5GuWRM zyQ+cvE!7R-isr_!B`Mpgx5hL3S5{(hfkwY`b9o3qJm1q3t#95_UV(5|bEKgKyIZn- zCBO9^wh|uplI1C>EBlJcnahyZwhe`r*w|9L$4VS38Jsl$ zR!UTI=eEKOIR%`|)Ss0%noUP$YZ&_J+@U>Wqjr(iE$t1_me|_vu2^JOv>_G`Z=rUy zYgnCiImDIC?Ol=jXjg1CHvf<@yvORdI-9b)TzLi3p=jOmt8Uo5HBw&Q&=l>A#5<$S z@h)_`FyhumTUcE? z#mTbo+_IE<_n!-%&28Mz4!0UP51j*^8_Ko&4CeA$q5Y>F9VixUE#85`tGi-t@y@;L z%fnl?u82@`Lub#$7i9P>ShRG7dszI@oV;dG`HaVM64Qfd1x$1oHqq;&jrGm(NEH8f z2LFqXW*9YWtjC)=+wWBkO%%_+ssX#{o#DG#{$vixT9`H+fZ=AQo=m4^49A8FWj>{8 zFxT-|XKQm?G#-;tJtOh3{n3aSt2q0R|IGv^9ArA1A*he-Zf=WUI5#vgC9!}5L$ac8 zXX5)Z@w+->dF=W#iBfekI<(+YFxsyFugI2IS9fcS(Ss5lF=s5DaS>;Tk*%>-Xp}U= zM4^M07*qL_L~zcuAYzrZMw{Dk0=b~I2^ESpbjM@(Utgk}N|xPemGgXIq1BJWr3Krp z&O%dtrZdUICDwvM!)<-Kq;R^hCPAA!2Ru-cX{oSvykya$sZW(yixwqtxVmW3{XAk_ zkWDdDlAHf|tBM=b0z*FB_a^pTz&p;5-8*e$vm6vrpnb z^Ya`0C4=*9l<7-wSW9_%MoRn|;8(snkv@mC4L*2>rY`_ZU~wXSj(H1l(EI0Y4F>nm z)slwZKd1Y=!Toc+zb9UTL5YXuKZTbU&Uqe8{Kp#4V&(aFd=bwid9F|-jo+Aue<%;< z?Fjv+`Geq`jko0CU(3TUSeQ${F%LhOho8v9f1QV44Bcm!e@z~KV;hkrB=|8IHt zH}ddr=ixud!%G+E%JWBpCm^CeS8IB@Jwkc(Z_mSb=HYF5_<=n9V|n=J6i=7;r9Aw9 z=Ha|Y^bWkWc+q9$I?`4^uD-Y9eJ9>`;pJ~wyc_R(@b1L>Uc3>!b$FwAV|aJtZNl4( z_inuJ!;4{J?ZVrNmlt~4@#3k%o45_1JMrFy_jbH^5_}u+X~FxumMv&;ui>m0+sM4F z&>PBiWQx;dyFR2SbhHY&6w>5!DRc)T;wk1kpsey4ES|H-(jI?i+00a3fS-c}^SNZ! zDouJWM8hKaw-zjGb$kwaor5}NuZMqI$z++B#pG;Oz-O8LtcsR3Z>B)A$(V`A-CxYa zXRnGgvFX8aX3OCs8Pn_FOj^zYSfew8T~@w`%w6&#B5xOzF=w^`DwH`Jo`{isI4py+ zmZ@e@Tx30MQnDA)1uJM1k+*aTE8OcV&)@6|*Z1sg+`1;ht-+PiM*Q!bG2D@CX?}mK zGaO!DUftZhsX2sA!FOzn(3y;-XHuk81+ai0;PXY7f(>84BLTeT_P?#IO zk@9?_vn>)1Z(SWx8=juw&zBZx%x!?*qJrGe*Ay{rJoi}H z66@-!1|?5HRz@h&%8qDrXZ1$JZM_xIc+!piA5nI0D`)hMgB|SS$5izZz1A*Ahv&XA zph`1MWm;s7w#X{9NM*4Wk+Z1swGnvw9Y#30^WGV&MiXugvw2OEtwGytsxL?LY~H#m zQjN62mio_L+yAfHc1>dp=LhY3!?oxH<>lIuBW>-C=o{fk?VfNrtDl=eOv-a=VTyI? zEHhKQQ|H-_^Gmk8KBWF+)-FJvi=ks$4eeQ~HtlZ0JuUDzF`Cz4G>hfU)0?sD=ZS2& zdMn4-p{h>K<6lWebR&P3I{PjvdaQ&=JdGYJp~v-N6@6LH6W?Hc#mlTI@_Iyd^Pcb; zb2CKdsMD{8R%YpRWsz}@xWd)Y%Hpe`brE#&aJV@f-m_&}~Wy7b9-SD8OTOW#^) zX!)HA$c~ezEX;uaR}!zJgMEGoyndS7i026-D|I z;tKoHisF4~ExndvO{I$_b}+wxr3a{ywjvbk!TJYxuk@x=XE#^-)tgx7HRcZ$!>hvK zwY;bhXX-1|Yjsw=Ru`!k;tK1vx_G_TpQc_qW;SoNBjv004+?RaC?4C5YZSQrb{_3* zl!xTLndlK$S9;D{*(%O6*(%J|OQgBAqlNn~GGA|tY$@o+dEUeCc=JlOi125+)wOv$ zFzIo2zQ~uqg_W6Te!mi~l|Kb;i?+thGBT~X%mkSJD{PFR<}DjfdyZ03%(he%*p}Q; zSfrS=TbP#R_p76F84er@5%jfit=f_Er3Le)+jyRdx2Cx(k~>S}Y*^$^N2_sxYQ`ilQ&{V{WTac&CPx-M^Ty-0O7gp6h~UCU5qq*-}yQ#78VKp5yVy!6Zc zu)HOmcqZG*;cV4%Lu|{&t!vLf&VuDeftZCuxRAuJZF-a!k`xqlJ{SW027{Rw3#LWg zKde|8iF9s-C2*hnYUPgII{f3oMEZCzj6%g!Yo+qXX?c0<9=d(9 zxfpJ4ei_mgvG!g0v&;F_kd5cPf0K6rlO8kYgB~+8W25wfjQRRKOSdXB_3UbC7rS}; z`t{~kH5QWiot2Q4AG5X~rlPzsYIR{$Va63}3aD087DTNqWWTbI{mMf2s|tl*wSp~x YQ}lEPF-EoUdQKrtW|j%#ZWEvX1IimE#Q*>R diff --git a/doc/examples/json_pointer b/doc/examples/json_pointer deleted file mode 100755 index 83f8491caf97ba3c05ab47ff0005c6be85579bdc..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 47612 zcmeHw4SZcymG4Q@LO|dK1dU&CRX*B5X;PpK)Jj@<;YJdOw1E!Txk=O8CXkOyJ|M^x zZ%wb}a6FnBy-pOJiKFu-<2X*7@iGy(ytH5w(dWdG(L6P|F!nH zA14XpK$*vHUhgmGtiATf+G~HUz4qFBpZw!co3D=C6uJI7`WF96iqR!8(&~j&oVf8vsU3DPrgBd^rr7tdYcRp;zjLINF451 ztgrKqw$9zLw$43Wf)}@M`TLaK5tEL1kf%bD?017GODwi^^Y+aJAxWhp9iHC6<;MYl zANw1|M2^>3tg}7Vb7#k{uJ%}>w|iN7Cb`gKdjH4%@fIHejK%hLL%fB!dc_{&$b4a@OZr$A8Q`8^+} zAAYM*UZ$fQEBrB^N<0jc{=f2K)Iv!WI3jGB~4lDoWs-FFM#=Iu&;;Nff!i+_{ZTN9<)j(n zofIrsQVT}jy|RLWiO1<$uxc$>lq|WMW}qIU3^l}@lz7^y9l~|CK|K{qHHl}cRY5{Q zKf$sE69RU2n4H_GA+p?!0Rs0R+ueGXk>H>Zw?2R>M36^9mLSmMxeNu(Pe?+)g!-?osdYU_}KS;R2L2r$M3WXsi4u~b;YRx7Af zP))FG!GwUFZ6N2EP{{I6(U0J+f!i+_{ZTN9<)j(n{Y5-bun8>~dG|gCW#2EDc>E1w zqhS47uqat_Hx0K-!D@(Mj48W21Y{f3Q?XQ&hP5h4DCj3xwqQcQ&Q_CiJ5@=RPt$^} zhTAU~{ZTN9<)j(neFKXM3HA#xqy;1I-q$e``~Kqb7-FMfjN?7Qz%5uWz@rQz+x`}E zw&-ww!+tzU$}NZ#2$;JPen>NgNP!@)mucR*M4D1fL(v>eQ^Pb*!ekK$n0pw0$ftFQ zG^a4lXO~DbplK*j0_41z7M5p;G*@aGj@4kA&oRwgmq=5pX*djmX%bBHb$QU z;wH=vWKW|1(sgt?>TJ3|y7gi29F_s?9zypcF?x9aM8bBe+z<4AxC;y;g5SMT zHO$A33`X5mJ#CB}Nk*Oe!Kht74EIpfi4cz98O1Y(XMEYwP(Kv4{}I`hAv>sW7CE31 z23`x#=0}4BI~fK;^J9qjP;)U`*&Srd&5%1UTbg4xD;|dt803ovnehexP?DOH$ZoWz5HpOhw&QhdD>tkqNZf6xwVCZgh?a;Vd2x z&pe)m<=bpp+iV8eq0OerL7jG~DW3xrQ-YlggVAOT@lI|kW-A*eTW*Hjc_*=w$urtX zJkCIDxEaTLBLg2pV6d7uBa)W%%92H)e*2JIFyWVDi`u6I6#Vvi62Yi%iwx%4XM$ys zF-4cO&$p?{pnbm1w03G7{=$ymKJ@VJ*1$18&~rdtxkiy#$O#s51EaZFPQ>1Y>ZZ2g zouh85RNkxAO})`OQQcIIy?;|WvvmJT-C@dxC*gKvg%;ia2$yR+9}d>U^M9}uGoM!? z!pxr)4=Dq3`W}xsejxe)J-NMWbbejS2N#sgSCT;f%tfTKCY|llZ za}EGnmaH!aj+963eG5^$Z$9b(I(_qADZBc>Bjr0E-h*GI@?nGO8zf4eE$)eeK8l5Q6X^@3|v7U>TIC{Nd`M9 zBIvuKaXGGn(=oEc;-9XN)CO}!9d%UAa}?XSxh!mU&8oO-Vog{ZUZ z=i#0QTmW3CcTTEztok(v>+GrMC!)_hi}d4YurWx_QFN4XNZ2vZ8wEZMNjrt`3HSp- z+B*AYyms)0VaZndkNWdIMwNbF)SrX>(|dKUA1xjq`;X8>9<>*uuG>n%=B676>`Vo~ z;nZvKX6F#8>|_MknbnNqD6lh?iV@V>s0y9ICP5dJC$GY~fqb$~Gf_sJ6m>y-9r3Ai zi~^D2WDEkrf+HAOx=H98a0aL(=wYXYU#!Ul`wsm3#pNa?z5ypi?USR9%hI674Qh}8 z+Z|F&fb9+omKI>AM-&rar$-ed*c!U;ciE z5MxTxx&kmYDki{oXB1OVTLunyfWllcfiaCFz;?$K6JWa&iV=kFJN@8uCD9g$Frfkt zNGv+AV1-hz0ZZDNC6lRg#76(J=C0&I9s`#|phyr*re*=Oan#37%`?DJ03}Q@0hkRj z0ib>b*eLUP>M==3P9HnI*U~GqkDJYe?A-4=erv(T- zR5tMJ+R%Zmkox15V9B2JzI-F9)t)?@`bR`4fpn1yb}!wF`3>W2S}2><9yQ%cczC~s zxHr0v82R9zWZv$oJxm4*g>aG;PiPlUV&;VJYNYj(c_i$(^g}LIQN~g4fr%1Z!sn(Sucg@I?8x?^hi(Nx z-D(wNrA8ls{Y~O%L!hT$8l?*7ob_e_(gNN&sK>dxu-p+Hs0^Lkc30v(cij?x8<`nb z$_4S>AUS85b0ElTX(UrwHn?{q+)rzd2&JGJTXQfrff?oEqX?b*!AbP+hvPhoe>n9k zyw`G60BI)7_zu9ZEiR z^yxz-rt5M4H|lOW?IwatPZerf0JSag|#-#tPiXe!IB~Z4IA1$jZ3geIkZdFj{xG4 zI-)URuC6e8s#Pfqu+3t^tT5bu4HF=BRG$Q~B!~_I4zd?8PuTVeN>kO05Am#?1X3?m zp#$3?r+vlYuItdVs5rb2qq-#Ox=E2YSTeY$eoW^3daNQE$FLNcKvrB$5MuREKaFPw z&+M|hqm5XIWXH(>cSzG?;!2GQa7O^VPyQcaau{wMR;Kfq1|4>r{)Jvpy$4IPw0Rw4O)L2nrfAsj~cmAn!D!d0K0z zcp_!lymygeEhsJ}rlh4J3x^}m+>NU8u;5y_rx9!Yh!=Y3DX!Ib*Rug+vYGoZi>@=A zH@#TywMx^t6Do1XMJT5PbGT^5l$esXE|SbLcwhpmSkXSL<1#_ic{f-|FDMD&5we3r-%eKu$drp9P){hT z>bY7#szyM%T0JWXp!5pxJsTvMh)Niz)96q}&%q)M`pn@}7v5_z zGMhxZ$sCBmxH}U>!_z?nT)yh6k%Q#gI@hF zg>ZuW?0+?}IQ z_tOP+iZ*OT)gDgm#T&IC7(Dx>_7%Lq+>4mQjES|Y$fi%LAXRILlM##t6D+Gm*3sDr z;noM?rZ$l}dnY6j>>Ezsc%9t{DU_8yVn1Vl6V%y>>S$1t&gFgWUD8zAwP03u8fQTe zV-SzsI1e}vI5QccN!J?K_ZSug=fLGe9^-1DaVEDEh&;x1K;*Hx?+RmC>0nQ=N_9J5 z`x*Lx6fY%R)Nxr?X#Az7GBsyGg%x@Vz`tSyg?dUk=;WqjBFHr=QC$I8uPV00Y~_f8 zQiepnf?^0%Y7zndRHz0q=(8F|P^h4g3OcR;=+uH`5(3-?0#tp;A@sOPLDH7-gCYZ$ zHyrf`q~jk47-0;o<|LwSr8`AC>dt?tf)&odlsdVJ5)lMTf@GnZj&1CQ3W}8wos;K5 z!B?COfhi>*0LB!>1i+Z0m;lvVX4OZ4t!kE&)3!t?rkDp>uuLjhrJ7%5hrQI*A__e6 z!Ct9jt>kGLav1LrOx9c&FPWpF$jx@QN}f{dUJKG2iKZUclOVR=7-#<@9O*ZURA5r? zUIZ?-Ux#VWv-|#41`FqROj-_5jvJ?b*88ejzkTy^-oCuZo_rxr-UjW3`;W{*|G>yy zAWN=)+(4?sUG*VMrl=fCd!Tq4_0KmAo4x%Pv74_W5$7+MF*p3&esQG|tc%dk;p@oI zhWeo)hrAy|z>n)InKInmRsYO605%zAQap{0h}EBXrgjkED1?Gm1qlTr-Im*#*Rd?y zk*&g=+eGee`#t z6-%zY?;;VWa=O1x_x?xN_08Vb1iWqcLA?D9!YTUD)RPbLKB_ckN#lcLB4-cENAlA% z(_s2-1w33$w_*-*o0Axpx*vAs7K*K znke$Fv%hO+w!$Hnplb1<8M+cKZ+Z(0nlU*EzzzaKe($Hk7%6Z!fdD%A!e+)#(FS!m zwGD3wEts$m+(1EvBbtonW~s1wZwF~rx5)t$1Fw^ABr}P%{DIIz9>0A~RFtGbs ztLW=Y5~*E}p7QV)$s-Y60o6h6&~R=&;YEp1LmzRgibQfyllKTPviNxd{NF;*x9*kE}7JH?m#JpKq7 z)!NdnA2%WK(*lgi&tHA`2Ecxyl^POYtyP4K(w~#%GDMTsjX2)&W#XyOKs6nR9mLl^DI3!Hkhi|%IB!f&zSdoO;jdv z@H8U{QXYY$z}JLTA%@jc!j{s6v8P5o#Su`TUdxdXaKlZzoR%#+cvuW*3IRpy2e2*r zkz;^ZzPR7#YuScCxe^c%wD*NIOn_?HtJOz9a8sYeQD_0aI+ih7nW-01&ahWWDn>_x zO;%}w7QMoy7gHjU z0&NJ#aB;|cH*^A!njm#{K8&^r9X9lWrZfuGeH6UG90yCDoC%;sk7od2)#F(Jht|Eu zX+S(n4+)&r;<+qXt^l!kHs`Y(oYBIWt;xva((m^@o?Ape&K{2?x8L0(kIN$Zaalwg zM$2@~#9aW-%$#T?fCA_0{HN??^xcXN^U!!Vd^tB!EwcYJ_&;1>WDvDNt z-(5@i-MoZfPJb&%`v$~A(OAp?S?{Bl-Q;1Grk1;4lb7x|v8Nm3TYm%C~Q z=|Gst%$pTY%p;d$vkAUDMBopI22ylfIadGyC&KT)`>dj6Hv4pRv=Wf zW+tHE&P^{yipvD(sPV<0>(uyy-5{l{K&F=&w!7JUG6Y67k^oz6N=urSbb6W4*aG}% z*UY!({7i~@pb9*RnNi@^%e;8L_=z^dc?HBN?Q=CDqQCe^TX5#9*11H1*!?K~eY>9_ z@Hk#*rK5^yhuZnLfVm$D9k~S>Bt9reEUYI#y)VdRLNLDl_a%F==F-8kM#f$w%Exp zdY}L^JDbxsikb5@Fo;UYvNHno2o97|#ROoGC?)_>SuiyqK)x#`00vo+;{pmL!>CkS zY6KuR9*BU$N=+dEW^Ba-_{TEL#ZQq8!${;y21B4slL&yVUNHesLlqOCk|C@<0&vPp zF#-N*z7+ESV^ImqDDdm0K|Ejl(U8oL5+4M})dUe_IH=4B2p-uO)-VMk8InlkO9n$A zp^*fR9pI~Nz36Y|*K8z)`Gt%hP#h)V?h0+zo}}&a$X;J5-X}pxe3QN@kV-MX3J%+c+3c7N#f{4fp#>%O;5?f>i!wY4F9H8^QhIInwV#*CKt8Pxi-j~$Px!C&) zbyw5<3A*<`LS@Wopq22p-M8UAcQ4oyRk6VDt|k0#UcwL3YEjxZzz@t(Oa;h#rz=mw z_bPP@-_KqoNrdlTs9X5{5#2DRNmEaHE0_fP8!Xb>;C`X=^4NEn_x?dlbZ|UILgujX z0q~bUKaY42@_4AZ+xsBs@!0!@03qo2Q4kLUj$i_SK$c_Bqkv<;$3pj&;V7+~q|H*+ z=T2;h;OFfVY;ovTJ+|v{*cvz<5g!Ro0$4bpK)P__lFGrM2pfkY{DToRPK&U>0SU(( zl5l9OHs?7~qylahz|F$4fYzMNIxTF*dk z+VctnuQQy;BiQOQPxIs&CYM#+q*tFKwbc3TrV|m<&NgwM+0N_`kZvHrbgh^iRe-rt zG1-yUtE^D&`z2;ZfNie1DzGzy8b%Ph?}UNb64+spq=vYE$e9S_%BVzj1wf@!OaKmS zDW*W=T{V1sd1nX=X(R!V-iir;imDhvp}Zp%w7mdYh(C5P`Rk+jd-|Zi-aLmb=5G35 zQ~}&e&j+|aLpkW4`VF|f%D4GSxH+ut=)q~&TQNW={s<=Jd@CriqiH~DoB@zz8f`)` z0nktdOAD~m(~1eOb%E^&DA-e$1Hi~`H4ErhED=;}8KoMmKLjr{BkGt4BauI)8Uixy zL(D3xp0X++w{I{{Pn}X1G+aQD27t^-9919ymUCgu0Jl=Obra&57J$AbSVjQ)mSO_X zw-h4?-S^IcpOy^po`GCIL>DUuxm-;UVHcAWdcG1}OfdoIVu~ryO?D*myO<#`q>%)m zizz0c=nQ9J7b|1dSO^Qy?tmkQno9RkaB1tO@I zR!Qg|=}0Ih!0aMBx|yu!EvAvk?+S)MgC-Gx&0@s_VB<$Ig2JvqDk!o7w?Z4ONt;xg zP(NdWPm2hXnFa~V&I*=|5`!6EJXIgvD94dQAC&1H22ciF=m#-kXrP?kVKUVUgul6; z6CcPj5SKb$t)0b?W~8}C)l2zuHRMpVgvYXle#sPE%2`Hu!A4BUGj-0Y*NA?Yof2+w zJ`m^En25Odew>5{?>x2W>yC88s~UMCmV&h8TB_A>*|$QeAorNZ*ttz2iM?iu%}LG~ znc5C6|2@b|e=tST5{4*I*7rb`0KI_+<$X??7y^T~0Ds0yN*flRKh6x&)6NV@m;yU9 ztQbLI9~_hNxuXKoJO!ESgG9j5xFpom$i>Hyi;Wc82Zxc!?}LWGh!PM`bPO4yc1*(v z3i}|bpiUIH4O(PnHSI}~jB>0;4X`Yvr+-c+Yog3b%(R4|LrWBMGN;%~;@{k}Q0b^* z0Y&=}oRKt)05_v>D{2-G)r8ED)Khv;KsG5L)en%XNg}98wMnRHqIiR%_$-xC`;f?& za)v;K@*u!mQC#wDv`V>3jV&N})+#}cU{X`Yc4$lmenmBjXVF;_|B7KcAz=zI{V7IJ zAn)B)CEf&(Q+J7|(W2mBvTPM_yFezi9f&dqm7OEVL{CDcO+wv;%8kda9|9aI>@Q4% z=nAw6up$&sZ5JGi16GU68vo-hr0nWw(qV+ldD6qo2RCqu(cwkLedD+Vi35L;3BU=! zDZr@+RQZByaShA_jVT~qBOp^PAX^ELD>)IA+ym=*1s}I5f#Z>quSOKV9sOHTuBmf7g6eGZZI59sU+_2=9sQ}0g0U{Uzt7Yl#3VJ?{*E2Du#M9iHOb;;(r${AAMu43e zQA~iH8C8s+P`Z{0owNmzs|g~gi783(yWW;uSU^n}oyiawQvw2h$J>&mf+8z$O>dZ0 zIs$NZP%!~eh6GCs!0~Fu1eo0toI8LrR50pO^UMQu0CGhpf+AO{&Mg4Vt(X8<%@k80 zjX#S-e&ZVgWg1BU8c;C-P#P5zpp73^9|6U8BVoQLN7$}a#-b84rN9giGg#6M5~cu` zu?5Qrurp1H39vJ*iV+m{ff>oeofeRu0?5@15!B0|B-DG1u%Rg?0NT4^3PegwAd%k{ z41t735&*?eF#)(`r5Hh>+#nTHfC5+hskup<)+S8%OLJ$Gf4z(BWVIOCR))cLE>Ti~ zZ4^6OPRoac;rVm-PrI=NY=;wL|k!kd|CawHgYvyd3vN%QJ~2wrF|6d4ZPq z1!nrpLsH)iLq?o$=FZ}odwiGzb`9Qu4}Q-fff;BP$IfBH!S*;Vzi=jk_=jQ(d<`^6 zn1BZMjicsHl~TwjOcv(O!WK3xDl?oZqGU)dP|=N%@sMj`f@Wkvsvp-{^w2rZwnqnk zSrT3IYiiMU|p)T!N~KWhx%|uV&C(ZJskVI9Tr zU60HT^=|<1{#;CM<}RN`-6&Y-10iV@J;?<#-s@kNFAP3UDKzw36Zmow>)y_kNfRir zGt{HOq5v&dh587}L1IR$glMvImmME^q_=5Ud}$=IstL~r&# z+hO(-(Sh9^4Bl808ZZpjNOU;i1$n>~c)@t_g`cg$RYALvqjHqP z4DK=bLyo}OY~ya79NZB%WdV9W^;sOaarkAJ#)YW;9WVM?vuyR{t7IoPpLF})H5WLt zJCL*Y*C@TlG`yaSfeXpV5rSI->M0upt%z>ByP4=G*r7lNqL243V}umTbs4UdkTLZ%|`^mV*KIZUUU)6bwBzW0A4V96%TW8mB_z z2+(VCVf7J!v%-og&|@XeiBab|w6Pu<88}o1$_+xzeE!m{5zLsM^wQc!2@4%~LYrLT z!*(ImPlEju?Nnq6!uo(=@*Ns=7F`OWsZ0Bb4gzgLV}P>tSR{7iguF3FnX{7SAf!0~ zX)fGouk^1+tEcIz_)vqEW=x2s1wd66PrU(cr^h8s4?w~)tzpyRX^v``%N^#Pj1gbv z;D%ZFn>#Yb8sJ}*Sx_npPLJB(*$fh1+JpVI1(e zf)Q%vgc^xr^Q#Q>m9T8=j3G57g|3(ZAtA0n@)0;k0jB`*kjxm#$ajGFEui%pVMA_w z;KCg*-RVb(hTtATWXK^tS!RgQmS8pEx(va*Jt78CR!HXdBVqwq*4w-WrWhe63Wd; zObn``VzS`&BTsg6OA^-_x%tSQO;MM4mJ^~8a|fqHF`$Y%D3kQut7Japclpd=9R*OF zr78`eaIW6@u-yOJf5f7+!vTN9sB=@wQMOVT%d8e)E@CXqyUV$Mm z{XBP;XbZ-Eb~QPJ051o`#|EULj2JS;rGbMLQqSB$E;KOVQW06o&Esq;-jz4aG#Qza z65m=s&GfSvp0nOJpy#0x%8*bD*Rott)2KhUGYq;0`H4bsI3>|rrJ^gMCDT!RGy2qK z^nCgV``(#R?&Py)x8gc?3HA^u<9MPOwI!dF zMA7$&-LT$?mO-xKQ)#9%=f0SRq>!WyV6nDjf@2C4@Q?yzD349=8-zF>gy0?m90DA! za~g+hap)D-kVoJi0UQM!t#{7gC)uC}>KLi>m3{JOPQi;^P{)OKWBV|s2=;Wjy)4&o z>u?DhGY$N_K8~mTkFeU8SpQ)CQq&%;Qqvi278RxVei=QDwC1ExEL_+M=<^QRt8oy{ zK{U)UX%Liv#gtm@Tih;vsJVL7Ts>+Ik6k~88XPmVIu184d86$bCkkto?G&h0dPPyI zHfAb^AKhTnVX}%0*WD@y`zz{TpfIzLjn6O0v{7dtq8%F7_C&|^t9N<*m4gv|bSAKV ztYQ1uX0Jm5OthRk@HH0!(K0i1AucRw3UmF}>~I);#cXXDOfPeo9TlVCII28t+*}?y z@EP)*yX(Orjs7J-k#HEr=eW+OhA7brMkdx(d1n9*9stbdNxgzLA3ikac{wAdCzKO(P}q1Js%>0l(Vno+qfCOo2Ydlx3*IJfP2y&N z2u3bP0{vzGk)-yAD2s1Yo16wBp03OtUiPHiYD+&=Sv1GqQ#RwhLw^DtXOs&HVhM(SNL+#?h=%)J(~Ry7#xMb~@L5*VJAqFOsnK zFo|+c_1122+0v zpScUkC0ZvkW!FC7k2Qm)pnp`y99Wc;DW>$cfH*3EK3>1NmCum&aegUr5Q)9>d@j5j zC7KR}{f}^@?|j%)g%0Xpq7-xABX?*xS$Qg(-kW?H8r@|3I`}ds&LA9MPQO2T^eBvz z1?TtldkI?yG2o0a%V^)y^|$WaL(3gS^1;+!vaACVF>lD%E-g(s_ouqTVUP2Bso2Vg z#l!#j#*AVLY9N};g$@N9#dsC^uzoqmw`55dpmHU*K6K?UDGAsb=zZhCJXyD3mcR;q z_%4kl%$o7zU)eM(cwH~xtQM*l^w@4ukw;DbvH`bc>1j_M{Z@3Nv|Z@H*`zxu*{^b{zZ|gl;!)6yB_t%l#*1?oowT-w^&wJF< z94JuT12J_{G+%iz5<9E9;F7AetPgALXY0)QQ8gHOzu+`rR=>E*1=d0jrAgb{AQOaP zT14M6;$3#}7ayF(4zhpl(iphhD9QU@HX}ljge4=oBOuHojs`2K2|INzVCVo(jS4G{c;M@Q`U#`ZfnPC%7Gs}NGjL@r=_VJ8 zyv!hySh+z|CPTxhoaqIogZ+q}xj|$JreXhyK~(GW4}znlYi=7)%5*NJ&#A%?CjOl* zPohREkZMX(O)FCZ{I6v|Lbt-(`z7cs*0=0^zkz1v^!;XI-$KDIQO+JSjuIW+`S8%u z^L}!YWgR`wllRZ$eMH{Bl=r;6pOW{#%KK?~FUb2jdH+V<&&%6_8hG@)74lvw?-S%* zChwEw&3~(S^gRAwm80jC%liy@pDFLx$h$(`ub1~5<$boi!}5NMyw8>Q`SMMoXiLQs~nxgBcbUj4Zlrt2egeV(o# z(KSNXC6It~AE#@Ot`E`mI9>0j>tc$`e!2$e>ZI#?bnT|=d`vKNH__EjR|8#tOV?F& zeS@xd&~*wY%5`+DrE4`^^>m$0*R6D&N!JJII+?CNql-PXb%L&^(J5NLN7oU$o}%l= zbcHz6&d|mG3^Vslx-O^d5xTCU>wdcafG&=wqxa5~mQJDsIIYbWe{^tv=pUd@Sq-Qe z+?2JxOXQYfvi;c@^5h;s+-gat@IdpvRCah(v%eWE+w1n~)wNXQM(G@hTKRb z4dhK=R(^j69~?`!zKWh@abSSD^#qZ3;q?x{X8=ow@FIa%!6aDv7!XFg3j1ZN{x=X8 zi2U{-nAGk(1+PB^lB4!K5K{XSgp_W@i`4#_$RJ)hYA%pF5hA}6uuc9pxE|zo`8)B- zQTw_1@f3hOGowg7&H2n)#r5Ar(~4BH?r z1pWq<_=F#F^xie47ZBOx=S;^pf%-3$6Mk0Heg97 zUJo-Be^uog@)bb-PZs=gZS#arkq+Few$`%l&cbUmQfQzknZM*K; zaH6ZNvp3!y?&$034eyGF;~k0KJHuVwaPMvF4Bye#+Zw)L?FHethwo_axjcLU80ks0 zxAj^VU9`5RyJhVyJzbp_wI&i)cYII0JKot6kM(xl8t=S0(bcmvylGSTg1axU;yo?R ziFmlBwYj^wg}K2qm*bwU_V%tj;=4^=l@~Dg3#u-#63yK`@mRdOyQ^E>VReU_d&Ab{ zVe4keTmRX48@RA8vUY~8yQ?e|`)=@)Xm18LJ@|j^`Y+5oyL!WWE|LPSUW2;B@7n6l z_O8~B=FZObyW_phZS7T61-V8RhhLVg7us|^nDZhAE&E%|@UT*TMl z;s1o^PjCq0!G6n{e*?xH`cLvg%Q_Rws`<5u3z)nZ0}gO#6Xe(*TGm)C{$Dj<@;z_^ zPPbduIv%3^|J_-25Kp*ikuZ@Aq`PLzx+ zn1G`tlCblR4lCdn+uPqCOT@cj-fD)q>(1Eil~z)=9+lB0R5rW+JgKvfmZU$Nfq=CvlEw zw6&l%yW2YV*0x-`Ef%b<2>dQLQ(|xA z`fImU#e)oooplG*yb@m?fUg8drjrId1g3?06$H_~v8mNZ#%Y-vjM4IV?H z9WAWNp<_X@IgyB?iBWWt96q)rrQjpSf@WK1PrSP~(qZ@xCp`#@y8t895XT(8!`PY@kB2+TA1Z%K7wo%#UPejS{X8y55d^Y`;X`S_jJbt zzHmFy0K=Aj+Tl#wAvDSDqr|;D_SykM}DYYif#Ey>9W`gsO znYJ6ZTQ{#JX??I{@f2-+xx`u#n0H4@@@*2azF4wy<@9e$td%S0@wQeJ zhW&F%AnErJc^=!s;9`Jm%WvY$**8{(D;vtg;qXLhxN_*6p|YCr(1}B5&YxL1TY`}J zH_xxAtSp~BAzU+G7EZ31EgdSIT@fByIbAk#a#*QY*2&oZJ-Z}nja2AhY1n=x&iypvw-|iL>{or;;QqeIvj(3x;h({7zof_h7}KAO4S(X?ha=u? zaDP8#*x>%Y%KsD``$lFTHa z=M2uhSmLK*T#z5`$r29(Kl@cl%ijl$7<_uahI0UiP@n!j>RX8;zQ2DLHMqY|hibDz zC!qe#zTZa-?(Y+RiFgS*1%Ax`t9W?eyawZj_}A4xCpW&YID_PO9$fmTcsziAD1d)H zfIkz!Bgjd;l-^JP=TWu7@HN=TF2rvS;D-Zv2z6Z;KNi407r>tk;ODF=PG1?oqXGO6 z0{EW<@V^h>j|T7`1@NZ>_=!B+B7a(6Jak@&Um3s~1y3T1{&og%{x5U{=GwbEkp4pf z{LcdTSO7m9z~wMb0l#w^p3Cpo0sPqj&I2+vcsAqNg6B#+Tk%BjT!rUqJRAcMhjLi! z8o)X{^?0tu^KLvix?pX?vmH+(o*VGoi05WJ@4>@^I}m?X6P{)~yYRdd&rUqM@x<}$ z!Gm87p5z9+uEW!Sry37_Nj?z4Rtuh23eRCHZ+lrrTj7d+u|g71D3*>?sDm)ZVr6Fu zGK(V>lUb~M#Y8U)s(UQhu`v-8PFO{?40{^Ab%dPB>6(N5~|5EA6D!G8ji&@0K$mD^n zjbS=Ztk@=<_b*)E=Y4Yn`gdE{FOe{}qR)pFE#;T=_WNyKg396@af?^&@CX>x>GQkU z(j{h*?HoR0)CHvRHWn*fa06S$GGIaqO#;R%ZwoLU0V9C%)zv?cCCu;s0^eWRFEYS` z*}S2m z2o7wdac?ak1-HkxSH`wip%XS-gTA;#4J>tF9&)Y?I|0iox$72-lxw%GTf|QBU|%Yu z+`xE63e)PoPQCGfEr?*pS*kQwu6y~aEw45(mzUl)+cpPgv84*MwbIDMYjW*lqUG7S z#Ab0L|;!UHU(R5jrHK(Q!LTk+ggi8|X zr&LaG==e7^F_GNX$6K9C>yk?^2t|3sl5j%|%Ht(Pg1ia?G|{|UPD5afBCq8LQzyNw zOOJYB0OuyATA%uQr4DI=Kdwu22AT*mEv&l0vZJahejCnI^~!0&NL$+rM7=oPwFk4o z)mL6=jz3^p+|?ehT;x-=Zn4+Ji@nw__S&%6t9p^wx=R;%UA&lyOR5&dx?~+o2QyGm T?I}{RMEETcDX$x0Bj(=$@v}rg From 84072fbd6d77619525f7ee4cdbc6cb7ca35db143 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 14 Mar 2017 16:07:28 +0100 Subject: [PATCH 079/530] :lipstick: fixed indentation --- test/src/unit-class_parser.cpp | 2 +- test/src/unit-regression.cpp | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/test/src/unit-class_parser.cpp b/test/src/unit-class_parser.cpp index 61aa1c37..7f2c9a95 100644 --- a/test/src/unit-class_parser.cpp +++ b/test/src/unit-class_parser.cpp @@ -279,7 +279,7 @@ TEST_CASE("parser class") // overflows during parsing yield an exception CHECK_THROWS_AS(json::parser("1.18973e+4932").parse() == json(), json::out_of_range); CHECK_THROWS_WITH(json::parser("1.18973e+4932").parse() == json(), - "[json.exception.out_of_range.406] number overflow parsing '1.18973e+4932'"); + "[json.exception.out_of_range.406] number overflow parsing '1.18973e+4932'"); } SECTION("invalid numbers") diff --git a/test/src/unit-regression.cpp b/test/src/unit-regression.cpp index 33f7a9ef..880f8325 100644 --- a/test/src/unit-regression.cpp +++ b/test/src/unit-regression.cpp @@ -592,7 +592,7 @@ TEST_CASE("regression tests") { CHECK_THROWS_AS(json::parse("22e2222"), json::out_of_range); CHECK_THROWS_WITH(json::parse("22e2222"), - "[json.exception.out_of_range.406] number overflow parsing '22e2222'"); + "[json.exception.out_of_range.406] number overflow parsing '22e2222'"); } SECTION("issue #366 - json::parse on failed stream gets stuck") From bfe4788e323ef91792af9fdd181a2ddb29f67487 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 14 Mar 2017 16:14:05 +0100 Subject: [PATCH 080/530] :ambulance: fix for #500 Removed a check that already failed in MSVC. --- test/src/unit-regression.cpp | 22 ---------------------- 1 file changed, 22 deletions(-) diff --git a/test/src/unit-regression.cpp b/test/src/unit-regression.cpp index 880f8325..558a0955 100644 --- a/test/src/unit-regression.cpp +++ b/test/src/unit-regression.cpp @@ -460,16 +460,6 @@ TEST_CASE("regression tests") { setlocale(LC_NUMERIC, "de_DE.UTF-8"); - // Verify that snprintf uses special decimal and grouping characters. - // Disabled, because can't trigger locale-specific behavior in AppVeyor -#ifndef _MSC_VER - { - std::array buf; - std::snprintf(buf.data(), buf.size(), "%.2f", 12345.67); - CHECK(strcmp(buf.data(), "12345,67") == 0); - } -#endif - // verify that dumped correctly with '.' and no grouping const json j1 = 12345.67; CHECK(json(12345.67).dump() == "12345.67"); @@ -480,18 +470,6 @@ TEST_CASE("regression tests") { setlocale(LC_NUMERIC, "de_DE.UTF-8"); - // disabled, because locale-specific beharivor is not - // triggered in AppVeyor for some reason -#ifndef _MSC_VER - { - // verify that strtod now uses commas as decimal-separator - CHECK(std::strtod("3,14", nullptr) == 3.14); - - // verify that strtod does not understand dots as decimal separator - CHECK(std::strtod("3.14", nullptr) == 3); - } -#endif - // verify that parsed correctly despite using strtod internally CHECK(json::parse("3.14").get() == 3.14); From b026591e9e566b3ecd80a0161bd845756986c4b3 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 14 Mar 2017 21:05:38 +0100 Subject: [PATCH 081/530] :ambulance: added special case to fuzzers to fix #504 MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Since #329, NaN and inf numbers do not yield an exception, but are stored internally and are dumped as “null”. This commit adjusts the fuzz testers to deal with this special case. --- src/json.hpp | 39 ++++++++++++++++++++++++++++++++-- src/json.hpp.re2c | 39 ++++++++++++++++++++++++++++++++-- test/src/fuzzer-parse_cbor.cpp | 4 ++-- test/src/unit-regression.cpp | 20 +++++++++++++++++ 4 files changed, 96 insertions(+), 6 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 5511dcd5..baa421fa 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -174,6 +174,7 @@ json.exception.parse_error.109 | parse error: array index 'one' is not a number json.exception.parse_error.110 | parse error at 1: cannot read 2 bytes from vector | When parsing CBOR or MessagePack, the byte vector ends before the complete value has been read. json.exception.parse_error.111 | parse error: bad input stream | Parsing CBOR or MessagePack from an input stream where the [`badbit` or `failbit`](http://en.cppreference.com/w/cpp/io/ios_base/iostate) is set. json.exception.parse_error.112 | parse error at 1: error reading CBOR; last byte: 0xf8 | Not all types of CBOR or MessagePack are supported. This exception occurs if an unsupported byte was read. +json.exception.parse_error.113 | | While parsing a map key, a value that is not a string has been read. @since version 3.0.0 */ @@ -8020,7 +8021,7 @@ class basic_json @param[in] v MessagePack serialization @param[in] idx byte index in @a v to check for a string - @throw std::invalid_argument if `v[idx]` does not belong to a string + @throw parse_error.113 if `v[idx]` does not belong to a string */ static void msgpack_expect_string(const std::vector& v, size_t idx) { @@ -8032,7 +8033,39 @@ class basic_json return; } - JSON_THROW(std::invalid_argument("error parsing a msgpack string @ " + std::to_string(idx) + ": " + std::to_string(static_cast(v[idx])))); + std::stringstream ss; + ss << std::hex << static_cast(v[idx]); + JSON_THROW(parse_error(113, idx + 1, "expected a MessagePack string; last byte: 0x" + ss.str())); + } + + /*! + @brief check if the next byte belongs to a string + + While parsing a map, the keys must be strings. This function checks if the + current byte is one of the start bytes for a string in CBOR: + + - 0x60 - 0x77: fixed length + - 0x78 - 0x7b: variable length + - 0x7f: indefinity length + + @param[in] v CBOR serialization + @param[in] idx byte index in @a v to check for a string + + @throw parse_error.113 if `v[idx]` does not belong to a string + */ + static void cbor_expect_string(const std::vector& v, size_t idx) + { + check_length(v.size(), 1, idx); + + const auto byte = v[idx]; + if ((byte >= 0x60 and byte <= 0x7b) or byte == 0x7f) + { + return; + } + + std::stringstream ss; + ss << std::hex << static_cast(v[idx]); + JSON_THROW(parse_error(113, idx + 1, "expected a CBOR string; last byte: 0x" + ss.str())); } /*! @@ -8046,6 +8079,7 @@ class basic_json @throw parse_error.110 if the given vector ends prematurely @throw parse_error.112 if unsupported features from MessagePack were used in the given vector @a v or if the input is not valid MessagePack + @throw parse_error.113 if a string was expected as map key, but not found @sa https://github.com/msgpack/msgpack/blob/master/spec.md */ @@ -8291,6 +8325,7 @@ class basic_json @throw parse_error.110 if the given vector ends prematurely @throw parse_error.112 if unsupported features from CBOR were used in the given vector @a v or if the input is not valid CBOR + @throw parse_error.113 if a string was expected as map key, but not found @sa https://tools.ietf.org/html/rfc7049 */ diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index f1173688..c7777057 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -174,6 +174,7 @@ json.exception.parse_error.109 | parse error: array index 'one' is not a number json.exception.parse_error.110 | parse error at 1: cannot read 2 bytes from vector | When parsing CBOR or MessagePack, the byte vector ends before the complete value has been read. json.exception.parse_error.111 | parse error: bad input stream | Parsing CBOR or MessagePack from an input stream where the [`badbit` or `failbit`](http://en.cppreference.com/w/cpp/io/ios_base/iostate) is set. json.exception.parse_error.112 | parse error at 1: error reading CBOR; last byte: 0xf8 | Not all types of CBOR or MessagePack are supported. This exception occurs if an unsupported byte was read. +json.exception.parse_error.113 | | While parsing a map key, a value that is not a string has been read. @since version 3.0.0 */ @@ -8020,7 +8021,7 @@ class basic_json @param[in] v MessagePack serialization @param[in] idx byte index in @a v to check for a string - @throw std::invalid_argument if `v[idx]` does not belong to a string + @throw parse_error.113 if `v[idx]` does not belong to a string */ static void msgpack_expect_string(const std::vector& v, size_t idx) { @@ -8032,7 +8033,39 @@ class basic_json return; } - JSON_THROW(std::invalid_argument("error parsing a msgpack string @ " + std::to_string(idx) + ": " + std::to_string(static_cast(v[idx])))); + std::stringstream ss; + ss << std::hex << static_cast(v[idx]); + JSON_THROW(parse_error(113, idx + 1, "expected a MessagePack string; last byte: 0x" + ss.str())); + } + + /*! + @brief check if the next byte belongs to a string + + While parsing a map, the keys must be strings. This function checks if the + current byte is one of the start bytes for a string in CBOR: + + - 0x60 - 0x77: fixed length + - 0x78 - 0x7b: variable length + - 0x7f: indefinity length + + @param[in] v CBOR serialization + @param[in] idx byte index in @a v to check for a string + + @throw parse_error.113 if `v[idx]` does not belong to a string + */ + static void cbor_expect_string(const std::vector& v, size_t idx) + { + check_length(v.size(), 1, idx); + + const auto byte = v[idx]; + if ((byte >= 0x60 and byte <= 0x7b) or byte == 0x7f) + { + return; + } + + std::stringstream ss; + ss << std::hex << static_cast(v[idx]); + JSON_THROW(parse_error(113, idx + 1, "expected a CBOR string; last byte: 0x" + ss.str())); } /*! @@ -8046,6 +8079,7 @@ class basic_json @throw parse_error.110 if the given vector ends prematurely @throw parse_error.112 if unsupported features from MessagePack were used in the given vector @a v or if the input is not valid MessagePack + @throw parse_error.113 if a string was expected as map key, but not found @sa https://github.com/msgpack/msgpack/blob/master/spec.md */ @@ -8291,6 +8325,7 @@ class basic_json @throw parse_error.110 if the given vector ends prematurely @throw parse_error.112 if unsupported features from CBOR were used in the given vector @a v or if the input is not valid CBOR + @throw parse_error.113 if a string was expected as map key, but not found @sa https://tools.ietf.org/html/rfc7049 */ diff --git a/test/src/fuzzer-parse_cbor.cpp b/test/src/fuzzer-parse_cbor.cpp index 6e7ae636..4bcfc7ef 100644 --- a/test/src/fuzzer-parse_cbor.cpp +++ b/test/src/fuzzer-parse_cbor.cpp @@ -41,8 +41,8 @@ extern "C" int LLVMFuzzerTestOneInput(const uint8_t* data, size_t size) // parse serialization json j2 = json::from_cbor(vec2); - // deserializations must match - assert(j1 == j2); + // serializations must match + assert(json::to_cbor(j2) == vec2); } catch (const json::parse_error&) { diff --git a/test/src/unit-regression.cpp b/test/src/unit-regression.cpp index 558a0955..bc45cb65 100644 --- a/test/src/unit-regression.cpp +++ b/test/src/unit-regression.cpp @@ -909,4 +909,24 @@ TEST_CASE("regression tests") CHECK(j["bool_vector"].dump() == "[false,true,false,false]"); } + + SECTION("issue #504 - assertion error (OSS-Fuzz 856)") + { + std::vector vec1 = {0xf9, 0xff, 0xff, 0x4a, 0x3a, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x01, 0x37, 0x02, 0x38}; + json j1 = json::from_cbor(vec1); + + // step 2: round trip + std::vector vec2 = json::to_cbor(j1); + + // parse serialization + json j2 = json::from_cbor(vec2); + + // NaN is dumped to "null" + CHECK(j2.is_number_float()); + CHECK(std::isnan(j2.get())); + CHECK(j2.dump() == "null"); + + // check if serializations match + CHECK(json::to_cbor(j2) == vec2); + } } From 95474e420de25ec7db4875e889201229820ef564 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 14 Mar 2017 21:11:20 +0100 Subject: [PATCH 082/530] :checkered_flag: added parentheses around max/min calls #506 When is included with MSVC, a macro NOMINMAX is defined that yields compilation errors when max/min calls are encountered. This can be fixed by un-defining NOMINMAX, or by placing parentheses around all min/max calls. We chose the latter. --- src/json.hpp | 38 +++++++++++++++++++------------------- src/json.hpp.re2c | 38 +++++++++++++++++++------------------- 2 files changed, 38 insertions(+), 38 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index baa421fa..847aa3e2 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -7494,25 +7494,25 @@ class basic_json // positive fixnum add_to_vector(v, 1, j.m_value.number_unsigned); } - else if (j.m_value.number_unsigned <= std::numeric_limits::max()) + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { // uint 8 v.push_back(0xcc); add_to_vector(v, 1, j.m_value.number_unsigned); } - else if (j.m_value.number_unsigned <= std::numeric_limits::max()) + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { // uint 16 v.push_back(0xcd); add_to_vector(v, 2, j.m_value.number_unsigned); } - else if (j.m_value.number_unsigned <= std::numeric_limits::max()) + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { // uint 32 v.push_back(0xce); add_to_vector(v, 4, j.m_value.number_unsigned); } - else if (j.m_value.number_unsigned <= std::numeric_limits::max()) + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { // uint 64 v.push_back(0xcf); @@ -7526,25 +7526,25 @@ class basic_json // negative fixnum add_to_vector(v, 1, j.m_value.number_integer); } - else if (j.m_value.number_integer >= std::numeric_limits::min() and j.m_value.number_integer <= std::numeric_limits::max()) + else if (j.m_value.number_integer >= (std::numeric_limits::min)() and j.m_value.number_integer <= (std::numeric_limits::max)()) { // int 8 v.push_back(0xd0); add_to_vector(v, 1, j.m_value.number_integer); } - else if (j.m_value.number_integer >= std::numeric_limits::min() and j.m_value.number_integer <= std::numeric_limits::max()) + else if (j.m_value.number_integer >= (std::numeric_limits::min)() and j.m_value.number_integer <= (std::numeric_limits::max)()) { // int 16 v.push_back(0xd1); add_to_vector(v, 2, j.m_value.number_integer); } - else if (j.m_value.number_integer >= std::numeric_limits::min() and j.m_value.number_integer <= std::numeric_limits::max()) + else if (j.m_value.number_integer >= (std::numeric_limits::min)() and j.m_value.number_integer <= (std::numeric_limits::max)()) { // int 32 v.push_back(0xd2); add_to_vector(v, 4, j.m_value.number_integer); } - else if (j.m_value.number_integer >= std::numeric_limits::min() and j.m_value.number_integer <= std::numeric_limits::max()) + else if (j.m_value.number_integer >= (std::numeric_limits::min)() and j.m_value.number_integer <= (std::numeric_limits::max)()) { // int 64 v.push_back(0xd3); @@ -7561,25 +7561,25 @@ class basic_json // positive fixnum add_to_vector(v, 1, j.m_value.number_unsigned); } - else if (j.m_value.number_unsigned <= std::numeric_limits::max()) + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { // uint 8 v.push_back(0xcc); add_to_vector(v, 1, j.m_value.number_unsigned); } - else if (j.m_value.number_unsigned <= std::numeric_limits::max()) + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { // uint 16 v.push_back(0xcd); add_to_vector(v, 2, j.m_value.number_unsigned); } - else if (j.m_value.number_unsigned <= std::numeric_limits::max()) + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { // uint 32 v.push_back(0xce); add_to_vector(v, 4, j.m_value.number_unsigned); } - else if (j.m_value.number_unsigned <= std::numeric_limits::max()) + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { // uint 64 v.push_back(0xcf); @@ -7736,19 +7736,19 @@ class basic_json { add_to_vector(v, 1, j.m_value.number_integer); } - else if (j.m_value.number_integer <= std::numeric_limits::max()) + else if (j.m_value.number_integer <= (std::numeric_limits::max)()) { v.push_back(0x18); // one-byte uint8_t add_to_vector(v, 1, j.m_value.number_integer); } - else if (j.m_value.number_integer <= std::numeric_limits::max()) + else if (j.m_value.number_integer <= (std::numeric_limits::max)()) { v.push_back(0x19); // two-byte uint16_t add_to_vector(v, 2, j.m_value.number_integer); } - else if (j.m_value.number_integer <= std::numeric_limits::max()) + else if (j.m_value.number_integer <= (std::numeric_limits::max)()) { v.push_back(0x1a); // four-byte uint32_t @@ -7770,19 +7770,19 @@ class basic_json { v.push_back(static_cast(0x20 + positive_number)); } - else if (positive_number <= std::numeric_limits::max()) + else if (positive_number <= (std::numeric_limits::max)()) { // int 8 v.push_back(0x38); add_to_vector(v, 1, positive_number); } - else if (positive_number <= std::numeric_limits::max()) + else if (positive_number <= (std::numeric_limits::max)()) { // int 16 v.push_back(0x39); add_to_vector(v, 2, positive_number); } - else if (positive_number <= std::numeric_limits::max()) + else if (positive_number <= (std::numeric_limits::max)()) { // int 32 v.push_back(0x3a); @@ -7995,7 +7995,7 @@ class basic_json } // second case: adding offset would result in overflow - if ((size > (std::numeric_limits::max() - offset))) + if ((size > ((std::numeric_limits::max)() - offset))) { JSON_THROW(parse_error(110, offset + 1, "cannot read " + std::to_string(len) + " bytes from vector")); } diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index c7777057..e78df080 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -7494,25 +7494,25 @@ class basic_json // positive fixnum add_to_vector(v, 1, j.m_value.number_unsigned); } - else if (j.m_value.number_unsigned <= std::numeric_limits::max()) + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { // uint 8 v.push_back(0xcc); add_to_vector(v, 1, j.m_value.number_unsigned); } - else if (j.m_value.number_unsigned <= std::numeric_limits::max()) + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { // uint 16 v.push_back(0xcd); add_to_vector(v, 2, j.m_value.number_unsigned); } - else if (j.m_value.number_unsigned <= std::numeric_limits::max()) + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { // uint 32 v.push_back(0xce); add_to_vector(v, 4, j.m_value.number_unsigned); } - else if (j.m_value.number_unsigned <= std::numeric_limits::max()) + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { // uint 64 v.push_back(0xcf); @@ -7526,25 +7526,25 @@ class basic_json // negative fixnum add_to_vector(v, 1, j.m_value.number_integer); } - else if (j.m_value.number_integer >= std::numeric_limits::min() and j.m_value.number_integer <= std::numeric_limits::max()) + else if (j.m_value.number_integer >= (std::numeric_limits::min)() and j.m_value.number_integer <= (std::numeric_limits::max)()) { // int 8 v.push_back(0xd0); add_to_vector(v, 1, j.m_value.number_integer); } - else if (j.m_value.number_integer >= std::numeric_limits::min() and j.m_value.number_integer <= std::numeric_limits::max()) + else if (j.m_value.number_integer >= (std::numeric_limits::min)() and j.m_value.number_integer <= (std::numeric_limits::max)()) { // int 16 v.push_back(0xd1); add_to_vector(v, 2, j.m_value.number_integer); } - else if (j.m_value.number_integer >= std::numeric_limits::min() and j.m_value.number_integer <= std::numeric_limits::max()) + else if (j.m_value.number_integer >= (std::numeric_limits::min)() and j.m_value.number_integer <= (std::numeric_limits::max)()) { // int 32 v.push_back(0xd2); add_to_vector(v, 4, j.m_value.number_integer); } - else if (j.m_value.number_integer >= std::numeric_limits::min() and j.m_value.number_integer <= std::numeric_limits::max()) + else if (j.m_value.number_integer >= (std::numeric_limits::min)() and j.m_value.number_integer <= (std::numeric_limits::max)()) { // int 64 v.push_back(0xd3); @@ -7561,25 +7561,25 @@ class basic_json // positive fixnum add_to_vector(v, 1, j.m_value.number_unsigned); } - else if (j.m_value.number_unsigned <= std::numeric_limits::max()) + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { // uint 8 v.push_back(0xcc); add_to_vector(v, 1, j.m_value.number_unsigned); } - else if (j.m_value.number_unsigned <= std::numeric_limits::max()) + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { // uint 16 v.push_back(0xcd); add_to_vector(v, 2, j.m_value.number_unsigned); } - else if (j.m_value.number_unsigned <= std::numeric_limits::max()) + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { // uint 32 v.push_back(0xce); add_to_vector(v, 4, j.m_value.number_unsigned); } - else if (j.m_value.number_unsigned <= std::numeric_limits::max()) + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { // uint 64 v.push_back(0xcf); @@ -7736,19 +7736,19 @@ class basic_json { add_to_vector(v, 1, j.m_value.number_integer); } - else if (j.m_value.number_integer <= std::numeric_limits::max()) + else if (j.m_value.number_integer <= (std::numeric_limits::max)()) { v.push_back(0x18); // one-byte uint8_t add_to_vector(v, 1, j.m_value.number_integer); } - else if (j.m_value.number_integer <= std::numeric_limits::max()) + else if (j.m_value.number_integer <= (std::numeric_limits::max)()) { v.push_back(0x19); // two-byte uint16_t add_to_vector(v, 2, j.m_value.number_integer); } - else if (j.m_value.number_integer <= std::numeric_limits::max()) + else if (j.m_value.number_integer <= (std::numeric_limits::max)()) { v.push_back(0x1a); // four-byte uint32_t @@ -7770,19 +7770,19 @@ class basic_json { v.push_back(static_cast(0x20 + positive_number)); } - else if (positive_number <= std::numeric_limits::max()) + else if (positive_number <= (std::numeric_limits::max)()) { // int 8 v.push_back(0x38); add_to_vector(v, 1, positive_number); } - else if (positive_number <= std::numeric_limits::max()) + else if (positive_number <= (std::numeric_limits::max)()) { // int 16 v.push_back(0x39); add_to_vector(v, 2, positive_number); } - else if (positive_number <= std::numeric_limits::max()) + else if (positive_number <= (std::numeric_limits::max)()) { // int 32 v.push_back(0x3a); @@ -7995,7 +7995,7 @@ class basic_json } // second case: adding offset would result in overflow - if ((size > (std::numeric_limits::max() - offset))) + if ((size > ((std::numeric_limits::max)() - offset))) { JSON_THROW(parse_error(110, offset + 1, "cannot read " + std::to_string(len) + " bytes from vector")); } From 1de80e8af4ae6ac2840d41efb2f960ee95035303 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 14 Mar 2017 21:31:36 +0100 Subject: [PATCH 083/530] :hammer: added user-defined exception #493 Replaced old std::invalid_argument exception by parse_error.111 to have unified exceptions in case of input stream errors. --- src/json.hpp | 2 +- src/json.hpp.re2c | 2 +- test/src/unit-regression.cpp | 6 ++++-- 3 files changed, 6 insertions(+), 4 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 3fedc045..5a9399fd 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -11334,7 +11334,7 @@ basic_json_parser_74: // check if stream is still good if (m_stream->fail()) { - JSON_THROW(std::invalid_argument("stream error")); + JSON_THROW(parse_error(111, 0, "bad input stream")); } std::getline(*m_stream, m_line_buffer_tmp, '\n'); diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 7e7e16e3..e2dd777b 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -10367,7 +10367,7 @@ class basic_json // check if stream is still good if (m_stream->fail()) { - JSON_THROW(std::invalid_argument("stream error")); + JSON_THROW(parse_error(111, 0, "bad input stream")); } std::getline(*m_stream, m_line_buffer_tmp, '\n'); diff --git a/test/src/unit-regression.cpp b/test/src/unit-regression.cpp index ce984b0d..3f17a917 100644 --- a/test/src/unit-regression.cpp +++ b/test/src/unit-regression.cpp @@ -924,7 +924,8 @@ TEST_CASE("regression tests") l.m_stream->setstate(std::ios_base::failbit); - CHECK_THROWS_AS(l.fill_line_buffer(), std::invalid_argument); + CHECK_THROWS_AS(l.fill_line_buffer(), json::parse_error); + CHECK_THROWS_WITH(l.fill_line_buffer(), "[json.exception.parse_error.111] parse error: bad input stream"); } SECTION("setting badbit") @@ -938,7 +939,8 @@ TEST_CASE("regression tests") l.m_stream->setstate(std::ios_base::badbit); - CHECK_THROWS_AS(l.fill_line_buffer(), std::invalid_argument); + CHECK_THROWS_AS(l.fill_line_buffer(), json::parse_error); + CHECK_THROWS_WITH(l.fill_line_buffer(), "[json.exception.parse_error.111] parse error: bad input stream"); } } From 710f8a2686034f274be771be9fee86c08d365d41 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 14 Mar 2017 23:08:05 +0100 Subject: [PATCH 084/530] :memo: improved documentation for to-CBOR and to-msgpack functions --- src/json.hpp | 108 ++++++++++++++++++++++++++++++++++++++++++++++ src/json.hpp.re2c | 108 ++++++++++++++++++++++++++++++++++++++++++++++ 2 files changed, 216 insertions(+) diff --git a/src/json.hpp b/src/json.hpp index 5a9399fd..cbf12dc5 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -8815,6 +8815,55 @@ class basic_json serialization format. MessagePack is a binary serialization format which aims to be more compact than JSON itself, yet more efficient to parse. + The library uses the following mapping from JSON values types to + MessagePack types according to the MessagePack specification: + + JSON value type | value/range | MessagePack type | first byte + --------------- | --------------------------------- | ---------------- | ---------- + null | `null` | nil | 0xc0 + boolean | `true` | true | 0xc3 + boolean | `false` | false | 0xc2 + number_integer | -9223372036854775808..-2147483649 | int64 | 0xd3 + number_integer | -2147483648..-32769 | int32 | 0xd2 + number_integer | -32768..-129 | int16 | 0xd1 + number_integer | -128..-33 | int8 | 0xd0 + number_integer | -32..-1 | negative fixint | 0xe0..0xff + number_integer | 0..127 | positive fixint | 0x00..0x7f + number_integer | 128..255 | uint 8 | 0xcc + number_integer | 256..65535 | uint 16 | 0xcd + number_integer | 65536..4294967295 | uint 32 | 0xce + number_integer | 4294967296..18446744073709551615 | uint 64 | 0xcf + number_unsigned | 0..127 | positive fixint | 0x00..0x7f + number_unsigned | 128..255 | uint 8 | 0xcc + number_unsigned | 256..65535 | uint 16 | 0xcd + number_unsigned | 65536..4294967295 | uint 32 | 0xce + number_unsigned | 4294967296..18446744073709551615 | uint 64 | 0xcf + number_float | *any value* | float 64 | 0xcb + string | *length*: 0..31 | fixstr | 0xa0..0xbf + string | *length*: 32..255 | str 8 | 0xd9 + string | *length*: 256..65535 | str 16 | 0xda + string | *length*: 65536..4294967295 | str 32 | 0xdb + array | *size*: 0..15 | fixarray | 0x90..0x9f + array | *size*: 16..65535 | array 16 | 0xdc + array | *size*: 65536..4294967295 | array 32 | 0xdd + object | *size*: 0..15 | fix map | 0x80..0x8f + object | *size*: 16..65535 | map 16 | 0xde + object | *size*: 65536..4294967295 | map 32 | 0xdf + + @note The mapping is **complete** in the sense that any JSON value type + can be converted to a MessagePack value. + + @note The following values can **not** be converted to a MessagePack value: + - strings with more than 4294967295 bytes + - arrays with more than 4294967295 elements + - objects with more than 4294967295 elements + + @note The following MessagePack types are not used in the conversion: + - bin 8 - bin 32 (0xc4..0xc6) + - ext 8 - ext 32 (0xc7..0xc9) + - float 32 (0xca) + - fixext 1 - fixext 16 (0xd4..0xd8) + @param[in] j JSON value to serialize @return MessagePack serialization as byte vector @@ -8878,6 +8927,65 @@ class basic_json serialization format which aims to be more compact than JSON itself, yet more efficient to parse. + The library uses the following mapping from JSON values types to + CBOR types according to the CBOR specification (RFC 7049): + + JSON value type | value/range | CBOR type | first byte + --------------- | ------------------------------------------ | ---------------------------------- | --------------- + null | `null` | Null | 0xf6 + boolean | `true` | True | 0xf5 + boolean | `false` | False | 0xf4 + number_integer | -9223372036854775808..-2147483649 | Negative integer (8 bytes follow) | 0x3b + number_integer | -2147483648..-32769 | Negative integer (4 bytes follow) | 0x3a + number_integer | -32768..-129 | Negative integer (2 bytes follow) | 0x39 + number_integer | -128..-25 | Negative integer (1 byte follow) | 0x38 + number_integer | -24..-1 | Negative integer | 0x20..0x37 + number_integer | 0..23 | Integer | 0x00..0x17 + number_integer | 24..255 | Unsigned integer (1 byte follow) | 0x18 + number_integer | 256..65535 | Unsigned integer (2 bytes follow) | 0x19 + number_integer | 65536..4294967295 | Unsigned integer (4 bytes follow) | 0x1a + number_integer | 4294967296..18446744073709551615 | Unsigned integer (8 bytes follow) | 0x1b + number_unsigned | 0..23 | Integer | 0x00..0x17 + number_unsigned | 24..255 | Unsigned integer (1 byte follow) | 0x18 + number_unsigned | 256..65535 | Unsigned integer (2 bytes follow) | 0x19 + number_unsigned | 65536..4294967295 | Unsigned integer (4 bytes follow) | 0x1a + number_unsigned | 4294967296..18446744073709551615 | Unsigned integer (8 bytes follow) | 0x1b + number_float | *any value* | Double-Precision Float | 0xfb + string | *length*: 0..23 | UTF-8 string | 0x60..0x77 + string | *length*: 23..255 | UTF-8 string (1 byte follow) | 0x78 + string | *length*: 256..65535 | UTF-8 string (2 bytes follow) | 0x79 + string | *length*: 65536..4294967295 | UTF-8 string (4 bytes follow) | 0x7a + string | *length*: 4294967296..18446744073709551615 | UTF-8 string (8 bytes follow) | 0x7b + array | *size*: 0..23 | array | 0x80..0x97 + array | *size*: 23..255 | array (1 byte follow) | 0x98 + array | *size*: 256..65535 | array (2 bytes follow) | 0x99 + array | *size*: 65536..4294967295 | array (4 bytes follow) | 0x9a + array | *size*: 4294967296..18446744073709551615 | array (8 bytes follow) | 0x9b + object | *size*: 0..23 | map | 0xa0..0xb7 + object | *size*: 23..255 | map (1 byte follow) | 0xb8 + object | *size*: 256..65535 | map (2 bytes follow) | 0xb9 + object | *size*: 65536..4294967295 | map (4 bytes follow) | 0xba + object | *size*: 4294967296..18446744073709551615 | map (8 bytes follow) | 0xbb + + @note The mapping is **complete** in the sense that any JSON value type + can be converted to a CBOR value. + + @note The following CBOR types are not used in the conversion: + - byte strings (0x40..0x5f) + - UTF-8 strings terminated by "break" (0x7f) + - arrays terminated by "break" (0x9f) + - maps terminated by "break" (0xbf) + - date/time (0xc0..0xc1) + - bignum (0xc2..0xc3) + - decimal fraction (0xc4) + - bigfloat (0xc5) + - tagged items (0xc6..0xd4, 0xd8..0xdb) + - expected conversions (0xd5..0xd7) + - simple values (0xe0..0xf3, 0xf8) + - undefined (0xf7) + - half and single-precision floats (0xf9-0xfa) + - break (0xff) + @param[in] j JSON value to serialize @return MessagePack serialization as byte vector diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index e2dd777b..c4d142b7 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -8815,6 +8815,55 @@ class basic_json serialization format. MessagePack is a binary serialization format which aims to be more compact than JSON itself, yet more efficient to parse. + The library uses the following mapping from JSON values types to + MessagePack types according to the MessagePack specification: + + JSON value type | value/range | MessagePack type | first byte + --------------- | --------------------------------- | ---------------- | ---------- + null | `null` | nil | 0xc0 + boolean | `true` | true | 0xc3 + boolean | `false` | false | 0xc2 + number_integer | -9223372036854775808..-2147483649 | int64 | 0xd3 + number_integer | -2147483648..-32769 | int32 | 0xd2 + number_integer | -32768..-129 | int16 | 0xd1 + number_integer | -128..-33 | int8 | 0xd0 + number_integer | -32..-1 | negative fixint | 0xe0..0xff + number_integer | 0..127 | positive fixint | 0x00..0x7f + number_integer | 128..255 | uint 8 | 0xcc + number_integer | 256..65535 | uint 16 | 0xcd + number_integer | 65536..4294967295 | uint 32 | 0xce + number_integer | 4294967296..18446744073709551615 | uint 64 | 0xcf + number_unsigned | 0..127 | positive fixint | 0x00..0x7f + number_unsigned | 128..255 | uint 8 | 0xcc + number_unsigned | 256..65535 | uint 16 | 0xcd + number_unsigned | 65536..4294967295 | uint 32 | 0xce + number_unsigned | 4294967296..18446744073709551615 | uint 64 | 0xcf + number_float | *any value* | float 64 | 0xcb + string | *length*: 0..31 | fixstr | 0xa0..0xbf + string | *length*: 32..255 | str 8 | 0xd9 + string | *length*: 256..65535 | str 16 | 0xda + string | *length*: 65536..4294967295 | str 32 | 0xdb + array | *size*: 0..15 | fixarray | 0x90..0x9f + array | *size*: 16..65535 | array 16 | 0xdc + array | *size*: 65536..4294967295 | array 32 | 0xdd + object | *size*: 0..15 | fix map | 0x80..0x8f + object | *size*: 16..65535 | map 16 | 0xde + object | *size*: 65536..4294967295 | map 32 | 0xdf + + @note The mapping is **complete** in the sense that any JSON value type + can be converted to a MessagePack value. + + @note The following values can **not** be converted to a MessagePack value: + - strings with more than 4294967295 bytes + - arrays with more than 4294967295 elements + - objects with more than 4294967295 elements + + @note The following MessagePack types are not used in the conversion: + - bin 8 - bin 32 (0xc4..0xc6) + - ext 8 - ext 32 (0xc7..0xc9) + - float 32 (0xca) + - fixext 1 - fixext 16 (0xd4..0xd8) + @param[in] j JSON value to serialize @return MessagePack serialization as byte vector @@ -8878,6 +8927,65 @@ class basic_json serialization format which aims to be more compact than JSON itself, yet more efficient to parse. + The library uses the following mapping from JSON values types to + CBOR types according to the CBOR specification (RFC 7049): + + JSON value type | value/range | CBOR type | first byte + --------------- | ------------------------------------------ | ---------------------------------- | --------------- + null | `null` | Null | 0xf6 + boolean | `true` | True | 0xf5 + boolean | `false` | False | 0xf4 + number_integer | -9223372036854775808..-2147483649 | Negative integer (8 bytes follow) | 0x3b + number_integer | -2147483648..-32769 | Negative integer (4 bytes follow) | 0x3a + number_integer | -32768..-129 | Negative integer (2 bytes follow) | 0x39 + number_integer | -128..-25 | Negative integer (1 byte follow) | 0x38 + number_integer | -24..-1 | Negative integer | 0x20..0x37 + number_integer | 0..23 | Integer | 0x00..0x17 + number_integer | 24..255 | Unsigned integer (1 byte follow) | 0x18 + number_integer | 256..65535 | Unsigned integer (2 bytes follow) | 0x19 + number_integer | 65536..4294967295 | Unsigned integer (4 bytes follow) | 0x1a + number_integer | 4294967296..18446744073709551615 | Unsigned integer (8 bytes follow) | 0x1b + number_unsigned | 0..23 | Integer | 0x00..0x17 + number_unsigned | 24..255 | Unsigned integer (1 byte follow) | 0x18 + number_unsigned | 256..65535 | Unsigned integer (2 bytes follow) | 0x19 + number_unsigned | 65536..4294967295 | Unsigned integer (4 bytes follow) | 0x1a + number_unsigned | 4294967296..18446744073709551615 | Unsigned integer (8 bytes follow) | 0x1b + number_float | *any value* | Double-Precision Float | 0xfb + string | *length*: 0..23 | UTF-8 string | 0x60..0x77 + string | *length*: 23..255 | UTF-8 string (1 byte follow) | 0x78 + string | *length*: 256..65535 | UTF-8 string (2 bytes follow) | 0x79 + string | *length*: 65536..4294967295 | UTF-8 string (4 bytes follow) | 0x7a + string | *length*: 4294967296..18446744073709551615 | UTF-8 string (8 bytes follow) | 0x7b + array | *size*: 0..23 | array | 0x80..0x97 + array | *size*: 23..255 | array (1 byte follow) | 0x98 + array | *size*: 256..65535 | array (2 bytes follow) | 0x99 + array | *size*: 65536..4294967295 | array (4 bytes follow) | 0x9a + array | *size*: 4294967296..18446744073709551615 | array (8 bytes follow) | 0x9b + object | *size*: 0..23 | map | 0xa0..0xb7 + object | *size*: 23..255 | map (1 byte follow) | 0xb8 + object | *size*: 256..65535 | map (2 bytes follow) | 0xb9 + object | *size*: 65536..4294967295 | map (4 bytes follow) | 0xba + object | *size*: 4294967296..18446744073709551615 | map (8 bytes follow) | 0xbb + + @note The mapping is **complete** in the sense that any JSON value type + can be converted to a CBOR value. + + @note The following CBOR types are not used in the conversion: + - byte strings (0x40..0x5f) + - UTF-8 strings terminated by "break" (0x7f) + - arrays terminated by "break" (0x9f) + - maps terminated by "break" (0xbf) + - date/time (0xc0..0xc1) + - bignum (0xc2..0xc3) + - decimal fraction (0xc4) + - bigfloat (0xc5) + - tagged items (0xc6..0xd4, 0xd8..0xdb) + - expected conversions (0xd5..0xd7) + - simple values (0xe0..0xf3, 0xf8) + - undefined (0xf7) + - half and single-precision floats (0xf9-0xfa) + - break (0xff) + @param[in] j JSON value to serialize @return MessagePack serialization as byte vector From dbebf8de47b285a0da5d5a14d7dc43eeaffb31cb Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Th=C3=A9o=20DELRIEU?= Date: Thu, 16 Mar 2017 13:43:53 +0100 Subject: [PATCH 085/530] add to_json method for C arrays If the array can be converted to basic_json::string_t, the overload in this commit is not chosen. --- src/json.hpp | 1521 +++++++++++---------------------- src/json.hpp.re2c | 7 + test/src/unit-conversions.cpp | 14 + 3 files changed, 511 insertions(+), 1031 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index cbf12dc5..60618c3b 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -800,6 +800,13 @@ void to_json(BasicJsonType& j, const CompatibleObjectType& arr) external_constructor::construct(j, arr); } +template ::value, + int> = 0> +void to_json(BasicJsonType &j, T (&arr)[N]) { + external_constructor::construct(j, arr); +} /////////////// // from_json // @@ -10326,1038 +10333,490 @@ class basic_json m_start = m_cursor; assert(m_start != nullptr); - - { - lexer_char_t yych; - unsigned int yyaccept = 0; - static const unsigned char yybm[] = - { - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 32, 32, 0, 0, 32, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 160, 128, 0, 128, 128, 128, 128, 128, - 128, 128, 128, 128, 128, 128, 128, 128, - 192, 192, 192, 192, 192, 192, 192, 192, - 192, 192, 128, 128, 128, 128, 128, 128, - 128, 128, 128, 128, 128, 128, 128, 128, - 128, 128, 128, 128, 128, 128, 128, 128, - 128, 128, 128, 128, 128, 128, 128, 128, - 128, 128, 128, 128, 0, 128, 128, 128, - 128, 128, 128, 128, 128, 128, 128, 128, - 128, 128, 128, 128, 128, 128, 128, 128, - 128, 128, 128, 128, 128, 128, 128, 128, - 128, 128, 128, 128, 128, 128, 128, 128, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - }; - if ((m_limit - m_cursor) < 5) - { - fill_line_buffer(5); // LCOV_EXCL_LINE - } - yych = *m_cursor; - if (yybm[0 + yych] & 32) - { - goto basic_json_parser_6; - } - if (yych <= '[') - { - if (yych <= '-') - { - if (yych <= '"') - { - if (yych <= 0x00) - { - goto basic_json_parser_2; - } - if (yych <= '!') - { - goto basic_json_parser_4; - } - goto basic_json_parser_9; - } - else - { - if (yych <= '+') - { - goto basic_json_parser_4; - } - if (yych <= ',') - { - goto basic_json_parser_10; - } - goto basic_json_parser_12; - } - } - else - { - if (yych <= '9') - { - if (yych <= '/') - { - goto basic_json_parser_4; - } - if (yych <= '0') - { - goto basic_json_parser_13; - } - goto basic_json_parser_15; - } - else - { - if (yych <= ':') - { - goto basic_json_parser_17; - } - if (yych <= 'Z') - { - goto basic_json_parser_4; - } - goto basic_json_parser_19; - } - } - } - else - { - if (yych <= 'n') - { - if (yych <= 'e') - { - if (yych == ']') - { - goto basic_json_parser_21; - } - goto basic_json_parser_4; - } - else - { - if (yych <= 'f') - { - goto basic_json_parser_23; - } - if (yych <= 'm') - { - goto basic_json_parser_4; - } - goto basic_json_parser_24; - } - } - else - { - if (yych <= 'z') - { - if (yych == 't') - { - goto basic_json_parser_25; - } - goto basic_json_parser_4; - } - else - { - if (yych <= '{') - { - goto basic_json_parser_26; - } - if (yych == '}') - { - goto basic_json_parser_28; - } - goto basic_json_parser_4; - } - } - } -basic_json_parser_2: - ++m_cursor; - { - last_token_type = token_type::end_of_input; - break; - } -basic_json_parser_4: - ++m_cursor; -basic_json_parser_5: - { - last_token_type = token_type::parse_error; - break; - } -basic_json_parser_6: - ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } - yych = *m_cursor; - if (yybm[0 + yych] & 32) - { - goto basic_json_parser_6; - } - { - position += static_cast((m_cursor - m_start)); - continue; - } -basic_json_parser_9: - yyaccept = 0; - yych = *(m_marker = ++m_cursor); - if (yych <= 0x1F) - { - goto basic_json_parser_5; - } - if (yych <= 0x7F) - { - goto basic_json_parser_31; - } - if (yych <= 0xC1) - { - goto basic_json_parser_5; - } - if (yych <= 0xF4) - { - goto basic_json_parser_31; - } - goto basic_json_parser_5; -basic_json_parser_10: - ++m_cursor; - { - last_token_type = token_type::value_separator; - break; - } -basic_json_parser_12: - yych = *++m_cursor; - if (yych <= '/') - { - goto basic_json_parser_5; - } - if (yych <= '0') - { - goto basic_json_parser_43; - } - if (yych <= '9') - { - goto basic_json_parser_45; - } - goto basic_json_parser_5; -basic_json_parser_13: - yyaccept = 1; - yych = *(m_marker = ++m_cursor); - if (yych <= '9') - { - if (yych == '.') - { - goto basic_json_parser_47; - } - if (yych >= '0') - { - goto basic_json_parser_48; - } - } - else - { - if (yych <= 'E') - { - if (yych >= 'E') - { - goto basic_json_parser_51; - } - } - else - { - if (yych == 'e') - { - goto basic_json_parser_51; - } - } - } -basic_json_parser_14: - { - last_token_type = token_type::value_unsigned; - break; - } -basic_json_parser_15: - yyaccept = 1; - m_marker = ++m_cursor; - if ((m_limit - m_cursor) < 3) - { - fill_line_buffer(3); // LCOV_EXCL_LINE - } - yych = *m_cursor; - if (yybm[0 + yych] & 64) - { - goto basic_json_parser_15; - } - if (yych <= 'D') - { - if (yych == '.') - { - goto basic_json_parser_47; - } - goto basic_json_parser_14; - } - else - { - if (yych <= 'E') - { - goto basic_json_parser_51; - } - if (yych == 'e') - { - goto basic_json_parser_51; - } - goto basic_json_parser_14; - } -basic_json_parser_17: - ++m_cursor; - { - last_token_type = token_type::name_separator; - break; - } -basic_json_parser_19: - ++m_cursor; - { - last_token_type = token_type::begin_array; - break; - } -basic_json_parser_21: - ++m_cursor; - { - last_token_type = token_type::end_array; - break; - } -basic_json_parser_23: - yyaccept = 0; - yych = *(m_marker = ++m_cursor); - if (yych == 'a') - { - goto basic_json_parser_52; - } - goto basic_json_parser_5; -basic_json_parser_24: - yyaccept = 0; - yych = *(m_marker = ++m_cursor); - if (yych == 'u') - { - goto basic_json_parser_53; - } - goto basic_json_parser_5; -basic_json_parser_25: - yyaccept = 0; - yych = *(m_marker = ++m_cursor); - if (yych == 'r') - { - goto basic_json_parser_54; - } - goto basic_json_parser_5; -basic_json_parser_26: - ++m_cursor; - { - last_token_type = token_type::begin_object; - break; - } -basic_json_parser_28: - ++m_cursor; - { - last_token_type = token_type::end_object; - break; - } -basic_json_parser_30: - ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } - yych = *m_cursor; -basic_json_parser_31: - if (yybm[0 + yych] & 128) - { - goto basic_json_parser_30; - } - if (yych <= 0xE0) - { - if (yych <= '\\') - { - if (yych <= 0x1F) - { - goto basic_json_parser_32; - } - if (yych <= '"') - { - goto basic_json_parser_33; - } - goto basic_json_parser_35; - } - else - { - if (yych <= 0xC1) - { - goto basic_json_parser_32; - } - if (yych <= 0xDF) - { - goto basic_json_parser_36; - } - goto basic_json_parser_37; - } - } - else - { - if (yych <= 0xEF) - { - if (yych == 0xED) - { - goto basic_json_parser_39; - } - goto basic_json_parser_38; - } - else - { - if (yych <= 0xF0) - { - goto basic_json_parser_40; - } - if (yych <= 0xF3) - { - goto basic_json_parser_41; - } - if (yych <= 0xF4) - { - goto basic_json_parser_42; - } - } - } -basic_json_parser_32: - m_cursor = m_marker; - if (yyaccept <= 1) - { - if (yyaccept == 0) - { - goto basic_json_parser_5; - } - else - { - goto basic_json_parser_14; - } - } - else - { - if (yyaccept == 2) - { - goto basic_json_parser_44; - } - else - { - goto basic_json_parser_58; - } - } -basic_json_parser_33: - ++m_cursor; - { - last_token_type = token_type::value_string; - break; - } -basic_json_parser_35: - ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } - yych = *m_cursor; - if (yych <= 'e') - { - if (yych <= '/') - { - if (yych == '"') - { - goto basic_json_parser_30; - } - if (yych <= '.') - { - goto basic_json_parser_32; - } - goto basic_json_parser_30; - } - else - { - if (yych <= '\\') - { - if (yych <= '[') - { - goto basic_json_parser_32; - } - goto basic_json_parser_30; - } - else - { - if (yych == 'b') - { - goto basic_json_parser_30; - } - goto basic_json_parser_32; - } - } - } - else - { - if (yych <= 'q') - { - if (yych <= 'f') - { - goto basic_json_parser_30; - } - if (yych == 'n') - { - goto basic_json_parser_30; - } - goto basic_json_parser_32; - } - else - { - if (yych <= 's') - { - if (yych <= 'r') - { - goto basic_json_parser_30; - } - goto basic_json_parser_32; - } - else - { - if (yych <= 't') - { - goto basic_json_parser_30; - } - if (yych <= 'u') - { - goto basic_json_parser_55; - } - goto basic_json_parser_32; - } - } - } -basic_json_parser_36: - ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } - yych = *m_cursor; - if (yych <= 0x7F) - { - goto basic_json_parser_32; - } - if (yych <= 0xBF) - { - goto basic_json_parser_30; - } - goto basic_json_parser_32; -basic_json_parser_37: - ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } - yych = *m_cursor; - if (yych <= 0x9F) - { - goto basic_json_parser_32; - } - if (yych <= 0xBF) - { - goto basic_json_parser_36; - } - goto basic_json_parser_32; -basic_json_parser_38: - ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } - yych = *m_cursor; - if (yych <= 0x7F) - { - goto basic_json_parser_32; - } - if (yych <= 0xBF) - { - goto basic_json_parser_36; - } - goto basic_json_parser_32; -basic_json_parser_39: - ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } - yych = *m_cursor; - if (yych <= 0x7F) - { - goto basic_json_parser_32; - } - if (yych <= 0x9F) - { - goto basic_json_parser_36; - } - goto basic_json_parser_32; -basic_json_parser_40: - ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } - yych = *m_cursor; - if (yych <= 0x8F) - { - goto basic_json_parser_32; - } - if (yych <= 0xBF) - { - goto basic_json_parser_38; - } - goto basic_json_parser_32; -basic_json_parser_41: - ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } - yych = *m_cursor; - if (yych <= 0x7F) - { - goto basic_json_parser_32; - } - if (yych <= 0xBF) - { - goto basic_json_parser_38; - } - goto basic_json_parser_32; -basic_json_parser_42: - ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } - yych = *m_cursor; - if (yych <= 0x7F) - { - goto basic_json_parser_32; - } - if (yych <= 0x8F) - { - goto basic_json_parser_38; - } - goto basic_json_parser_32; -basic_json_parser_43: - yyaccept = 2; - yych = *(m_marker = ++m_cursor); - if (yych <= '9') - { - if (yych == '.') - { - goto basic_json_parser_47; - } - if (yych >= '0') - { - goto basic_json_parser_48; - } - } - else - { - if (yych <= 'E') - { - if (yych >= 'E') - { - goto basic_json_parser_51; - } - } - else - { - if (yych == 'e') - { - goto basic_json_parser_51; - } - } - } -basic_json_parser_44: - { - last_token_type = token_type::value_integer; - break; - } -basic_json_parser_45: - yyaccept = 2; - m_marker = ++m_cursor; - if ((m_limit - m_cursor) < 3) - { - fill_line_buffer(3); // LCOV_EXCL_LINE - } - yych = *m_cursor; - if (yych <= '9') - { - if (yych == '.') - { - goto basic_json_parser_47; - } - if (yych <= '/') - { - goto basic_json_parser_44; - } - goto basic_json_parser_45; - } - else - { - if (yych <= 'E') - { - if (yych <= 'D') - { - goto basic_json_parser_44; - } - goto basic_json_parser_51; - } - else - { - if (yych == 'e') - { - goto basic_json_parser_51; - } - goto basic_json_parser_44; - } - } -basic_json_parser_47: - yych = *++m_cursor; - if (yych <= '/') - { - goto basic_json_parser_32; - } - if (yych <= '9') - { - goto basic_json_parser_56; - } - goto basic_json_parser_32; -basic_json_parser_48: - ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } - yych = *m_cursor; - if (yych <= '/') - { - goto basic_json_parser_50; - } - if (yych <= '9') - { - goto basic_json_parser_48; - } -basic_json_parser_50: - { - last_token_type = token_type::parse_error; - break; - } -basic_json_parser_51: - yych = *++m_cursor; - if (yych <= ',') - { - if (yych == '+') - { - goto basic_json_parser_59; - } - goto basic_json_parser_32; - } - else - { - if (yych <= '-') - { - goto basic_json_parser_59; - } - if (yych <= '/') - { - goto basic_json_parser_32; - } - if (yych <= '9') - { - goto basic_json_parser_60; - } - goto basic_json_parser_32; - } -basic_json_parser_52: - yych = *++m_cursor; - if (yych == 'l') - { - goto basic_json_parser_62; - } - goto basic_json_parser_32; -basic_json_parser_53: - yych = *++m_cursor; - if (yych == 'l') - { - goto basic_json_parser_63; - } - goto basic_json_parser_32; -basic_json_parser_54: - yych = *++m_cursor; - if (yych == 'u') - { - goto basic_json_parser_64; - } - goto basic_json_parser_32; -basic_json_parser_55: - ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } - yych = *m_cursor; - if (yych <= '@') - { - if (yych <= '/') - { - goto basic_json_parser_32; - } - if (yych <= '9') - { - goto basic_json_parser_65; - } - goto basic_json_parser_32; - } - else - { - if (yych <= 'F') - { - goto basic_json_parser_65; - } - if (yych <= '`') - { - goto basic_json_parser_32; - } - if (yych <= 'f') - { - goto basic_json_parser_65; - } - goto basic_json_parser_32; - } -basic_json_parser_56: - yyaccept = 3; - m_marker = ++m_cursor; - if ((m_limit - m_cursor) < 3) - { - fill_line_buffer(3); // LCOV_EXCL_LINE - } - yych = *m_cursor; - if (yych <= 'D') - { - if (yych <= '/') - { - goto basic_json_parser_58; - } - if (yych <= '9') - { - goto basic_json_parser_56; - } - } - else - { - if (yych <= 'E') - { - goto basic_json_parser_51; - } - if (yych == 'e') - { - goto basic_json_parser_51; - } - } -basic_json_parser_58: - { - last_token_type = token_type::value_float; - break; - } -basic_json_parser_59: - yych = *++m_cursor; - if (yych <= '/') - { - goto basic_json_parser_32; - } - if (yych >= ':') - { - goto basic_json_parser_32; - } -basic_json_parser_60: - ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } - yych = *m_cursor; - if (yych <= '/') - { - goto basic_json_parser_58; - } - if (yych <= '9') - { - goto basic_json_parser_60; - } - goto basic_json_parser_58; -basic_json_parser_62: - yych = *++m_cursor; - if (yych == 's') - { - goto basic_json_parser_66; - } - goto basic_json_parser_32; -basic_json_parser_63: - yych = *++m_cursor; - if (yych == 'l') - { - goto basic_json_parser_67; - } - goto basic_json_parser_32; -basic_json_parser_64: - yych = *++m_cursor; - if (yych == 'e') - { - goto basic_json_parser_69; - } - goto basic_json_parser_32; -basic_json_parser_65: - ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } - yych = *m_cursor; - if (yych <= '@') - { - if (yych <= '/') - { - goto basic_json_parser_32; - } - if (yych <= '9') - { - goto basic_json_parser_71; - } - goto basic_json_parser_32; - } - else - { - if (yych <= 'F') - { - goto basic_json_parser_71; - } - if (yych <= '`') - { - goto basic_json_parser_32; - } - if (yych <= 'f') - { - goto basic_json_parser_71; - } - goto basic_json_parser_32; - } -basic_json_parser_66: - yych = *++m_cursor; - if (yych == 'e') - { - goto basic_json_parser_72; - } - goto basic_json_parser_32; -basic_json_parser_67: - ++m_cursor; - { - last_token_type = token_type::literal_null; - break; - } -basic_json_parser_69: - ++m_cursor; - { - last_token_type = token_type::literal_true; - break; - } -basic_json_parser_71: - ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } - yych = *m_cursor; - if (yych <= '@') - { - if (yych <= '/') - { - goto basic_json_parser_32; - } - if (yych <= '9') - { - goto basic_json_parser_74; - } - goto basic_json_parser_32; - } - else - { - if (yych <= 'F') - { - goto basic_json_parser_74; - } - if (yych <= '`') - { - goto basic_json_parser_32; - } - if (yych <= 'f') - { - goto basic_json_parser_74; - } - goto basic_json_parser_32; - } -basic_json_parser_72: - ++m_cursor; - { - last_token_type = token_type::literal_false; - break; - } -basic_json_parser_74: - ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } - yych = *m_cursor; - if (yych <= '@') - { - if (yych <= '/') - { - goto basic_json_parser_32; - } - if (yych <= '9') - { - goto basic_json_parser_30; - } - goto basic_json_parser_32; - } - else - { - if (yych <= 'F') - { - goto basic_json_parser_30; - } - if (yych <= '`') - { - goto basic_json_parser_32; - } - if (yych <= 'f') - { - goto basic_json_parser_30; - } - goto basic_json_parser_32; - } + + { + lexer_char_t yych; + unsigned int yyaccept = 0; + static const unsigned char yybm[] = { + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 32, 32, 0, 0, 32, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 160, 128, 0, 128, 128, 128, 128, 128, + 128, 128, 128, 128, 128, 128, 128, 128, + 192, 192, 192, 192, 192, 192, 192, 192, + 192, 192, 128, 128, 128, 128, 128, 128, + 128, 128, 128, 128, 128, 128, 128, 128, + 128, 128, 128, 128, 128, 128, 128, 128, + 128, 128, 128, 128, 128, 128, 128, 128, + 128, 128, 128, 128, 0, 128, 128, 128, + 128, 128, 128, 128, 128, 128, 128, 128, + 128, 128, 128, 128, 128, 128, 128, 128, + 128, 128, 128, 128, 128, 128, 128, 128, + 128, 128, 128, 128, 128, 128, 128, 128, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + }; + if ((m_limit - m_cursor) < 5) fill_line_buffer(5); // LCOV_EXCL_LINE + yych = *m_cursor; + if (yybm[0+yych] & 32) { + goto basic_json_parser_6; + } + if (yych <= '[') { + if (yych <= '-') { + if (yych <= '"') { + if (yych <= 0x00) goto basic_json_parser_2; + if (yych <= '!') goto basic_json_parser_4; + goto basic_json_parser_9; + } else { + if (yych <= '+') goto basic_json_parser_4; + if (yych <= ',') goto basic_json_parser_10; + goto basic_json_parser_12; } + } else { + if (yych <= '9') { + if (yych <= '/') goto basic_json_parser_4; + if (yych <= '0') goto basic_json_parser_13; + goto basic_json_parser_15; + } else { + if (yych <= ':') goto basic_json_parser_17; + if (yych <= 'Z') goto basic_json_parser_4; + goto basic_json_parser_19; + } + } + } else { + if (yych <= 'n') { + if (yych <= 'e') { + if (yych == ']') goto basic_json_parser_21; + goto basic_json_parser_4; + } else { + if (yych <= 'f') goto basic_json_parser_23; + if (yych <= 'm') goto basic_json_parser_4; + goto basic_json_parser_24; + } + } else { + if (yych <= 'z') { + if (yych == 't') goto basic_json_parser_25; + goto basic_json_parser_4; + } else { + if (yych <= '{') goto basic_json_parser_26; + if (yych == '}') goto basic_json_parser_28; + goto basic_json_parser_4; + } + } + } +basic_json_parser_2: + ++m_cursor; + { last_token_type = token_type::end_of_input; break; } +basic_json_parser_4: + ++m_cursor; +basic_json_parser_5: + { last_token_type = token_type::parse_error; break; } +basic_json_parser_6: + ++m_cursor; + if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE + yych = *m_cursor; + if (yybm[0+yych] & 32) { + goto basic_json_parser_6; + } + { position += static_cast((m_cursor - m_start)); continue; } +basic_json_parser_9: + yyaccept = 0; + yych = *(m_marker = ++m_cursor); + if (yych <= 0x1F) goto basic_json_parser_5; + if (yych <= 0x7F) goto basic_json_parser_31; + if (yych <= 0xC1) goto basic_json_parser_5; + if (yych <= 0xF4) goto basic_json_parser_31; + goto basic_json_parser_5; +basic_json_parser_10: + ++m_cursor; + { last_token_type = token_type::value_separator; break; } +basic_json_parser_12: + yych = *++m_cursor; + if (yych <= '/') goto basic_json_parser_5; + if (yych <= '0') goto basic_json_parser_43; + if (yych <= '9') goto basic_json_parser_45; + goto basic_json_parser_5; +basic_json_parser_13: + yyaccept = 1; + yych = *(m_marker = ++m_cursor); + if (yych <= '9') { + if (yych == '.') goto basic_json_parser_47; + if (yych >= '0') goto basic_json_parser_48; + } else { + if (yych <= 'E') { + if (yych >= 'E') goto basic_json_parser_51; + } else { + if (yych == 'e') goto basic_json_parser_51; + } + } +basic_json_parser_14: + { last_token_type = token_type::value_unsigned; break; } +basic_json_parser_15: + yyaccept = 1; + m_marker = ++m_cursor; + if ((m_limit - m_cursor) < 3) fill_line_buffer(3); // LCOV_EXCL_LINE + yych = *m_cursor; + if (yybm[0+yych] & 64) { + goto basic_json_parser_15; + } + if (yych <= 'D') { + if (yych == '.') goto basic_json_parser_47; + goto basic_json_parser_14; + } else { + if (yych <= 'E') goto basic_json_parser_51; + if (yych == 'e') goto basic_json_parser_51; + goto basic_json_parser_14; + } +basic_json_parser_17: + ++m_cursor; + { last_token_type = token_type::name_separator; break; } +basic_json_parser_19: + ++m_cursor; + { last_token_type = token_type::begin_array; break; } +basic_json_parser_21: + ++m_cursor; + { last_token_type = token_type::end_array; break; } +basic_json_parser_23: + yyaccept = 0; + yych = *(m_marker = ++m_cursor); + if (yych == 'a') goto basic_json_parser_52; + goto basic_json_parser_5; +basic_json_parser_24: + yyaccept = 0; + yych = *(m_marker = ++m_cursor); + if (yych == 'u') goto basic_json_parser_53; + goto basic_json_parser_5; +basic_json_parser_25: + yyaccept = 0; + yych = *(m_marker = ++m_cursor); + if (yych == 'r') goto basic_json_parser_54; + goto basic_json_parser_5; +basic_json_parser_26: + ++m_cursor; + { last_token_type = token_type::begin_object; break; } +basic_json_parser_28: + ++m_cursor; + { last_token_type = token_type::end_object; break; } +basic_json_parser_30: + ++m_cursor; + if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE + yych = *m_cursor; +basic_json_parser_31: + if (yybm[0+yych] & 128) { + goto basic_json_parser_30; + } + if (yych <= 0xE0) { + if (yych <= '\\') { + if (yych <= 0x1F) goto basic_json_parser_32; + if (yych <= '"') goto basic_json_parser_33; + goto basic_json_parser_35; + } else { + if (yych <= 0xC1) goto basic_json_parser_32; + if (yych <= 0xDF) goto basic_json_parser_36; + goto basic_json_parser_37; + } + } else { + if (yych <= 0xEF) { + if (yych == 0xED) goto basic_json_parser_39; + goto basic_json_parser_38; + } else { + if (yych <= 0xF0) goto basic_json_parser_40; + if (yych <= 0xF3) goto basic_json_parser_41; + if (yych <= 0xF4) goto basic_json_parser_42; + } + } +basic_json_parser_32: + m_cursor = m_marker; + if (yyaccept <= 1) { + if (yyaccept == 0) { + goto basic_json_parser_5; + } else { + goto basic_json_parser_14; + } + } else { + if (yyaccept == 2) { + goto basic_json_parser_44; + } else { + goto basic_json_parser_58; + } + } +basic_json_parser_33: + ++m_cursor; + { last_token_type = token_type::value_string; break; } +basic_json_parser_35: + ++m_cursor; + if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE + yych = *m_cursor; + if (yych <= 'e') { + if (yych <= '/') { + if (yych == '"') goto basic_json_parser_30; + if (yych <= '.') goto basic_json_parser_32; + goto basic_json_parser_30; + } else { + if (yych <= '\\') { + if (yych <= '[') goto basic_json_parser_32; + goto basic_json_parser_30; + } else { + if (yych == 'b') goto basic_json_parser_30; + goto basic_json_parser_32; + } + } + } else { + if (yych <= 'q') { + if (yych <= 'f') goto basic_json_parser_30; + if (yych == 'n') goto basic_json_parser_30; + goto basic_json_parser_32; + } else { + if (yych <= 's') { + if (yych <= 'r') goto basic_json_parser_30; + goto basic_json_parser_32; + } else { + if (yych <= 't') goto basic_json_parser_30; + if (yych <= 'u') goto basic_json_parser_55; + goto basic_json_parser_32; + } + } + } +basic_json_parser_36: + ++m_cursor; + if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE + yych = *m_cursor; + if (yych <= 0x7F) goto basic_json_parser_32; + if (yych <= 0xBF) goto basic_json_parser_30; + goto basic_json_parser_32; +basic_json_parser_37: + ++m_cursor; + if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE + yych = *m_cursor; + if (yych <= 0x9F) goto basic_json_parser_32; + if (yych <= 0xBF) goto basic_json_parser_36; + goto basic_json_parser_32; +basic_json_parser_38: + ++m_cursor; + if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE + yych = *m_cursor; + if (yych <= 0x7F) goto basic_json_parser_32; + if (yych <= 0xBF) goto basic_json_parser_36; + goto basic_json_parser_32; +basic_json_parser_39: + ++m_cursor; + if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE + yych = *m_cursor; + if (yych <= 0x7F) goto basic_json_parser_32; + if (yych <= 0x9F) goto basic_json_parser_36; + goto basic_json_parser_32; +basic_json_parser_40: + ++m_cursor; + if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE + yych = *m_cursor; + if (yych <= 0x8F) goto basic_json_parser_32; + if (yych <= 0xBF) goto basic_json_parser_38; + goto basic_json_parser_32; +basic_json_parser_41: + ++m_cursor; + if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE + yych = *m_cursor; + if (yych <= 0x7F) goto basic_json_parser_32; + if (yych <= 0xBF) goto basic_json_parser_38; + goto basic_json_parser_32; +basic_json_parser_42: + ++m_cursor; + if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE + yych = *m_cursor; + if (yych <= 0x7F) goto basic_json_parser_32; + if (yych <= 0x8F) goto basic_json_parser_38; + goto basic_json_parser_32; +basic_json_parser_43: + yyaccept = 2; + yych = *(m_marker = ++m_cursor); + if (yych <= '9') { + if (yych == '.') goto basic_json_parser_47; + if (yych >= '0') goto basic_json_parser_48; + } else { + if (yych <= 'E') { + if (yych >= 'E') goto basic_json_parser_51; + } else { + if (yych == 'e') goto basic_json_parser_51; + } + } +basic_json_parser_44: + { last_token_type = token_type::value_integer; break; } +basic_json_parser_45: + yyaccept = 2; + m_marker = ++m_cursor; + if ((m_limit - m_cursor) < 3) fill_line_buffer(3); // LCOV_EXCL_LINE + yych = *m_cursor; + if (yych <= '9') { + if (yych == '.') goto basic_json_parser_47; + if (yych <= '/') goto basic_json_parser_44; + goto basic_json_parser_45; + } else { + if (yych <= 'E') { + if (yych <= 'D') goto basic_json_parser_44; + goto basic_json_parser_51; + } else { + if (yych == 'e') goto basic_json_parser_51; + goto basic_json_parser_44; + } + } +basic_json_parser_47: + yych = *++m_cursor; + if (yych <= '/') goto basic_json_parser_32; + if (yych <= '9') goto basic_json_parser_56; + goto basic_json_parser_32; +basic_json_parser_48: + ++m_cursor; + if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE + yych = *m_cursor; + if (yych <= '/') goto basic_json_parser_50; + if (yych <= '9') goto basic_json_parser_48; +basic_json_parser_50: + { last_token_type = token_type::parse_error; break; } +basic_json_parser_51: + yych = *++m_cursor; + if (yych <= ',') { + if (yych == '+') goto basic_json_parser_59; + goto basic_json_parser_32; + } else { + if (yych <= '-') goto basic_json_parser_59; + if (yych <= '/') goto basic_json_parser_32; + if (yych <= '9') goto basic_json_parser_60; + goto basic_json_parser_32; + } +basic_json_parser_52: + yych = *++m_cursor; + if (yych == 'l') goto basic_json_parser_62; + goto basic_json_parser_32; +basic_json_parser_53: + yych = *++m_cursor; + if (yych == 'l') goto basic_json_parser_63; + goto basic_json_parser_32; +basic_json_parser_54: + yych = *++m_cursor; + if (yych == 'u') goto basic_json_parser_64; + goto basic_json_parser_32; +basic_json_parser_55: + ++m_cursor; + if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE + yych = *m_cursor; + if (yych <= '@') { + if (yych <= '/') goto basic_json_parser_32; + if (yych <= '9') goto basic_json_parser_65; + goto basic_json_parser_32; + } else { + if (yych <= 'F') goto basic_json_parser_65; + if (yych <= '`') goto basic_json_parser_32; + if (yych <= 'f') goto basic_json_parser_65; + goto basic_json_parser_32; + } +basic_json_parser_56: + yyaccept = 3; + m_marker = ++m_cursor; + if ((m_limit - m_cursor) < 3) fill_line_buffer(3); // LCOV_EXCL_LINE + yych = *m_cursor; + if (yych <= 'D') { + if (yych <= '/') goto basic_json_parser_58; + if (yych <= '9') goto basic_json_parser_56; + } else { + if (yych <= 'E') goto basic_json_parser_51; + if (yych == 'e') goto basic_json_parser_51; + } +basic_json_parser_58: + { last_token_type = token_type::value_float; break; } +basic_json_parser_59: + yych = *++m_cursor; + if (yych <= '/') goto basic_json_parser_32; + if (yych >= ':') goto basic_json_parser_32; +basic_json_parser_60: + ++m_cursor; + if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE + yych = *m_cursor; + if (yych <= '/') goto basic_json_parser_58; + if (yych <= '9') goto basic_json_parser_60; + goto basic_json_parser_58; +basic_json_parser_62: + yych = *++m_cursor; + if (yych == 's') goto basic_json_parser_66; + goto basic_json_parser_32; +basic_json_parser_63: + yych = *++m_cursor; + if (yych == 'l') goto basic_json_parser_67; + goto basic_json_parser_32; +basic_json_parser_64: + yych = *++m_cursor; + if (yych == 'e') goto basic_json_parser_69; + goto basic_json_parser_32; +basic_json_parser_65: + ++m_cursor; + if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE + yych = *m_cursor; + if (yych <= '@') { + if (yych <= '/') goto basic_json_parser_32; + if (yych <= '9') goto basic_json_parser_71; + goto basic_json_parser_32; + } else { + if (yych <= 'F') goto basic_json_parser_71; + if (yych <= '`') goto basic_json_parser_32; + if (yych <= 'f') goto basic_json_parser_71; + goto basic_json_parser_32; + } +basic_json_parser_66: + yych = *++m_cursor; + if (yych == 'e') goto basic_json_parser_72; + goto basic_json_parser_32; +basic_json_parser_67: + ++m_cursor; + { last_token_type = token_type::literal_null; break; } +basic_json_parser_69: + ++m_cursor; + { last_token_type = token_type::literal_true; break; } +basic_json_parser_71: + ++m_cursor; + if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE + yych = *m_cursor; + if (yych <= '@') { + if (yych <= '/') goto basic_json_parser_32; + if (yych <= '9') goto basic_json_parser_74; + goto basic_json_parser_32; + } else { + if (yych <= 'F') goto basic_json_parser_74; + if (yych <= '`') goto basic_json_parser_32; + if (yych <= 'f') goto basic_json_parser_74; + goto basic_json_parser_32; + } +basic_json_parser_72: + ++m_cursor; + { last_token_type = token_type::literal_false; break; } +basic_json_parser_74: + ++m_cursor; + if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE + yych = *m_cursor; + if (yych <= '@') { + if (yych <= '/') goto basic_json_parser_32; + if (yych <= '9') goto basic_json_parser_30; + goto basic_json_parser_32; + } else { + if (yych <= 'F') goto basic_json_parser_30; + if (yych <= '`') goto basic_json_parser_32; + if (yych <= 'f') goto basic_json_parser_30; + goto basic_json_parser_32; + } + } } diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index c4d142b7..1b8a9067 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -800,6 +800,13 @@ void to_json(BasicJsonType& j, const CompatibleObjectType& arr) external_constructor::construct(j, arr); } +template ::value, + int> = 0> +void to_json(BasicJsonType &j, T (&arr)[N]) { + external_constructor::construct(j, arr); +} /////////////// // from_json // diff --git a/test/src/unit-conversions.cpp b/test/src/unit-conversions.cpp index 3a83aef3..829e9e2d 100644 --- a/test/src/unit-conversions.cpp +++ b/test/src/unit-conversions.cpp @@ -190,6 +190,20 @@ TEST_CASE("value conversion") #endif } + SECTION("built-in arrays") + { + const char str[] = "a string"; + const int nbs[] = {0, 1, 2}; + + json j2 = nbs; + json j3 = str; + + auto v = j2.get>(); + auto s = j3.get(); + CHECK(std::equal(v.begin(), v.end(), std::begin(nbs))); + CHECK(s == str); + } + SECTION("std::deque") { std::deque a = j.get>(); From 483a58f62500b3d15aeaf1ce2a8a97829e574073 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Thu, 16 Mar 2017 18:35:45 +0100 Subject: [PATCH 086/530] :rotating_light: added pedantic flags for GCC With GCC 7, there are even more diagnostic flags. --- Makefile | 62 +++++++++++++++++++++++++++++++++++++++++++++++++++++++- 1 file changed, 61 insertions(+), 1 deletion(-) diff --git a/Makefile b/Makefile index f52511c6..df05e62e 100644 --- a/Makefile +++ b/Makefile @@ -51,7 +51,7 @@ doctest: # -Wno-keyword-macro: unit-tests use "#define private public" # -Wno-weak-vtables: exception class is defined inline, but has virtual method # -Wno-range-loop-analysis: iterator_wrapper tests tests "for(const auto i...)" -pedantic: +pedantic_clang: $(MAKE) json_unit CXXFLAGS="\ -std=c++11 \ -Werror \ @@ -62,6 +62,66 @@ pedantic: -Wno-weak-vtables \ -Wno-range-loop-analysis" +# calling GCC with most warnings +pedantic_gcc: + $(MAKE) json_unit CXX=g++ CXXFLAGS="\ + -std=c++11 \ + -Werror \ + -Wall -Wpedantic -Wextra \ + -Walloca \ + -Warray-bounds=2 \ + -Wcast-qual -Wcast-align \ + -Wchar-subscripts \ + -Wconditionally-supported \ + -Wconversion \ + -Wdate-time \ + -Wdeprecated \ + -Wdisabled-optimization \ + -Wdouble-promotion \ + -Wduplicated-branches \ + -Wduplicated-cond \ + -Weffc++ \ + -Wformat-overflow=2 \ + -Wformat-signedness \ + -Wformat-truncation=2 \ + -Wformat=2 \ + -Wimplicit-fallthrough=5 \ + -Wlogical-op \ + -Wmissing-declarations \ + -Wmissing-format-attribute \ + -Wmissing-include-dirs \ + -Wnoexcept \ + -Wnonnull \ + -Wnull-dereference \ + -Wold-style-cast \ + -Woverloaded-virtual \ + -Wparentheses \ + -Wplacement-new=2 \ + -Wredundant-decls \ + -Wreorder \ + -Wrestrict \ + -Wshadow=global \ + -Wshift-overflow=2 \ + -Wsign-conversion \ + -Wsign-promo \ + -Wsized-deallocation \ + -Wstrict-overflow=5 \ + -Wsuggest-attribute=const \ + -Wsuggest-attribute=format \ + -Wsuggest-attribute=noreturn \ + -Wsuggest-attribute=pure \ + -Wsuggest-final-methods \ + -Wsuggest-final-types \ + -Wsuggest-override \ + -Wtrigraphs \ + -Wundef \ + -Wuninitialized -Wunknown-pragmas \ + -Wunused \ + -Wunused-const-variable=2 \ + -Wunused-macros \ + -Wunused-parameter \ + -Wuseless-cast \ + -Wvariadic-macros" ########################################################################## # fuzzing From c5711f30729c05d4f7f0b99de9db1bc41f173679 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Thu, 16 Mar 2017 18:39:33 +0100 Subject: [PATCH 087/530] :construction: a lot of minor changes - Removed unused headers. - Added override where needed. - Added description for parse_error.113 exception. - Fixed some conversion warnings. - Integrated cbor_expect_string function for CBOR maps. - Added documentation on the supported CBOR/MessagePack features. - Added test to check all initial bytes for CBOR input. --- src/json.hpp | 137 +++++++++++++++++++++++++++--- src/json.hpp.re2c | 137 +++++++++++++++++++++++++++--- test/src/unit-cbor.cpp | 109 ++++++++++++++++++++---- test/src/unit-class_lexer.cpp | 2 +- test/src/unit-class_parser.cpp | 10 +-- test/src/unit-concepts.cpp | 4 +- test/src/unit-element_access1.cpp | 8 +- test/src/unit-element_access2.cpp | 16 ++-- test/src/unit-msgpack.cpp | 40 ++++----- test/src/unit-regression.cpp | 12 +-- 10 files changed, 385 insertions(+), 90 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index cbf12dc5..339cd095 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -32,7 +32,6 @@ SOFTWARE. #include // all_of, copy, fill, find, for_each, none_of, remove, reverse, transform #include // array #include // assert -#include // isdigit #include // and, not, or #include // lconv, localeconv #include // isfinite, labs, ldexp, signbit @@ -43,7 +42,6 @@ SOFTWARE. #include // forward_list #include // function, hash, less #include // initializer_list -#include // setw #include // istream, ostream #include // advance, begin, back_inserter, bidirectional_iterator_tag, distance, end, inserter, iterator, iterator_traits, next, random_access_iterator_tag, reverse_iterator #include // numeric_limits @@ -130,7 +128,7 @@ class exception : public std::exception {} /// returns the explanatory string - virtual const char* what() const noexcept + virtual const char* what() const noexcept override { return what_arg.c_str(); } @@ -174,7 +172,7 @@ json.exception.parse_error.109 | parse error: array index 'one' is not a number json.exception.parse_error.110 | parse error at 1: cannot read 2 bytes from vector | When parsing CBOR or MessagePack, the byte vector ends before the complete value has been read. json.exception.parse_error.111 | parse error: bad input stream | Parsing CBOR or MessagePack from an input stream where the [`badbit` or `failbit`](http://en.cppreference.com/w/cpp/io/ios_base/iostate) is set. json.exception.parse_error.112 | parse error at 1: error reading CBOR; last byte: 0xf8 | Not all types of CBOR or MessagePack are supported. This exception occurs if an unsupported byte was read. -json.exception.parse_error.113 | | While parsing a map key, a value that is not a string has been read. +json.exception.parse_error.113 | parse error at 2: expected a CBOR string; last byte: 0x98 | While parsing a map key, a value that is not a string has been read. @since version 3.0.0 */ @@ -6468,7 +6466,7 @@ class basic_json auto i = val.m_value.object->cbegin(); for (size_t cnt = 0; cnt < val.m_value.object->size() - 1; ++cnt, ++i) { - o.write(indent_string.c_str(), new_indent); + o.write(indent_string.c_str(), static_cast(new_indent)); o.put('\"'); dump_escaped(i->first); o.write("\": ", 3); @@ -6478,14 +6476,14 @@ class basic_json // last element assert(i != val.m_value.object->cend()); - o.write(indent_string.c_str(), new_indent); + o.write(indent_string.c_str(), static_cast(new_indent)); o.put('\"'); dump_escaped(i->first); o.write("\": ", 3); dump(i->second, true, indent_step, new_indent); o.put('\n'); - o.write(indent_string.c_str(), current_indent); + o.write(indent_string.c_str(), static_cast(current_indent)); o.put('}'); } else @@ -6538,18 +6536,18 @@ class basic_json // first n-1 elements for (auto i = val.m_value.array->cbegin(); i != val.m_value.array->cend() - 1; ++i) { - o.write(indent_string.c_str(), new_indent); + o.write(indent_string.c_str(), static_cast(new_indent)); dump(*i, true, indent_step, new_indent); o.write(",\n", 2); } // last element assert(not val.m_value.array->empty()); - o.write(indent_string.c_str(), new_indent); + o.write(indent_string.c_str(), static_cast(new_indent)); dump(val.m_value.array->back(), true, indent_step, new_indent); o.put('\n'); - o.write(indent_string.c_str(), current_indent); + o.write(indent_string.c_str(), static_cast(current_indent)); o.put(']'); } else @@ -7848,7 +7846,7 @@ class basic_json const auto N = j.m_value.string->size(); if (N <= 0x17) { - v.push_back(0x60 + static_cast(N)); // 1 byte for string + size + v.push_back(static_cast(0x60 + N)); // 1 byte for string + size } else if (N <= 0xff) { @@ -7884,7 +7882,7 @@ class basic_json const auto N = j.m_value.array->size(); if (N <= 0x17) { - v.push_back(0x80 + static_cast(N)); // 1 byte for array + size + v.push_back(static_cast(0x80 + N)); // 1 byte for array + size } else if (N <= 0xff) { @@ -7922,7 +7920,7 @@ class basic_json const auto N = j.m_value.object->size(); if (N <= 0x17) { - v.push_back(0xa0 + static_cast(N)); // 1 byte for object + size + v.push_back(static_cast(0xa0 + N)); // 1 byte for object + size } else if (N <= 0xff) { @@ -8653,6 +8651,7 @@ class basic_json const auto len = static_cast(v[current_idx] - 0xa0); for (size_t i = 0; i < len; ++i) { + cbor_expect_string(v, idx); std::string key = from_cbor_internal(v, idx); result[key] = from_cbor_internal(v, idx); } @@ -8666,6 +8665,7 @@ class basic_json idx += 1; // skip 1 size byte for (size_t i = 0; i < len; ++i) { + cbor_expect_string(v, idx); std::string key = from_cbor_internal(v, idx); result[key] = from_cbor_internal(v, idx); } @@ -8679,6 +8679,7 @@ class basic_json idx += 2; // skip 2 size bytes for (size_t i = 0; i < len; ++i) { + cbor_expect_string(v, idx); std::string key = from_cbor_internal(v, idx); result[key] = from_cbor_internal(v, idx); } @@ -8692,6 +8693,7 @@ class basic_json idx += 4; // skip 4 size bytes for (size_t i = 0; i < len; ++i) { + cbor_expect_string(v, idx); std::string key = from_cbor_internal(v, idx); result[key] = from_cbor_internal(v, idx); } @@ -8705,6 +8707,7 @@ class basic_json idx += 8; // skip 8 size bytes for (size_t i = 0; i < len; ++i) { + cbor_expect_string(v, idx); std::string key = from_cbor_internal(v, idx); result[key] = from_cbor_internal(v, idx); } @@ -8716,6 +8719,7 @@ class basic_json basic_json result = value_t::object; while (check_length(v.size(), 1, idx), v[idx] != 0xff) { + cbor_expect_string(v, idx); std::string key = from_cbor_internal(v, idx); result[key] = from_cbor_internal(v, idx); } @@ -8864,6 +8868,9 @@ class basic_json - float 32 (0xca) - fixext 1 - fixext 16 (0xd4..0xd8) + @note Any MessagePack output created @ref to_msgpack can be successfully + parsed by @ref from_msgpack. + @param[in] j JSON value to serialize @return MessagePack serialization as byte vector @@ -8892,6 +8899,46 @@ class basic_json Deserializes a given byte vector @a v to a JSON value using the MessagePack serialization format. + The library maps MessagePack types to JSON value types as follows: + + MessagePack type | JSON value type | first byte + ---------------- | --------------- | ---------- + positive fixint | number_unsigned | 0x00..0x7f + fixmap | object | 0x80..0x8f + fixarray | array | 0x90..0x9f + fixstr | string | 0xa0..0xbf + nil | `null` | 0xc0 + false | `false` | 0xc2 + true | `true` | 0xc3 + float 32 | number_float | 0xca + float 64 | number_float | 0xcb + uint 8 | number_unsigned | 0xcc + uint 16 | number_unsigned | 0xcd + uint 32 | number_unsigned | 0xce + uint 64 | number_unsigned | 0xcf + int 8 | number_integer | 0xd0 + int 16 | number_integer | 0xd1 + int 32 | number_integer | 0xd2 + int 64 | number_integer | 0xd3 + str 8 | string | 0xd9 + str 16 | string | 0xda + str 32 | string | 0xdb + array 16 | array | 0xdc + array 32 | array | 0xdd + map 16 | object | 0xde + map 32 | object | 0xdf + negative fixint | number_integer | 0xe0-0xff + + @warning The mapping is **incomplete** in the sense that not all + MessagePack types can be converted to a JSON value. The following + MessagePack types are not supported and will yield parse errors: + - bin 8 - bin 32 (0xc4..0xc6) + - ext 8 - ext 32 (0xc7..0xc9) + - fixext 1 - fixext 16 (0xd4..0xd8) + + @note Any MessagePack output created @ref to_msgpack can be successfully + parsed by @ref from_msgpack. + @param[in] v a byte vector in MessagePack format @param[in] start_index the index to start reading from @a v (0 by default) @return deserialized JSON value @@ -8899,6 +8946,7 @@ class basic_json @throw parse_error.110 if the given vector ends prematurely @throw parse_error.112 if unsupported features from MessagePack were used in the given vector @a v or if the input is not valid MessagePack + @throw parse_error.113 if a string was expected as map key, but not found @complexity Linear in the size of the byte vector @a v. @@ -9014,6 +9062,66 @@ class basic_json Deserializes a given byte vector @a v to a JSON value using the CBOR (Concise Binary Object Representation) serialization format. + The library maps CBOR types to JSON value types as follows: + + CBOR type | JSON value type | first byte + ---------------------- | --------------- | ---------- + Integer | number_unsigned | 0x00..0x17 + Unsigned integer | number_unsigned | 0x18 + Unsigned integer | number_unsigned | 0x19 + Unsigned integer | number_unsigned | 0x1a + Unsigned integer | number_unsigned | 0x1b + Negative integer | number_integer | 0x20..0x37 + Negative integer | number_integer | 0x38 + Negative integer | number_integer | 0x39 + Negative integer | number_integer | 0x3a + Negative integer | number_integer | 0x3b + Negative integer | number_integer | 0x40..0x57 + UTF-8 string | string | 0x60..0x77 + UTF-8 string | string | 0x78 + UTF-8 string | string | 0x79 + UTF-8 string | string | 0x7a + UTF-8 string | string | 0x7b + UTF-8 string | string | 0x7f + array | array | 0x80..0x97 + array | array | 0x98 + array | array | 0x99 + array | array | 0x9a + array | array | 0x9b + array | array | 0x9f + map | object | 0xa0..0xb7 + map | object | 0xb8 + map | object | 0xb9 + map | object | 0xba + map | object | 0xbb + map | object | 0xbf + False | `false` | 0xf4 + True | `true` | 0xf5 + Nill | `null` | 0xf6 + Half-Precision Float | number_float | 0xf9 + Single-Precision Float | number_float | 0xfa + Double-Precision Float | number_float | 0xfb + + @warning The mapping is **incomplete** in the sense that not all CBOR + types can be converted to a JSON value. The following CBOR types + are not supported and will yield parse errors (parse_error.112): + - byte strings (0x40..0x5f) + - date/time (0xc0..0xc1) + - bignum (0xc2..0xc3) + - decimal fraction (0xc4) + - bigfloat (0xc5) + - tagged items (0xc6..0xd4, 0xd8..0xdb) + - expected conversions (0xd5..0xd7) + - simple values (0xe0..0xf3, 0xf8) + - undefined (0xf7) + + @warning CBOR allows map keys of any type, whereas JSON only allows + strings as keys in object values. Therefore, CBOR maps with keys + other than UTF-8 strings are rejected (parse_error.113). + + @note Any CBOR output created @ref to_cbor can be successfully parsed by + @ref from_cbor. + @param[in] v a byte vector in CBOR format @param[in] start_index the index to start reading from @a v (0 by default) @return deserialized JSON value @@ -9021,6 +9129,7 @@ class basic_json @throw parse_error.110 if the given vector ends prematurely @throw parse_error.112 if unsupported features from CBOR were used in the given vector @a v or if the input is not valid CBOR + @throw parse_error.113 if a string was expected as map key, but not found @complexity Linear in the size of the byte vector @a v. @@ -12423,7 +12532,7 @@ basic_json_parser_74: reference_token.end(), [](const char x) { - return std::isdigit(x); + return (x >= '0' and x <= '9'); }); // change value to array for numbers or "-" or to object diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index c4d142b7..5203fb8b 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -32,7 +32,6 @@ SOFTWARE. #include // all_of, copy, fill, find, for_each, none_of, remove, reverse, transform #include // array #include // assert -#include // isdigit #include // and, not, or #include // lconv, localeconv #include // isfinite, labs, ldexp, signbit @@ -43,7 +42,6 @@ SOFTWARE. #include // forward_list #include // function, hash, less #include // initializer_list -#include // setw #include // istream, ostream #include // advance, begin, back_inserter, bidirectional_iterator_tag, distance, end, inserter, iterator, iterator_traits, next, random_access_iterator_tag, reverse_iterator #include // numeric_limits @@ -130,7 +128,7 @@ class exception : public std::exception {} /// returns the explanatory string - virtual const char* what() const noexcept + virtual const char* what() const noexcept override { return what_arg.c_str(); } @@ -174,7 +172,7 @@ json.exception.parse_error.109 | parse error: array index 'one' is not a number json.exception.parse_error.110 | parse error at 1: cannot read 2 bytes from vector | When parsing CBOR or MessagePack, the byte vector ends before the complete value has been read. json.exception.parse_error.111 | parse error: bad input stream | Parsing CBOR or MessagePack from an input stream where the [`badbit` or `failbit`](http://en.cppreference.com/w/cpp/io/ios_base/iostate) is set. json.exception.parse_error.112 | parse error at 1: error reading CBOR; last byte: 0xf8 | Not all types of CBOR or MessagePack are supported. This exception occurs if an unsupported byte was read. -json.exception.parse_error.113 | | While parsing a map key, a value that is not a string has been read. +json.exception.parse_error.113 | parse error at 2: expected a CBOR string; last byte: 0x98 | While parsing a map key, a value that is not a string has been read. @since version 3.0.0 */ @@ -6468,7 +6466,7 @@ class basic_json auto i = val.m_value.object->cbegin(); for (size_t cnt = 0; cnt < val.m_value.object->size() - 1; ++cnt, ++i) { - o.write(indent_string.c_str(), new_indent); + o.write(indent_string.c_str(), static_cast(new_indent)); o.put('\"'); dump_escaped(i->first); o.write("\": ", 3); @@ -6478,14 +6476,14 @@ class basic_json // last element assert(i != val.m_value.object->cend()); - o.write(indent_string.c_str(), new_indent); + o.write(indent_string.c_str(), static_cast(new_indent)); o.put('\"'); dump_escaped(i->first); o.write("\": ", 3); dump(i->second, true, indent_step, new_indent); o.put('\n'); - o.write(indent_string.c_str(), current_indent); + o.write(indent_string.c_str(), static_cast(current_indent)); o.put('}'); } else @@ -6538,18 +6536,18 @@ class basic_json // first n-1 elements for (auto i = val.m_value.array->cbegin(); i != val.m_value.array->cend() - 1; ++i) { - o.write(indent_string.c_str(), new_indent); + o.write(indent_string.c_str(), static_cast(new_indent)); dump(*i, true, indent_step, new_indent); o.write(",\n", 2); } // last element assert(not val.m_value.array->empty()); - o.write(indent_string.c_str(), new_indent); + o.write(indent_string.c_str(), static_cast(new_indent)); dump(val.m_value.array->back(), true, indent_step, new_indent); o.put('\n'); - o.write(indent_string.c_str(), current_indent); + o.write(indent_string.c_str(), static_cast(current_indent)); o.put(']'); } else @@ -7848,7 +7846,7 @@ class basic_json const auto N = j.m_value.string->size(); if (N <= 0x17) { - v.push_back(0x60 + static_cast(N)); // 1 byte for string + size + v.push_back(static_cast(0x60 + N)); // 1 byte for string + size } else if (N <= 0xff) { @@ -7884,7 +7882,7 @@ class basic_json const auto N = j.m_value.array->size(); if (N <= 0x17) { - v.push_back(0x80 + static_cast(N)); // 1 byte for array + size + v.push_back(static_cast(0x80 + N)); // 1 byte for array + size } else if (N <= 0xff) { @@ -7922,7 +7920,7 @@ class basic_json const auto N = j.m_value.object->size(); if (N <= 0x17) { - v.push_back(0xa0 + static_cast(N)); // 1 byte for object + size + v.push_back(static_cast(0xa0 + N)); // 1 byte for object + size } else if (N <= 0xff) { @@ -8653,6 +8651,7 @@ class basic_json const auto len = static_cast(v[current_idx] - 0xa0); for (size_t i = 0; i < len; ++i) { + cbor_expect_string(v, idx); std::string key = from_cbor_internal(v, idx); result[key] = from_cbor_internal(v, idx); } @@ -8666,6 +8665,7 @@ class basic_json idx += 1; // skip 1 size byte for (size_t i = 0; i < len; ++i) { + cbor_expect_string(v, idx); std::string key = from_cbor_internal(v, idx); result[key] = from_cbor_internal(v, idx); } @@ -8679,6 +8679,7 @@ class basic_json idx += 2; // skip 2 size bytes for (size_t i = 0; i < len; ++i) { + cbor_expect_string(v, idx); std::string key = from_cbor_internal(v, idx); result[key] = from_cbor_internal(v, idx); } @@ -8692,6 +8693,7 @@ class basic_json idx += 4; // skip 4 size bytes for (size_t i = 0; i < len; ++i) { + cbor_expect_string(v, idx); std::string key = from_cbor_internal(v, idx); result[key] = from_cbor_internal(v, idx); } @@ -8705,6 +8707,7 @@ class basic_json idx += 8; // skip 8 size bytes for (size_t i = 0; i < len; ++i) { + cbor_expect_string(v, idx); std::string key = from_cbor_internal(v, idx); result[key] = from_cbor_internal(v, idx); } @@ -8716,6 +8719,7 @@ class basic_json basic_json result = value_t::object; while (check_length(v.size(), 1, idx), v[idx] != 0xff) { + cbor_expect_string(v, idx); std::string key = from_cbor_internal(v, idx); result[key] = from_cbor_internal(v, idx); } @@ -8864,6 +8868,9 @@ class basic_json - float 32 (0xca) - fixext 1 - fixext 16 (0xd4..0xd8) + @note Any MessagePack output created @ref to_msgpack can be successfully + parsed by @ref from_msgpack. + @param[in] j JSON value to serialize @return MessagePack serialization as byte vector @@ -8892,6 +8899,46 @@ class basic_json Deserializes a given byte vector @a v to a JSON value using the MessagePack serialization format. + The library maps MessagePack types to JSON value types as follows: + + MessagePack type | JSON value type | first byte + ---------------- | --------------- | ---------- + positive fixint | number_unsigned | 0x00..0x7f + fixmap | object | 0x80..0x8f + fixarray | array | 0x90..0x9f + fixstr | string | 0xa0..0xbf + nil | `null` | 0xc0 + false | `false` | 0xc2 + true | `true` | 0xc3 + float 32 | number_float | 0xca + float 64 | number_float | 0xcb + uint 8 | number_unsigned | 0xcc + uint 16 | number_unsigned | 0xcd + uint 32 | number_unsigned | 0xce + uint 64 | number_unsigned | 0xcf + int 8 | number_integer | 0xd0 + int 16 | number_integer | 0xd1 + int 32 | number_integer | 0xd2 + int 64 | number_integer | 0xd3 + str 8 | string | 0xd9 + str 16 | string | 0xda + str 32 | string | 0xdb + array 16 | array | 0xdc + array 32 | array | 0xdd + map 16 | object | 0xde + map 32 | object | 0xdf + negative fixint | number_integer | 0xe0-0xff + + @warning The mapping is **incomplete** in the sense that not all + MessagePack types can be converted to a JSON value. The following + MessagePack types are not supported and will yield parse errors: + - bin 8 - bin 32 (0xc4..0xc6) + - ext 8 - ext 32 (0xc7..0xc9) + - fixext 1 - fixext 16 (0xd4..0xd8) + + @note Any MessagePack output created @ref to_msgpack can be successfully + parsed by @ref from_msgpack. + @param[in] v a byte vector in MessagePack format @param[in] start_index the index to start reading from @a v (0 by default) @return deserialized JSON value @@ -8899,6 +8946,7 @@ class basic_json @throw parse_error.110 if the given vector ends prematurely @throw parse_error.112 if unsupported features from MessagePack were used in the given vector @a v or if the input is not valid MessagePack + @throw parse_error.113 if a string was expected as map key, but not found @complexity Linear in the size of the byte vector @a v. @@ -9014,6 +9062,66 @@ class basic_json Deserializes a given byte vector @a v to a JSON value using the CBOR (Concise Binary Object Representation) serialization format. + The library maps CBOR types to JSON value types as follows: + + CBOR type | JSON value type | first byte + ---------------------- | --------------- | ---------- + Integer | number_unsigned | 0x00..0x17 + Unsigned integer | number_unsigned | 0x18 + Unsigned integer | number_unsigned | 0x19 + Unsigned integer | number_unsigned | 0x1a + Unsigned integer | number_unsigned | 0x1b + Negative integer | number_integer | 0x20..0x37 + Negative integer | number_integer | 0x38 + Negative integer | number_integer | 0x39 + Negative integer | number_integer | 0x3a + Negative integer | number_integer | 0x3b + Negative integer | number_integer | 0x40..0x57 + UTF-8 string | string | 0x60..0x77 + UTF-8 string | string | 0x78 + UTF-8 string | string | 0x79 + UTF-8 string | string | 0x7a + UTF-8 string | string | 0x7b + UTF-8 string | string | 0x7f + array | array | 0x80..0x97 + array | array | 0x98 + array | array | 0x99 + array | array | 0x9a + array | array | 0x9b + array | array | 0x9f + map | object | 0xa0..0xb7 + map | object | 0xb8 + map | object | 0xb9 + map | object | 0xba + map | object | 0xbb + map | object | 0xbf + False | `false` | 0xf4 + True | `true` | 0xf5 + Nill | `null` | 0xf6 + Half-Precision Float | number_float | 0xf9 + Single-Precision Float | number_float | 0xfa + Double-Precision Float | number_float | 0xfb + + @warning The mapping is **incomplete** in the sense that not all CBOR + types can be converted to a JSON value. The following CBOR types + are not supported and will yield parse errors (parse_error.112): + - byte strings (0x40..0x5f) + - date/time (0xc0..0xc1) + - bignum (0xc2..0xc3) + - decimal fraction (0xc4) + - bigfloat (0xc5) + - tagged items (0xc6..0xd4, 0xd8..0xdb) + - expected conversions (0xd5..0xd7) + - simple values (0xe0..0xf3, 0xf8) + - undefined (0xf7) + + @warning CBOR allows map keys of any type, whereas JSON only allows + strings as keys in object values. Therefore, CBOR maps with keys + other than UTF-8 strings are rejected (parse_error.113). + + @note Any CBOR output created @ref to_cbor can be successfully parsed by + @ref from_cbor. + @param[in] v a byte vector in CBOR format @param[in] start_index the index to start reading from @a v (0 by default) @return deserialized JSON value @@ -9021,6 +9129,7 @@ class basic_json @throw parse_error.110 if the given vector ends prematurely @throw parse_error.112 if unsupported features from CBOR were used in the given vector @a v or if the input is not valid CBOR + @throw parse_error.113 if a string was expected as map key, but not found @complexity Linear in the size of the byte vector @a v. @@ -11456,7 +11565,7 @@ class basic_json reference_token.end(), [](const char x) { - return std::isdigit(x); + return (x >= '0' and x <= '9'); }); // change value to array for numbers or "-" or to object diff --git a/test/src/unit-cbor.cpp b/test/src/unit-cbor.cpp index b4b53975..f1609f4f 100644 --- a/test/src/unit-cbor.cpp +++ b/test/src/unit-cbor.cpp @@ -130,14 +130,14 @@ TEST_CASE("CBOR") // check individual bytes CHECK(result[0] == 0x3b); - uint64_t restored = static_cast((static_cast(result[1]) << 070) + + uint64_t restored = (static_cast(result[1]) << 070) + (static_cast(result[2]) << 060) + (static_cast(result[3]) << 050) + (static_cast(result[4]) << 040) + (static_cast(result[5]) << 030) + (static_cast(result[6]) << 020) + (static_cast(result[7]) << 010) + - static_cast(result[8])); + static_cast(result[8]); CHECK(restored == positive); CHECK(-1 - static_cast(restored) == i); @@ -182,10 +182,10 @@ TEST_CASE("CBOR") // check individual bytes CHECK(result[0] == 0x3a); - uint32_t restored = static_cast((static_cast(result[1]) << 030) + + uint32_t restored = (static_cast(result[1]) << 030) + (static_cast(result[2]) << 020) + (static_cast(result[3]) << 010) + - static_cast(result[4])); + static_cast(result[4]); CHECK(restored == positive); CHECK(-1ll - restored == i); @@ -220,7 +220,7 @@ TEST_CASE("CBOR") // check individual bytes CHECK(result[0] == 0x39); - uint16_t restored = static_cast(result[1]) * 256 + static_cast(result[2]); + uint16_t restored = static_cast(static_cast(result[1]) * 256 + static_cast(result[2])); CHECK(restored == positive); CHECK(-1 - restored == i); @@ -289,7 +289,7 @@ TEST_CASE("CBOR") // create expected byte vector std::vector expected; - expected.push_back(0x20 - 1 - static_cast(i)); + expected.push_back(static_cast(0x20 - 1 - static_cast(i))); // compare result + size const auto result = json::to_cbor(j); @@ -392,7 +392,7 @@ TEST_CASE("CBOR") // check individual bytes CHECK(result[0] == 0x19); - uint16_t restored = static_cast(result[1]) * 256 + static_cast(result[2]); + uint16_t restored = static_cast(static_cast(result[1]) * 256 + static_cast(result[2])); CHECK(restored == i); // roundtrip @@ -431,10 +431,10 @@ TEST_CASE("CBOR") // check individual bytes CHECK(result[0] == 0x1a); - uint32_t restored = static_cast((static_cast(result[1]) << 030) + + uint32_t restored = (static_cast(result[1]) << 030) + (static_cast(result[2]) << 020) + (static_cast(result[3]) << 010) + - static_cast(result[4])); + static_cast(result[4]); CHECK(restored == i); // roundtrip @@ -477,14 +477,14 @@ TEST_CASE("CBOR") // check individual bytes CHECK(result[0] == 0x1b); - uint64_t restored = static_cast((static_cast(result[1]) << 070) + + uint64_t restored = (static_cast(result[1]) << 070) + (static_cast(result[2]) << 060) + (static_cast(result[3]) << 050) + (static_cast(result[4]) << 040) + (static_cast(result[5]) << 030) + (static_cast(result[6]) << 020) + (static_cast(result[7]) << 010) + - static_cast(result[8])); + static_cast(result[8]); CHECK(restored == i); // roundtrip @@ -616,7 +616,7 @@ TEST_CASE("CBOR") // check individual bytes CHECK(result[0] == 0x19); - uint16_t restored = static_cast(result[1]) * 256 + static_cast(result[2]); + uint16_t restored = static_cast(static_cast(result[1]) * 256 + static_cast(result[2])); CHECK(restored == i); // roundtrip @@ -654,10 +654,10 @@ TEST_CASE("CBOR") // check individual bytes CHECK(result[0] == 0x1a); - uint32_t restored = static_cast((static_cast(result[1]) << 030) + + uint32_t restored = (static_cast(result[1]) << 030) + (static_cast(result[2]) << 020) + (static_cast(result[3]) << 010) + - static_cast(result[4])); + static_cast(result[4]); CHECK(restored == i); // roundtrip @@ -699,14 +699,14 @@ TEST_CASE("CBOR") // check individual bytes CHECK(result[0] == 0x1b); - uint64_t restored = static_cast((static_cast(result[1]) << 070) + + uint64_t restored = (static_cast(result[1]) << 070) + (static_cast(result[2]) << 060) + (static_cast(result[3]) << 050) + (static_cast(result[4]) << 040) + (static_cast(result[5]) << 030) + (static_cast(result[6]) << 020) + (static_cast(result[7]) << 010) + - static_cast(result[8])); + static_cast(result[8]); CHECK(restored == i); // roundtrip @@ -1538,6 +1538,83 @@ TEST_CASE("CBOR roundtrips", "[hide]") } } +TEST_CASE("all first bytes", "[!throws]") +{ + // these bytes will fail immediately with exception parse_error.112 + std::set unsupported = + { + //// types not supported by this library + + // byte strings + 0x40, 0x41, 0x42, 0x43, 0x44, 0x45, 0x46, 0x47, + 0x48, 0x49, 0x4a, 0x4b, 0x4c, 0x4d, 0x4e, 0x4f, + 0x50, 0x51, 0x52, 0x53, 0x54, 0x55, 0x56, 0x57, + // byte strings + 0x58, 0x59, 0x5a, 0x5b, 0x5f, + // date/time + 0xc0, 0xc1, + // bignum + 0xc2, 0xc3, + // decimal fracion + 0xc4, + // bigfloat + 0xc5, + // tagged item + 0xc6, 0xc7, 0xc8, 0xc9, 0xca, 0xcb, 0xcc, 0xcd, + 0xce, 0xcf, 0xd0, 0xd1, 0xd2, 0xd3, 0xd4, 0xd8, + 0xd9, 0xda, 0xdb, + // expected conversion + 0xd5, 0xd6, 0xd7, + // simple value + 0xe0, 0xe1, 0xe2, 0xe3, 0xe4, 0xe5, 0xe6, 0xe7, + 0xe8, 0xe9, 0xea, 0xeb, 0xec, 0xed, 0xef, 0xf0, + 0xf1, 0xf2, 0xf3, + 0xf8, + // undefined + 0xf7, + + //// bytes not specified by CBOR + + 0x1c, 0x1d, 0x1e, 0x1f, + 0x3c, 0x3d, 0x3e, 0x3f, + 0x5c, 0x5d, 0x5e, + 0x7c, 0x7d, 0x7e, + 0x9c, 0x9d, 0x9e, + 0xbc, 0xbd, 0xbe, + 0xdc, 0xdd, 0xde, 0xdf, + 0xee, + 0xfc, 0xfe, 0xfd, + + /// break cannot be the first byte + + 0xff + }; + + for (auto i = 0; i < 256; ++i) + { + const auto byte = static_cast(i); + + try + { + json::from_cbor({byte}); + } + catch (const json::parse_error& e) + { + // check that parse_error.112 is only thrown if the + // first byte is in the unsupported set + CAPTURE(e.what()); + if (std::find(unsupported.begin(), unsupported.end(), byte) != unsupported.end()) + { + CHECK(e.id == 112); + } + else + { + CHECK(e.id != 112); + } + } + } +} + TEST_CASE("examples from RFC 7049 Appendix A") { SECTION("numbers") diff --git a/test/src/unit-class_lexer.cpp b/test/src/unit-class_lexer.cpp index b9e602b7..b88a5291 100644 --- a/test/src/unit-class_lexer.cpp +++ b/test/src/unit-class_lexer.cpp @@ -139,7 +139,7 @@ TEST_CASE("lexer class") for (int c = 1; c < 128; ++c) { // create string from the ASCII code - const auto s = std::string(1, c); + const auto s = std::string(1, static_cast(c)); // store scan() result const auto res = json::lexer(reinterpret_cast(s.c_str()), 1).scan(); diff --git a/test/src/unit-class_parser.cpp b/test/src/unit-class_parser.cpp index 7f2c9a95..2a5c07cf 100644 --- a/test/src/unit-class_parser.cpp +++ b/test/src/unit-class_parser.cpp @@ -494,7 +494,7 @@ TEST_CASE("parser class") // invalid escapes for (int c = 1; c < 128; ++c) { - auto s = std::string("\"\\") + std::string(1, c) + "\""; + auto s = std::string("\"\\") + std::string(1, static_cast(c)) + "\""; switch (c) { @@ -574,10 +574,10 @@ TEST_CASE("parser class") std::string s = "\"\\u"; // create a string with the iterated character at each position - auto s1 = s + "000" + std::string(1, c) + "\""; - auto s2 = s + "00" + std::string(1, c) + "0\""; - auto s3 = s + "0" + std::string(1, c) + "00\""; - auto s4 = s + std::string(1, c) + "000\""; + auto s1 = s + "000" + std::string(1, static_cast(c)) + "\""; + auto s2 = s + "00" + std::string(1, static_cast(c)) + "0\""; + auto s3 = s + "0" + std::string(1, static_cast(c)) + "00\""; + auto s4 = s + std::string(1, static_cast(c)) + "000\""; if (valid(c)) { diff --git a/test/src/unit-concepts.cpp b/test/src/unit-concepts.cpp index 79be78cd..d2c24866 100644 --- a/test/src/unit-concepts.cpp +++ b/test/src/unit-concepts.cpp @@ -72,8 +72,8 @@ TEST_CASE("concepts") // X::size_type must return an unsigned integer CHECK((std::is_unsigned::value)); // X::size_type can represent any non-negative value of X::difference_type - CHECK(static_cast(std::numeric_limits::max()) <= - static_cast(std::numeric_limits::max())); + CHECK(static_cast(std::numeric_limits::max()) <= + std::numeric_limits::max()); // the expression "X u" has the post-condition "u.empty()" { diff --git a/test/src/unit-element_access1.cpp b/test/src/unit-element_access1.cpp index b8a57af4..eb4a92cb 100644 --- a/test/src/unit-element_access1.cpp +++ b/test/src/unit-element_access1.cpp @@ -48,7 +48,7 @@ TEST_CASE("element access 1") CHECK(j.at(3) == json(nullptr)); CHECK(j.at(4) == json("string")); CHECK(j.at(5) == json(42.23)); - CHECK(j.at(6) == json(json::object())); + CHECK(j.at(6) == json::object()); CHECK(j.at(7) == json({1, 2, 3})); CHECK(j_const.at(0) == json(1)); @@ -57,7 +57,7 @@ TEST_CASE("element access 1") CHECK(j_const.at(3) == json(nullptr)); CHECK(j_const.at(4) == json("string")); CHECK(j_const.at(5) == json(42.23)); - CHECK(j_const.at(6) == json(json::object())); + CHECK(j_const.at(6) == json::object()); CHECK(j_const.at(7) == json({1, 2, 3})); } @@ -171,7 +171,7 @@ TEST_CASE("element access 1") CHECK(j[3] == json(nullptr)); CHECK(j[4] == json("string")); CHECK(j[5] == json(42.23)); - CHECK(j[6] == json(json::object())); + CHECK(j[6] == json::object()); CHECK(j[7] == json({1, 2, 3})); CHECK(j_const[0] == json(1)); @@ -180,7 +180,7 @@ TEST_CASE("element access 1") CHECK(j_const[3] == json(nullptr)); CHECK(j_const[4] == json("string")); CHECK(j_const[5] == json(42.23)); - CHECK(j_const[6] == json(json::object())); + CHECK(j_const[6] == json::object()); CHECK(j_const[7] == json({1, 2, 3})); } diff --git a/test/src/unit-element_access2.cpp b/test/src/unit-element_access2.cpp index 3533bdcd..ec03507e 100644 --- a/test/src/unit-element_access2.cpp +++ b/test/src/unit-element_access2.cpp @@ -48,7 +48,7 @@ TEST_CASE("element access 2") CHECK(j.at("null") == json(nullptr)); CHECK(j.at("string") == json("hello world")); CHECK(j.at("floating") == json(42.23)); - CHECK(j.at("object") == json(json::object())); + CHECK(j.at("object") == json::object()); CHECK(j.at("array") == json({1, 2, 3})); CHECK(j_const.at("integer") == json(1)); @@ -57,7 +57,7 @@ TEST_CASE("element access 2") CHECK(j_const.at("null") == json(nullptr)); CHECK(j_const.at("string") == json("hello world")); CHECK(j_const.at("floating") == json(42.23)); - CHECK(j_const.at("object") == json(json::object())); + CHECK(j_const.at("object") == json::object()); CHECK(j_const.at("array") == json({1, 2, 3})); } @@ -161,7 +161,7 @@ TEST_CASE("element access 2") CHECK(j.value("string", std::string("bar")) == "hello world"); CHECK(j.value("floating", 12.34) == Approx(42.23)); CHECK(j.value("floating", 12) == 42); - CHECK(j.value("object", json({{"foo", "bar"}})) == json(json::object())); + CHECK(j.value("object", json({{"foo", "bar"}})) == json::object()); CHECK(j.value("array", json({10, 100})) == json({1, 2, 3})); CHECK(j_const.value("integer", 2) == 1); @@ -173,7 +173,7 @@ TEST_CASE("element access 2") CHECK(j_const.value("string", std::string("bar")) == "hello world"); CHECK(j_const.value("floating", 12.34) == Approx(42.23)); CHECK(j_const.value("floating", 12) == 42); - CHECK(j_const.value("object", json({{"foo", "bar"}})) == json(json::object())); + CHECK(j_const.value("object", json({{"foo", "bar"}})) == json::object()); CHECK(j_const.value("array", json({10, 100})) == json({1, 2, 3})); } @@ -298,7 +298,7 @@ TEST_CASE("element access 2") CHECK(j.value("/string"_json_pointer, std::string("bar")) == "hello world"); CHECK(j.value("/floating"_json_pointer, 12.34) == Approx(42.23)); CHECK(j.value("/floating"_json_pointer, 12) == 42); - CHECK(j.value("/object"_json_pointer, json({{"foo", "bar"}})) == json(json::object())); + CHECK(j.value("/object"_json_pointer, json({{"foo", "bar"}})) == json::object()); CHECK(j.value("/array"_json_pointer, json({10, 100})) == json({1, 2, 3})); CHECK(j_const.value("/integer"_json_pointer, 2) == 1); @@ -310,7 +310,7 @@ TEST_CASE("element access 2") CHECK(j_const.value("/string"_json_pointer, std::string("bar")) == "hello world"); CHECK(j_const.value("/floating"_json_pointer, 12.34) == Approx(42.23)); CHECK(j_const.value("/floating"_json_pointer, 12) == 42); - CHECK(j_const.value("/object"_json_pointer, json({{"foo", "bar"}})) == json(json::object())); + CHECK(j_const.value("/object"_json_pointer, json({{"foo", "bar"}})) == json::object()); CHECK(j_const.value("/array"_json_pointer, json({10, 100})) == json({1, 2, 3})); } @@ -435,7 +435,7 @@ TEST_CASE("element access 2") CHECK(j["floating"] == json(42.23)); CHECK(j[json::object_t::key_type("floating")] == j["floating"]); - CHECK(j["object"] == json(json::object())); + CHECK(j["object"] == json::object()); CHECK(j[json::object_t::key_type("object")] == j["object"]); CHECK(j["array"] == json({1, 2, 3})); @@ -456,7 +456,7 @@ TEST_CASE("element access 2") CHECK(j_const["floating"] == json(42.23)); CHECK(j_const[json::object_t::key_type("floating")] == j["floating"]); - CHECK(j_const["object"] == json(json::object())); + CHECK(j_const["object"] == json::object()); CHECK(j_const[json::object_t::key_type("object")] == j["object"]); CHECK(j_const["array"] == json({1, 2, 3})); diff --git a/test/src/unit-msgpack.cpp b/test/src/unit-msgpack.cpp index 1da33153..5d157812 100644 --- a/test/src/unit-msgpack.cpp +++ b/test/src/unit-msgpack.cpp @@ -203,7 +203,7 @@ TEST_CASE("MessagePack") // check individual bytes CHECK(result[0] == 0xcd); - uint16_t restored = static_cast(result[1]) * 256 + static_cast(result[2]); + uint16_t restored = static_cast(static_cast(result[1]) * 256 + static_cast(result[2])); CHECK(restored == i); // roundtrip @@ -242,10 +242,10 @@ TEST_CASE("MessagePack") // check individual bytes CHECK(result[0] == 0xce); - uint32_t restored = static_cast((static_cast(result[1]) << 030) + + uint32_t restored = (static_cast(result[1]) << 030) + (static_cast(result[2]) << 020) + (static_cast(result[3]) << 010) + - static_cast(result[4])); + static_cast(result[4]); CHECK(restored == i); // roundtrip @@ -288,14 +288,14 @@ TEST_CASE("MessagePack") // check individual bytes CHECK(result[0] == 0xcf); - uint64_t restored = static_cast((static_cast(result[1]) << 070) + + uint64_t restored = (static_cast(result[1]) << 070) + (static_cast(result[2]) << 060) + (static_cast(result[3]) << 050) + (static_cast(result[4]) << 040) + (static_cast(result[5]) << 030) + (static_cast(result[6]) << 020) + (static_cast(result[7]) << 010) + - static_cast(result[8])); + static_cast(result[8]); CHECK(restored == i); // roundtrip @@ -415,10 +415,10 @@ TEST_CASE("MessagePack") // check individual bytes CHECK(result[0] == 0xd2); - uint32_t restored = static_cast((static_cast(result[1]) << 030) + + uint32_t restored = (static_cast(result[1]) << 030) + (static_cast(result[2]) << 020) + (static_cast(result[3]) << 010) + - static_cast(result[4])); + static_cast(result[4]); CHECK(restored == i); // roundtrip @@ -460,14 +460,14 @@ TEST_CASE("MessagePack") // check individual bytes CHECK(result[0] == 0xd3); - int64_t restored = static_cast((static_cast(result[1]) << 070) + - (static_cast(result[2]) << 060) + - (static_cast(result[3]) << 050) + - (static_cast(result[4]) << 040) + - (static_cast(result[5]) << 030) + - (static_cast(result[6]) << 020) + - (static_cast(result[7]) << 010) + - static_cast(result[8])); + int64_t restored = (static_cast(result[1]) << 070) + + (static_cast(result[2]) << 060) + + (static_cast(result[3]) << 050) + + (static_cast(result[4]) << 040) + + (static_cast(result[5]) << 030) + + (static_cast(result[6]) << 020) + + (static_cast(result[7]) << 010) + + static_cast(result[8]); CHECK(restored == i); // roundtrip @@ -564,7 +564,7 @@ TEST_CASE("MessagePack") // check individual bytes CHECK(result[0] == 0xcd); - uint16_t restored = static_cast(result[1]) * 256 + static_cast(result[2]); + uint16_t restored = static_cast(static_cast(result[1]) * 256 + static_cast(result[2])); CHECK(restored == i); // roundtrip @@ -602,10 +602,10 @@ TEST_CASE("MessagePack") // check individual bytes CHECK(result[0] == 0xce); - uint32_t restored = static_cast((static_cast(result[1]) << 030) + + uint32_t restored = (static_cast(result[1]) << 030) + (static_cast(result[2]) << 020) + (static_cast(result[3]) << 010) + - static_cast(result[4])); + static_cast(result[4]); CHECK(restored == i); // roundtrip @@ -647,14 +647,14 @@ TEST_CASE("MessagePack") // check individual bytes CHECK(result[0] == 0xcf); - uint64_t restored = static_cast((static_cast(result[1]) << 070) + + uint64_t restored = (static_cast(result[1]) << 070) + (static_cast(result[2]) << 060) + (static_cast(result[3]) << 050) + (static_cast(result[4]) << 040) + (static_cast(result[5]) << 030) + (static_cast(result[6]) << 020) + (static_cast(result[7]) << 010) + - static_cast(result[8])); + static_cast(result[8]); CHECK(restored == i); // roundtrip diff --git a/test/src/unit-regression.cpp b/test/src/unit-regression.cpp index 3f17a917..f7f4c350 100644 --- a/test/src/unit-regression.cpp +++ b/test/src/unit-regression.cpp @@ -412,17 +412,17 @@ TEST_CASE("regression tests") class CommaDecimalSeparator : public std::numpunct { protected: - char do_decimal_point() const + char do_decimal_point() const override { return ','; } - char do_thousands_sep() const + char do_thousands_sep() const override { return '.'; } - std::string do_grouping() const + std::string do_grouping() const override { return "\03"; } @@ -755,7 +755,7 @@ TEST_CASE("regression tests") }; CHECK_THROWS_AS(json::from_cbor(vec), json::parse_error); CHECK_THROWS_WITH(json::from_cbor(vec), - "[json.exception.parse_error.110] parse error at 137: cannot read 1 bytes from vector"); + "[json.exception.parse_error.113] parse error at 2: expected a CBOR string; last byte: 0x98"); // related test case: nonempty UTF-8 string (indefinite length) std::vector vec1 {0x7f, 0x61, 0x61}; @@ -808,7 +808,7 @@ TEST_CASE("regression tests") }; CHECK_THROWS_AS(json::from_cbor(vec1), json::parse_error); CHECK_THROWS_WITH(json::from_cbor(vec1), - "[json.exception.parse_error.110] parse error at 49: cannot read 4 bytes from vector"); + "[json.exception.parse_error.113] parse error at 13: expected a CBOR string; last byte: 0xb4"); // related test case: double-precision std::vector vec2 @@ -822,7 +822,7 @@ TEST_CASE("regression tests") }; CHECK_THROWS_AS(json::from_cbor(vec2), json::parse_error); CHECK_THROWS_WITH(json::from_cbor(vec2), - "[json.exception.parse_error.110] parse error at 49: cannot read 8 bytes from vector"); + "[json.exception.parse_error.113] parse error at 13: expected a CBOR string; last byte: 0xb4"); } SECTION("issue #452 - Heap-buffer-overflow (OSS-Fuzz issue 585)") From cd43600e6fe9f7e03594e6a898ab2b61293d7c58 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Thu, 16 Mar 2017 19:02:05 +0100 Subject: [PATCH 088/530] :lipstick: cleanup after #508 --- src/json.hpp | 1411 +++++++++++++++++++++++---------- src/json.hpp.re2c | 5 +- test/src/unit-conversions.cpp | 16 +- 3 files changed, 991 insertions(+), 441 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index fd65254f..9e539790 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -802,8 +802,9 @@ template ::value, int> = 0> -void to_json(BasicJsonType &j, T (&arr)[N]) { - external_constructor::construct(j, arr); +void to_json(BasicJsonType& j, T (&arr)[N]) +{ + external_constructor::construct(j, arr); } /////////////// @@ -10442,490 +10443,1038 @@ class basic_json m_start = m_cursor; assert(m_start != nullptr); - - { - lexer_char_t yych; - unsigned int yyaccept = 0; - static const unsigned char yybm[] = { - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 32, 32, 0, 0, 32, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 160, 128, 0, 128, 128, 128, 128, 128, - 128, 128, 128, 128, 128, 128, 128, 128, - 192, 192, 192, 192, 192, 192, 192, 192, - 192, 192, 128, 128, 128, 128, 128, 128, - 128, 128, 128, 128, 128, 128, 128, 128, - 128, 128, 128, 128, 128, 128, 128, 128, - 128, 128, 128, 128, 128, 128, 128, 128, - 128, 128, 128, 128, 0, 128, 128, 128, - 128, 128, 128, 128, 128, 128, 128, 128, - 128, 128, 128, 128, 128, 128, 128, 128, - 128, 128, 128, 128, 128, 128, 128, 128, - 128, 128, 128, 128, 128, 128, 128, 128, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - }; - if ((m_limit - m_cursor) < 5) fill_line_buffer(5); // LCOV_EXCL_LINE - yych = *m_cursor; - if (yybm[0+yych] & 32) { - goto basic_json_parser_6; - } - if (yych <= '[') { - if (yych <= '-') { - if (yych <= '"') { - if (yych <= 0x00) goto basic_json_parser_2; - if (yych <= '!') goto basic_json_parser_4; - goto basic_json_parser_9; - } else { - if (yych <= '+') goto basic_json_parser_4; - if (yych <= ',') goto basic_json_parser_10; - goto basic_json_parser_12; - } - } else { - if (yych <= '9') { - if (yych <= '/') goto basic_json_parser_4; - if (yych <= '0') goto basic_json_parser_13; - goto basic_json_parser_15; - } else { - if (yych <= ':') goto basic_json_parser_17; - if (yych <= 'Z') goto basic_json_parser_4; - goto basic_json_parser_19; - } - } - } else { - if (yych <= 'n') { - if (yych <= 'e') { - if (yych == ']') goto basic_json_parser_21; - goto basic_json_parser_4; - } else { - if (yych <= 'f') goto basic_json_parser_23; - if (yych <= 'm') goto basic_json_parser_4; - goto basic_json_parser_24; - } - } else { - if (yych <= 'z') { - if (yych == 't') goto basic_json_parser_25; - goto basic_json_parser_4; - } else { - if (yych <= '{') goto basic_json_parser_26; - if (yych == '}') goto basic_json_parser_28; - goto basic_json_parser_4; - } - } - } + + { + lexer_char_t yych; + unsigned int yyaccept = 0; + static const unsigned char yybm[] = + { + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 32, 32, 0, 0, 32, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 160, 128, 0, 128, 128, 128, 128, 128, + 128, 128, 128, 128, 128, 128, 128, 128, + 192, 192, 192, 192, 192, 192, 192, 192, + 192, 192, 128, 128, 128, 128, 128, 128, + 128, 128, 128, 128, 128, 128, 128, 128, + 128, 128, 128, 128, 128, 128, 128, 128, + 128, 128, 128, 128, 128, 128, 128, 128, + 128, 128, 128, 128, 0, 128, 128, 128, + 128, 128, 128, 128, 128, 128, 128, 128, + 128, 128, 128, 128, 128, 128, 128, 128, + 128, 128, 128, 128, 128, 128, 128, 128, + 128, 128, 128, 128, 128, 128, 128, 128, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, + }; + if ((m_limit - m_cursor) < 5) + { + fill_line_buffer(5); // LCOV_EXCL_LINE + } + yych = *m_cursor; + if (yybm[0 + yych] & 32) + { + goto basic_json_parser_6; + } + if (yych <= '[') + { + if (yych <= '-') + { + if (yych <= '"') + { + if (yych <= 0x00) + { + goto basic_json_parser_2; + } + if (yych <= '!') + { + goto basic_json_parser_4; + } + goto basic_json_parser_9; + } + else + { + if (yych <= '+') + { + goto basic_json_parser_4; + } + if (yych <= ',') + { + goto basic_json_parser_10; + } + goto basic_json_parser_12; + } + } + else + { + if (yych <= '9') + { + if (yych <= '/') + { + goto basic_json_parser_4; + } + if (yych <= '0') + { + goto basic_json_parser_13; + } + goto basic_json_parser_15; + } + else + { + if (yych <= ':') + { + goto basic_json_parser_17; + } + if (yych <= 'Z') + { + goto basic_json_parser_4; + } + goto basic_json_parser_19; + } + } + } + else + { + if (yych <= 'n') + { + if (yych <= 'e') + { + if (yych == ']') + { + goto basic_json_parser_21; + } + goto basic_json_parser_4; + } + else + { + if (yych <= 'f') + { + goto basic_json_parser_23; + } + if (yych <= 'm') + { + goto basic_json_parser_4; + } + goto basic_json_parser_24; + } + } + else + { + if (yych <= 'z') + { + if (yych == 't') + { + goto basic_json_parser_25; + } + goto basic_json_parser_4; + } + else + { + if (yych <= '{') + { + goto basic_json_parser_26; + } + if (yych == '}') + { + goto basic_json_parser_28; + } + goto basic_json_parser_4; + } + } + } basic_json_parser_2: - ++m_cursor; - { last_token_type = token_type::end_of_input; break; } + ++m_cursor; + { + last_token_type = token_type::end_of_input; + break; + } basic_json_parser_4: - ++m_cursor; + ++m_cursor; basic_json_parser_5: - { last_token_type = token_type::parse_error; break; } + { + last_token_type = token_type::parse_error; + break; + } basic_json_parser_6: - ++m_cursor; - if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE - yych = *m_cursor; - if (yybm[0+yych] & 32) { - goto basic_json_parser_6; - } - { position += static_cast((m_cursor - m_start)); continue; } + ++m_cursor; + if (m_limit <= m_cursor) + { + fill_line_buffer(1); // LCOV_EXCL_LINE + } + yych = *m_cursor; + if (yybm[0 + yych] & 32) + { + goto basic_json_parser_6; + } + { + position += static_cast((m_cursor - m_start)); + continue; + } basic_json_parser_9: - yyaccept = 0; - yych = *(m_marker = ++m_cursor); - if (yych <= 0x1F) goto basic_json_parser_5; - if (yych <= 0x7F) goto basic_json_parser_31; - if (yych <= 0xC1) goto basic_json_parser_5; - if (yych <= 0xF4) goto basic_json_parser_31; - goto basic_json_parser_5; + yyaccept = 0; + yych = *(m_marker = ++m_cursor); + if (yych <= 0x1F) + { + goto basic_json_parser_5; + } + if (yych <= 0x7F) + { + goto basic_json_parser_31; + } + if (yych <= 0xC1) + { + goto basic_json_parser_5; + } + if (yych <= 0xF4) + { + goto basic_json_parser_31; + } + goto basic_json_parser_5; basic_json_parser_10: - ++m_cursor; - { last_token_type = token_type::value_separator; break; } + ++m_cursor; + { + last_token_type = token_type::value_separator; + break; + } basic_json_parser_12: - yych = *++m_cursor; - if (yych <= '/') goto basic_json_parser_5; - if (yych <= '0') goto basic_json_parser_43; - if (yych <= '9') goto basic_json_parser_45; - goto basic_json_parser_5; + yych = *++m_cursor; + if (yych <= '/') + { + goto basic_json_parser_5; + } + if (yych <= '0') + { + goto basic_json_parser_43; + } + if (yych <= '9') + { + goto basic_json_parser_45; + } + goto basic_json_parser_5; basic_json_parser_13: - yyaccept = 1; - yych = *(m_marker = ++m_cursor); - if (yych <= '9') { - if (yych == '.') goto basic_json_parser_47; - if (yych >= '0') goto basic_json_parser_48; - } else { - if (yych <= 'E') { - if (yych >= 'E') goto basic_json_parser_51; - } else { - if (yych == 'e') goto basic_json_parser_51; - } - } + yyaccept = 1; + yych = *(m_marker = ++m_cursor); + if (yych <= '9') + { + if (yych == '.') + { + goto basic_json_parser_47; + } + if (yych >= '0') + { + goto basic_json_parser_48; + } + } + else + { + if (yych <= 'E') + { + if (yych >= 'E') + { + goto basic_json_parser_51; + } + } + else + { + if (yych == 'e') + { + goto basic_json_parser_51; + } + } + } basic_json_parser_14: - { last_token_type = token_type::value_unsigned; break; } + { + last_token_type = token_type::value_unsigned; + break; + } basic_json_parser_15: - yyaccept = 1; - m_marker = ++m_cursor; - if ((m_limit - m_cursor) < 3) fill_line_buffer(3); // LCOV_EXCL_LINE - yych = *m_cursor; - if (yybm[0+yych] & 64) { - goto basic_json_parser_15; - } - if (yych <= 'D') { - if (yych == '.') goto basic_json_parser_47; - goto basic_json_parser_14; - } else { - if (yych <= 'E') goto basic_json_parser_51; - if (yych == 'e') goto basic_json_parser_51; - goto basic_json_parser_14; - } + yyaccept = 1; + m_marker = ++m_cursor; + if ((m_limit - m_cursor) < 3) + { + fill_line_buffer(3); // LCOV_EXCL_LINE + } + yych = *m_cursor; + if (yybm[0 + yych] & 64) + { + goto basic_json_parser_15; + } + if (yych <= 'D') + { + if (yych == '.') + { + goto basic_json_parser_47; + } + goto basic_json_parser_14; + } + else + { + if (yych <= 'E') + { + goto basic_json_parser_51; + } + if (yych == 'e') + { + goto basic_json_parser_51; + } + goto basic_json_parser_14; + } basic_json_parser_17: - ++m_cursor; - { last_token_type = token_type::name_separator; break; } + ++m_cursor; + { + last_token_type = token_type::name_separator; + break; + } basic_json_parser_19: - ++m_cursor; - { last_token_type = token_type::begin_array; break; } + ++m_cursor; + { + last_token_type = token_type::begin_array; + break; + } basic_json_parser_21: - ++m_cursor; - { last_token_type = token_type::end_array; break; } + ++m_cursor; + { + last_token_type = token_type::end_array; + break; + } basic_json_parser_23: - yyaccept = 0; - yych = *(m_marker = ++m_cursor); - if (yych == 'a') goto basic_json_parser_52; - goto basic_json_parser_5; + yyaccept = 0; + yych = *(m_marker = ++m_cursor); + if (yych == 'a') + { + goto basic_json_parser_52; + } + goto basic_json_parser_5; basic_json_parser_24: - yyaccept = 0; - yych = *(m_marker = ++m_cursor); - if (yych == 'u') goto basic_json_parser_53; - goto basic_json_parser_5; + yyaccept = 0; + yych = *(m_marker = ++m_cursor); + if (yych == 'u') + { + goto basic_json_parser_53; + } + goto basic_json_parser_5; basic_json_parser_25: - yyaccept = 0; - yych = *(m_marker = ++m_cursor); - if (yych == 'r') goto basic_json_parser_54; - goto basic_json_parser_5; + yyaccept = 0; + yych = *(m_marker = ++m_cursor); + if (yych == 'r') + { + goto basic_json_parser_54; + } + goto basic_json_parser_5; basic_json_parser_26: - ++m_cursor; - { last_token_type = token_type::begin_object; break; } + ++m_cursor; + { + last_token_type = token_type::begin_object; + break; + } basic_json_parser_28: - ++m_cursor; - { last_token_type = token_type::end_object; break; } + ++m_cursor; + { + last_token_type = token_type::end_object; + break; + } basic_json_parser_30: - ++m_cursor; - if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE - yych = *m_cursor; + ++m_cursor; + if (m_limit <= m_cursor) + { + fill_line_buffer(1); // LCOV_EXCL_LINE + } + yych = *m_cursor; basic_json_parser_31: - if (yybm[0+yych] & 128) { - goto basic_json_parser_30; - } - if (yych <= 0xE0) { - if (yych <= '\\') { - if (yych <= 0x1F) goto basic_json_parser_32; - if (yych <= '"') goto basic_json_parser_33; - goto basic_json_parser_35; - } else { - if (yych <= 0xC1) goto basic_json_parser_32; - if (yych <= 0xDF) goto basic_json_parser_36; - goto basic_json_parser_37; - } - } else { - if (yych <= 0xEF) { - if (yych == 0xED) goto basic_json_parser_39; - goto basic_json_parser_38; - } else { - if (yych <= 0xF0) goto basic_json_parser_40; - if (yych <= 0xF3) goto basic_json_parser_41; - if (yych <= 0xF4) goto basic_json_parser_42; - } - } + if (yybm[0 + yych] & 128) + { + goto basic_json_parser_30; + } + if (yych <= 0xE0) + { + if (yych <= '\\') + { + if (yych <= 0x1F) + { + goto basic_json_parser_32; + } + if (yych <= '"') + { + goto basic_json_parser_33; + } + goto basic_json_parser_35; + } + else + { + if (yych <= 0xC1) + { + goto basic_json_parser_32; + } + if (yych <= 0xDF) + { + goto basic_json_parser_36; + } + goto basic_json_parser_37; + } + } + else + { + if (yych <= 0xEF) + { + if (yych == 0xED) + { + goto basic_json_parser_39; + } + goto basic_json_parser_38; + } + else + { + if (yych <= 0xF0) + { + goto basic_json_parser_40; + } + if (yych <= 0xF3) + { + goto basic_json_parser_41; + } + if (yych <= 0xF4) + { + goto basic_json_parser_42; + } + } + } basic_json_parser_32: - m_cursor = m_marker; - if (yyaccept <= 1) { - if (yyaccept == 0) { - goto basic_json_parser_5; - } else { - goto basic_json_parser_14; - } - } else { - if (yyaccept == 2) { - goto basic_json_parser_44; - } else { - goto basic_json_parser_58; - } - } + m_cursor = m_marker; + if (yyaccept <= 1) + { + if (yyaccept == 0) + { + goto basic_json_parser_5; + } + else + { + goto basic_json_parser_14; + } + } + else + { + if (yyaccept == 2) + { + goto basic_json_parser_44; + } + else + { + goto basic_json_parser_58; + } + } basic_json_parser_33: - ++m_cursor; - { last_token_type = token_type::value_string; break; } + ++m_cursor; + { + last_token_type = token_type::value_string; + break; + } basic_json_parser_35: - ++m_cursor; - if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE - yych = *m_cursor; - if (yych <= 'e') { - if (yych <= '/') { - if (yych == '"') goto basic_json_parser_30; - if (yych <= '.') goto basic_json_parser_32; - goto basic_json_parser_30; - } else { - if (yych <= '\\') { - if (yych <= '[') goto basic_json_parser_32; - goto basic_json_parser_30; - } else { - if (yych == 'b') goto basic_json_parser_30; - goto basic_json_parser_32; - } - } - } else { - if (yych <= 'q') { - if (yych <= 'f') goto basic_json_parser_30; - if (yych == 'n') goto basic_json_parser_30; - goto basic_json_parser_32; - } else { - if (yych <= 's') { - if (yych <= 'r') goto basic_json_parser_30; - goto basic_json_parser_32; - } else { - if (yych <= 't') goto basic_json_parser_30; - if (yych <= 'u') goto basic_json_parser_55; - goto basic_json_parser_32; - } - } - } + ++m_cursor; + if (m_limit <= m_cursor) + { + fill_line_buffer(1); // LCOV_EXCL_LINE + } + yych = *m_cursor; + if (yych <= 'e') + { + if (yych <= '/') + { + if (yych == '"') + { + goto basic_json_parser_30; + } + if (yych <= '.') + { + goto basic_json_parser_32; + } + goto basic_json_parser_30; + } + else + { + if (yych <= '\\') + { + if (yych <= '[') + { + goto basic_json_parser_32; + } + goto basic_json_parser_30; + } + else + { + if (yych == 'b') + { + goto basic_json_parser_30; + } + goto basic_json_parser_32; + } + } + } + else + { + if (yych <= 'q') + { + if (yych <= 'f') + { + goto basic_json_parser_30; + } + if (yych == 'n') + { + goto basic_json_parser_30; + } + goto basic_json_parser_32; + } + else + { + if (yych <= 's') + { + if (yych <= 'r') + { + goto basic_json_parser_30; + } + goto basic_json_parser_32; + } + else + { + if (yych <= 't') + { + goto basic_json_parser_30; + } + if (yych <= 'u') + { + goto basic_json_parser_55; + } + goto basic_json_parser_32; + } + } + } basic_json_parser_36: - ++m_cursor; - if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE - yych = *m_cursor; - if (yych <= 0x7F) goto basic_json_parser_32; - if (yych <= 0xBF) goto basic_json_parser_30; - goto basic_json_parser_32; + ++m_cursor; + if (m_limit <= m_cursor) + { + fill_line_buffer(1); // LCOV_EXCL_LINE + } + yych = *m_cursor; + if (yych <= 0x7F) + { + goto basic_json_parser_32; + } + if (yych <= 0xBF) + { + goto basic_json_parser_30; + } + goto basic_json_parser_32; basic_json_parser_37: - ++m_cursor; - if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE - yych = *m_cursor; - if (yych <= 0x9F) goto basic_json_parser_32; - if (yych <= 0xBF) goto basic_json_parser_36; - goto basic_json_parser_32; + ++m_cursor; + if (m_limit <= m_cursor) + { + fill_line_buffer(1); // LCOV_EXCL_LINE + } + yych = *m_cursor; + if (yych <= 0x9F) + { + goto basic_json_parser_32; + } + if (yych <= 0xBF) + { + goto basic_json_parser_36; + } + goto basic_json_parser_32; basic_json_parser_38: - ++m_cursor; - if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE - yych = *m_cursor; - if (yych <= 0x7F) goto basic_json_parser_32; - if (yych <= 0xBF) goto basic_json_parser_36; - goto basic_json_parser_32; + ++m_cursor; + if (m_limit <= m_cursor) + { + fill_line_buffer(1); // LCOV_EXCL_LINE + } + yych = *m_cursor; + if (yych <= 0x7F) + { + goto basic_json_parser_32; + } + if (yych <= 0xBF) + { + goto basic_json_parser_36; + } + goto basic_json_parser_32; basic_json_parser_39: - ++m_cursor; - if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE - yych = *m_cursor; - if (yych <= 0x7F) goto basic_json_parser_32; - if (yych <= 0x9F) goto basic_json_parser_36; - goto basic_json_parser_32; + ++m_cursor; + if (m_limit <= m_cursor) + { + fill_line_buffer(1); // LCOV_EXCL_LINE + } + yych = *m_cursor; + if (yych <= 0x7F) + { + goto basic_json_parser_32; + } + if (yych <= 0x9F) + { + goto basic_json_parser_36; + } + goto basic_json_parser_32; basic_json_parser_40: - ++m_cursor; - if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE - yych = *m_cursor; - if (yych <= 0x8F) goto basic_json_parser_32; - if (yych <= 0xBF) goto basic_json_parser_38; - goto basic_json_parser_32; + ++m_cursor; + if (m_limit <= m_cursor) + { + fill_line_buffer(1); // LCOV_EXCL_LINE + } + yych = *m_cursor; + if (yych <= 0x8F) + { + goto basic_json_parser_32; + } + if (yych <= 0xBF) + { + goto basic_json_parser_38; + } + goto basic_json_parser_32; basic_json_parser_41: - ++m_cursor; - if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE - yych = *m_cursor; - if (yych <= 0x7F) goto basic_json_parser_32; - if (yych <= 0xBF) goto basic_json_parser_38; - goto basic_json_parser_32; + ++m_cursor; + if (m_limit <= m_cursor) + { + fill_line_buffer(1); // LCOV_EXCL_LINE + } + yych = *m_cursor; + if (yych <= 0x7F) + { + goto basic_json_parser_32; + } + if (yych <= 0xBF) + { + goto basic_json_parser_38; + } + goto basic_json_parser_32; basic_json_parser_42: - ++m_cursor; - if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE - yych = *m_cursor; - if (yych <= 0x7F) goto basic_json_parser_32; - if (yych <= 0x8F) goto basic_json_parser_38; - goto basic_json_parser_32; + ++m_cursor; + if (m_limit <= m_cursor) + { + fill_line_buffer(1); // LCOV_EXCL_LINE + } + yych = *m_cursor; + if (yych <= 0x7F) + { + goto basic_json_parser_32; + } + if (yych <= 0x8F) + { + goto basic_json_parser_38; + } + goto basic_json_parser_32; basic_json_parser_43: - yyaccept = 2; - yych = *(m_marker = ++m_cursor); - if (yych <= '9') { - if (yych == '.') goto basic_json_parser_47; - if (yych >= '0') goto basic_json_parser_48; - } else { - if (yych <= 'E') { - if (yych >= 'E') goto basic_json_parser_51; - } else { - if (yych == 'e') goto basic_json_parser_51; - } - } + yyaccept = 2; + yych = *(m_marker = ++m_cursor); + if (yych <= '9') + { + if (yych == '.') + { + goto basic_json_parser_47; + } + if (yych >= '0') + { + goto basic_json_parser_48; + } + } + else + { + if (yych <= 'E') + { + if (yych >= 'E') + { + goto basic_json_parser_51; + } + } + else + { + if (yych == 'e') + { + goto basic_json_parser_51; + } + } + } basic_json_parser_44: - { last_token_type = token_type::value_integer; break; } + { + last_token_type = token_type::value_integer; + break; + } basic_json_parser_45: - yyaccept = 2; - m_marker = ++m_cursor; - if ((m_limit - m_cursor) < 3) fill_line_buffer(3); // LCOV_EXCL_LINE - yych = *m_cursor; - if (yych <= '9') { - if (yych == '.') goto basic_json_parser_47; - if (yych <= '/') goto basic_json_parser_44; - goto basic_json_parser_45; - } else { - if (yych <= 'E') { - if (yych <= 'D') goto basic_json_parser_44; - goto basic_json_parser_51; - } else { - if (yych == 'e') goto basic_json_parser_51; - goto basic_json_parser_44; - } - } + yyaccept = 2; + m_marker = ++m_cursor; + if ((m_limit - m_cursor) < 3) + { + fill_line_buffer(3); // LCOV_EXCL_LINE + } + yych = *m_cursor; + if (yych <= '9') + { + if (yych == '.') + { + goto basic_json_parser_47; + } + if (yych <= '/') + { + goto basic_json_parser_44; + } + goto basic_json_parser_45; + } + else + { + if (yych <= 'E') + { + if (yych <= 'D') + { + goto basic_json_parser_44; + } + goto basic_json_parser_51; + } + else + { + if (yych == 'e') + { + goto basic_json_parser_51; + } + goto basic_json_parser_44; + } + } basic_json_parser_47: - yych = *++m_cursor; - if (yych <= '/') goto basic_json_parser_32; - if (yych <= '9') goto basic_json_parser_56; - goto basic_json_parser_32; + yych = *++m_cursor; + if (yych <= '/') + { + goto basic_json_parser_32; + } + if (yych <= '9') + { + goto basic_json_parser_56; + } + goto basic_json_parser_32; basic_json_parser_48: - ++m_cursor; - if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE - yych = *m_cursor; - if (yych <= '/') goto basic_json_parser_50; - if (yych <= '9') goto basic_json_parser_48; + ++m_cursor; + if (m_limit <= m_cursor) + { + fill_line_buffer(1); // LCOV_EXCL_LINE + } + yych = *m_cursor; + if (yych <= '/') + { + goto basic_json_parser_50; + } + if (yych <= '9') + { + goto basic_json_parser_48; + } basic_json_parser_50: - { last_token_type = token_type::parse_error; break; } + { + last_token_type = token_type::parse_error; + break; + } basic_json_parser_51: - yych = *++m_cursor; - if (yych <= ',') { - if (yych == '+') goto basic_json_parser_59; - goto basic_json_parser_32; - } else { - if (yych <= '-') goto basic_json_parser_59; - if (yych <= '/') goto basic_json_parser_32; - if (yych <= '9') goto basic_json_parser_60; - goto basic_json_parser_32; - } + yych = *++m_cursor; + if (yych <= ',') + { + if (yych == '+') + { + goto basic_json_parser_59; + } + goto basic_json_parser_32; + } + else + { + if (yych <= '-') + { + goto basic_json_parser_59; + } + if (yych <= '/') + { + goto basic_json_parser_32; + } + if (yych <= '9') + { + goto basic_json_parser_60; + } + goto basic_json_parser_32; + } basic_json_parser_52: - yych = *++m_cursor; - if (yych == 'l') goto basic_json_parser_62; - goto basic_json_parser_32; + yych = *++m_cursor; + if (yych == 'l') + { + goto basic_json_parser_62; + } + goto basic_json_parser_32; basic_json_parser_53: - yych = *++m_cursor; - if (yych == 'l') goto basic_json_parser_63; - goto basic_json_parser_32; + yych = *++m_cursor; + if (yych == 'l') + { + goto basic_json_parser_63; + } + goto basic_json_parser_32; basic_json_parser_54: - yych = *++m_cursor; - if (yych == 'u') goto basic_json_parser_64; - goto basic_json_parser_32; + yych = *++m_cursor; + if (yych == 'u') + { + goto basic_json_parser_64; + } + goto basic_json_parser_32; basic_json_parser_55: - ++m_cursor; - if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE - yych = *m_cursor; - if (yych <= '@') { - if (yych <= '/') goto basic_json_parser_32; - if (yych <= '9') goto basic_json_parser_65; - goto basic_json_parser_32; - } else { - if (yych <= 'F') goto basic_json_parser_65; - if (yych <= '`') goto basic_json_parser_32; - if (yych <= 'f') goto basic_json_parser_65; - goto basic_json_parser_32; - } + ++m_cursor; + if (m_limit <= m_cursor) + { + fill_line_buffer(1); // LCOV_EXCL_LINE + } + yych = *m_cursor; + if (yych <= '@') + { + if (yych <= '/') + { + goto basic_json_parser_32; + } + if (yych <= '9') + { + goto basic_json_parser_65; + } + goto basic_json_parser_32; + } + else + { + if (yych <= 'F') + { + goto basic_json_parser_65; + } + if (yych <= '`') + { + goto basic_json_parser_32; + } + if (yych <= 'f') + { + goto basic_json_parser_65; + } + goto basic_json_parser_32; + } basic_json_parser_56: - yyaccept = 3; - m_marker = ++m_cursor; - if ((m_limit - m_cursor) < 3) fill_line_buffer(3); // LCOV_EXCL_LINE - yych = *m_cursor; - if (yych <= 'D') { - if (yych <= '/') goto basic_json_parser_58; - if (yych <= '9') goto basic_json_parser_56; - } else { - if (yych <= 'E') goto basic_json_parser_51; - if (yych == 'e') goto basic_json_parser_51; - } + yyaccept = 3; + m_marker = ++m_cursor; + if ((m_limit - m_cursor) < 3) + { + fill_line_buffer(3); // LCOV_EXCL_LINE + } + yych = *m_cursor; + if (yych <= 'D') + { + if (yych <= '/') + { + goto basic_json_parser_58; + } + if (yych <= '9') + { + goto basic_json_parser_56; + } + } + else + { + if (yych <= 'E') + { + goto basic_json_parser_51; + } + if (yych == 'e') + { + goto basic_json_parser_51; + } + } basic_json_parser_58: - { last_token_type = token_type::value_float; break; } + { + last_token_type = token_type::value_float; + break; + } basic_json_parser_59: - yych = *++m_cursor; - if (yych <= '/') goto basic_json_parser_32; - if (yych >= ':') goto basic_json_parser_32; + yych = *++m_cursor; + if (yych <= '/') + { + goto basic_json_parser_32; + } + if (yych >= ':') + { + goto basic_json_parser_32; + } basic_json_parser_60: - ++m_cursor; - if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE - yych = *m_cursor; - if (yych <= '/') goto basic_json_parser_58; - if (yych <= '9') goto basic_json_parser_60; - goto basic_json_parser_58; + ++m_cursor; + if (m_limit <= m_cursor) + { + fill_line_buffer(1); // LCOV_EXCL_LINE + } + yych = *m_cursor; + if (yych <= '/') + { + goto basic_json_parser_58; + } + if (yych <= '9') + { + goto basic_json_parser_60; + } + goto basic_json_parser_58; basic_json_parser_62: - yych = *++m_cursor; - if (yych == 's') goto basic_json_parser_66; - goto basic_json_parser_32; + yych = *++m_cursor; + if (yych == 's') + { + goto basic_json_parser_66; + } + goto basic_json_parser_32; basic_json_parser_63: - yych = *++m_cursor; - if (yych == 'l') goto basic_json_parser_67; - goto basic_json_parser_32; + yych = *++m_cursor; + if (yych == 'l') + { + goto basic_json_parser_67; + } + goto basic_json_parser_32; basic_json_parser_64: - yych = *++m_cursor; - if (yych == 'e') goto basic_json_parser_69; - goto basic_json_parser_32; + yych = *++m_cursor; + if (yych == 'e') + { + goto basic_json_parser_69; + } + goto basic_json_parser_32; basic_json_parser_65: - ++m_cursor; - if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE - yych = *m_cursor; - if (yych <= '@') { - if (yych <= '/') goto basic_json_parser_32; - if (yych <= '9') goto basic_json_parser_71; - goto basic_json_parser_32; - } else { - if (yych <= 'F') goto basic_json_parser_71; - if (yych <= '`') goto basic_json_parser_32; - if (yych <= 'f') goto basic_json_parser_71; - goto basic_json_parser_32; - } + ++m_cursor; + if (m_limit <= m_cursor) + { + fill_line_buffer(1); // LCOV_EXCL_LINE + } + yych = *m_cursor; + if (yych <= '@') + { + if (yych <= '/') + { + goto basic_json_parser_32; + } + if (yych <= '9') + { + goto basic_json_parser_71; + } + goto basic_json_parser_32; + } + else + { + if (yych <= 'F') + { + goto basic_json_parser_71; + } + if (yych <= '`') + { + goto basic_json_parser_32; + } + if (yych <= 'f') + { + goto basic_json_parser_71; + } + goto basic_json_parser_32; + } basic_json_parser_66: - yych = *++m_cursor; - if (yych == 'e') goto basic_json_parser_72; - goto basic_json_parser_32; + yych = *++m_cursor; + if (yych == 'e') + { + goto basic_json_parser_72; + } + goto basic_json_parser_32; basic_json_parser_67: - ++m_cursor; - { last_token_type = token_type::literal_null; break; } + ++m_cursor; + { + last_token_type = token_type::literal_null; + break; + } basic_json_parser_69: - ++m_cursor; - { last_token_type = token_type::literal_true; break; } + ++m_cursor; + { + last_token_type = token_type::literal_true; + break; + } basic_json_parser_71: - ++m_cursor; - if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE - yych = *m_cursor; - if (yych <= '@') { - if (yych <= '/') goto basic_json_parser_32; - if (yych <= '9') goto basic_json_parser_74; - goto basic_json_parser_32; - } else { - if (yych <= 'F') goto basic_json_parser_74; - if (yych <= '`') goto basic_json_parser_32; - if (yych <= 'f') goto basic_json_parser_74; - goto basic_json_parser_32; - } + ++m_cursor; + if (m_limit <= m_cursor) + { + fill_line_buffer(1); // LCOV_EXCL_LINE + } + yych = *m_cursor; + if (yych <= '@') + { + if (yych <= '/') + { + goto basic_json_parser_32; + } + if (yych <= '9') + { + goto basic_json_parser_74; + } + goto basic_json_parser_32; + } + else + { + if (yych <= 'F') + { + goto basic_json_parser_74; + } + if (yych <= '`') + { + goto basic_json_parser_32; + } + if (yych <= 'f') + { + goto basic_json_parser_74; + } + goto basic_json_parser_32; + } basic_json_parser_72: - ++m_cursor; - { last_token_type = token_type::literal_false; break; } + ++m_cursor; + { + last_token_type = token_type::literal_false; + break; + } basic_json_parser_74: - ++m_cursor; - if (m_limit <= m_cursor) fill_line_buffer(1); // LCOV_EXCL_LINE - yych = *m_cursor; - if (yych <= '@') { - if (yych <= '/') goto basic_json_parser_32; - if (yych <= '9') goto basic_json_parser_30; - goto basic_json_parser_32; - } else { - if (yych <= 'F') goto basic_json_parser_30; - if (yych <= '`') goto basic_json_parser_32; - if (yych <= 'f') goto basic_json_parser_30; - goto basic_json_parser_32; - } - } + ++m_cursor; + if (m_limit <= m_cursor) + { + fill_line_buffer(1); // LCOV_EXCL_LINE + } + yych = *m_cursor; + if (yych <= '@') + { + if (yych <= '/') + { + goto basic_json_parser_32; + } + if (yych <= '9') + { + goto basic_json_parser_30; + } + goto basic_json_parser_32; + } + else + { + if (yych <= 'F') + { + goto basic_json_parser_30; + } + if (yych <= '`') + { + goto basic_json_parser_32; + } + if (yych <= 'f') + { + goto basic_json_parser_30; + } + goto basic_json_parser_32; + } + } } diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 984431f5..a9a5067f 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -802,8 +802,9 @@ template ::value, int> = 0> -void to_json(BasicJsonType &j, T (&arr)[N]) { - external_constructor::construct(j, arr); +void to_json(BasicJsonType& j, T (&arr)[N]) +{ + external_constructor::construct(j, arr); } /////////////// diff --git a/test/src/unit-conversions.cpp b/test/src/unit-conversions.cpp index 829e9e2d..59969bf9 100644 --- a/test/src/unit-conversions.cpp +++ b/test/src/unit-conversions.cpp @@ -192,16 +192,16 @@ TEST_CASE("value conversion") SECTION("built-in arrays") { - const char str[] = "a string"; - const int nbs[] = {0, 1, 2}; + const char str[] = "a string"; + const int nbs[] = {0, 1, 2}; - json j2 = nbs; - json j3 = str; + json j2 = nbs; + json j3 = str; - auto v = j2.get>(); - auto s = j3.get(); - CHECK(std::equal(v.begin(), v.end(), std::begin(nbs))); - CHECK(s == str); + auto v = j2.get>(); + auto s = j3.get(); + CHECK(std::equal(v.begin(), v.end(), std::begin(nbs))); + CHECK(s == str); } SECTION("std::deque") From 836f1c418947ab5c512d5c913494e60a44c219c3 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Alexander=20=E2=80=9Cweej=E2=80=9D=20Jones?= Date: Thu, 16 Mar 2017 18:24:41 +0000 Subject: [PATCH 089/530] :art: Namespace `uint8_t` from the C++ stdlib --- README.md | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/README.md b/README.md index 2a896210..0a6c531f 100644 --- a/README.md +++ b/README.md @@ -230,17 +230,17 @@ Please note that setting the exception bit for `failbit` is inappropriate for th #### Read from iterator range -You can also read JSON from an iterator range; that is, from any container accessible by iterators whose content is stored as contiguous byte sequence, for instance a `std::vector`: +You can also read JSON from an iterator range; that is, from any container accessible by iterators whose content is stored as contiguous byte sequence, for instance a `std::vector`: ```cpp -std::vector v = {'t', 'r', 'u', 'e'}; +std::vector v = {'t', 'r', 'u', 'e'}; json j = json::parse(v.begin(), v.end()); ``` You may leave the iterators for the range [begin, end): ```cpp -std::vector v = {'t', 'r', 'u', 'e'}; +std::vector v = {'t', 'r', 'u', 'e'}; json j = json::parse(v); ``` @@ -682,7 +682,7 @@ Though JSON is a ubiquitous data format, it is not a very compact format suitabl json j = R"({"compact": true, "schema": 0})"_json; // serialize to CBOR -std::vector v_cbor = json::to_cbor(j); +std::vector v_cbor = json::to_cbor(j); // 0xa2, 0x67, 0x63, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0xf5, 0x66, 0x73, 0x63, 0x68, 0x65, 0x6d, 0x61, 0x00 @@ -690,7 +690,7 @@ std::vector v_cbor = json::to_cbor(j); json j_from_cbor = json::from_cbor(v_cbor); // serialize to MessagePack -std::vector v_msgpack = json::to_msgpack(j); +std::vector v_msgpack = json::to_msgpack(j); // 0x82, 0xa7, 0x63, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0xc3, 0xa6, 0x73, 0x63, 0x68, 0x65, 0x6d, 0x61, 0x00 From f547679de5ca07a5c7aef508c1b31dce6cff2173 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Fri, 17 Mar 2017 22:18:05 +0100 Subject: [PATCH 090/530] :ambulance: fix for #516 and #518 We should compare the binary serializations rather than the JSON values themselves. This fix was already done for CBOR and apparently forgotten for MessagePack. --- test/src/fuzzer-parse_msgpack.cpp | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/test/src/fuzzer-parse_msgpack.cpp b/test/src/fuzzer-parse_msgpack.cpp index 420283a4..667a68d6 100644 --- a/test/src/fuzzer-parse_msgpack.cpp +++ b/test/src/fuzzer-parse_msgpack.cpp @@ -41,8 +41,8 @@ extern "C" int LLVMFuzzerTestOneInput(const uint8_t* data, size_t size) // parse serialization json j2 = json::from_msgpack(vec2); - // deserializations must match - assert(j1 == j2); + // serializations must match + assert(json::to_msgpack(j2) == vec2); } catch (const json::parse_error&) { From 31a6c0910e5676130b0f5df11b743700b70fb8ec Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Fri, 17 Mar 2017 22:18:48 +0100 Subject: [PATCH 091/530] :ambulance: fix for #519 Added catch branch for out_of_range exception that can occur if input file contains a number overflow. --- test/src/fuzzer-parse_json.cpp | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/test/src/fuzzer-parse_json.cpp b/test/src/fuzzer-parse_json.cpp index ff585066..fdaf96a7 100644 --- a/test/src/fuzzer-parse_json.cpp +++ b/test/src/fuzzer-parse_json.cpp @@ -59,6 +59,10 @@ extern "C" int LLVMFuzzerTestOneInput(const uint8_t* data, size_t size) { // parse errors are ok, because input may be random bytes } + catch (const json::out_of_range&) + { + // parse errors are ok, because input may be random bytes + } // return 0 - non-zero return values are reserved for future use return 0; From 86579209c809dd332873c5ee60dfb71b93725c4c Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Krzysztof=20Wos=CC=81?= Date: Mon, 20 Mar 2017 12:42:44 +0900 Subject: [PATCH 092/530] Make exception base class visible in basic_json --- src/json.hpp | 2 ++ src/json.hpp.re2c | 2 ++ 2 files changed, 4 insertions(+) diff --git a/src/json.hpp b/src/json.hpp index 9e539790..20bd25f5 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -1264,6 +1264,8 @@ class basic_json /// Classes to implement user-defined exceptions. /// @{ + /// @copydoc detail::exception + using exception = detail::exception; /// @copydoc detail::parse_error using parse_error = detail::parse_error; /// @copydoc detail::invalid_iterator diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index a9a5067f..da5dd86d 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -1264,6 +1264,8 @@ class basic_json /// Classes to implement user-defined exceptions. /// @{ + /// @copydoc detail::exception + using exception = detail::exception; /// @copydoc detail::parse_error using parse_error = detail::parse_error; /// @copydoc detail::invalid_iterator From b9c3b8d688434919f9cb68f53eab8238f26e6c2c Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Krzysztof=20Wos=CC=81?= Date: Mon, 20 Mar 2017 13:07:01 +0900 Subject: [PATCH 093/530] Exit with an error if re2c is not available Instead of overwriting src/json.hpp with an empty file --- Makefile | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/Makefile b/Makefile index df05e62e..20857022 100644 --- a/Makefile +++ b/Makefile @@ -1,7 +1,7 @@ .PHONY: pretty clean ChangeLog.md # used programs -RE2C = re2c +RE2C := $(shell command -v re2c 2> /dev/null) SED = sed # main target @@ -185,6 +185,9 @@ clang_sanitize: clean # create scanner with re2c re2c: src/json.hpp.re2c +ifndef RE2C + $(error "re2c is not available, please install re2c") +endif $(RE2C) -W --utf-8 --encoding-policy fail --bit-vectors --nested-ifs --no-debug-info $< | $(SED) '1d' > src/json.hpp # pretty printer From d60851168e5b9450825b1effbfdc02856c568dbc Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Wed, 22 Mar 2017 08:10:02 +0100 Subject: [PATCH 094/530] :rotating_light: fixed a compiler warning #527 The result of snprintf is later used in situations where a long may overflow. --- src/json.hpp | 11 +++++------ src/json.hpp.re2c | 11 +++++------ 2 files changed, 10 insertions(+), 12 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 20bd25f5..1ccbfa61 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -6929,8 +6929,8 @@ class basic_json static constexpr auto d = std::numeric_limits::digits10; // the actual conversion - long len = snprintf(number_buffer.data(), number_buffer.size(), - "%.*g", d, x); + std::ptrdiff_t len = snprintf(number_buffer.data(), number_buffer.size(), + "%.*g", d, x); // negative value indicates an error assert(len > 0); @@ -10429,10 +10429,9 @@ class basic_json Proof (by contradiction): Assume a finite input. To loop forever, the loop must never hit code with a `break` statement. The only code - snippets without a `break` statement are the continue statements for - whitespace and byte-order-marks. To loop forever, the input must be an - infinite sequence of whitespace or byte-order-marks. This contradicts - the assumption of finite input, q.e.d. + snippets without a `break` statement is the continue statement for + whitespace. To loop forever, the input must be an infinite sequence + whitespace. This contradicts the assumption of finite input, q.e.d. */ token_type scan() { diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index da5dd86d..6a1e09e3 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -6929,8 +6929,8 @@ class basic_json static constexpr auto d = std::numeric_limits::digits10; // the actual conversion - long len = snprintf(number_buffer.data(), number_buffer.size(), - "%.*g", d, x); + std::ptrdiff_t len = snprintf(number_buffer.data(), number_buffer.size(), + "%.*g", d, x); // negative value indicates an error assert(len > 0); @@ -10429,10 +10429,9 @@ class basic_json Proof (by contradiction): Assume a finite input. To loop forever, the loop must never hit code with a `break` statement. The only code - snippets without a `break` statement are the continue statements for - whitespace and byte-order-marks. To loop forever, the input must be an - infinite sequence of whitespace or byte-order-marks. This contradicts - the assumption of finite input, q.e.d. + snippets without a `break` statement is the continue statement for + whitespace. To loop forever, the input must be an infinite sequence + whitespace. This contradicts the assumption of finite input, q.e.d. */ token_type scan() { From 54db53c23018a2ebc967e5c70d31a11c5ea88874 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Wed, 22 Mar 2017 08:10:31 +0100 Subject: [PATCH 095/530] :memo: update after #527 and #525 --- README.md | 2 ++ 1 file changed, 2 insertions(+) diff --git a/README.md b/README.md index 0a6c531f..4677db56 100644 --- a/README.md +++ b/README.md @@ -829,6 +829,8 @@ I deeply appreciate the help of the following people. - [Mihai Stan](https://github.com/stanmihai4) fixed a bug in the comparison with `nullptr`s. - [Tushar Maheshwari](https://github.com/tusharpm) added [cotire](https://github.com/sakra/cotire) support to speed up the compilation. - [TedLyngmo](https://github.com/TedLyngmo) noted a typo in the README and fixed some `-Weffc++` warnings. +- [Krzysztof Woś](https://github.com/krzysztofwos) made exceptions more visible. +- [ftillier](https://github.com/ftillier) fixed a compiler warning. Thanks a lot for helping out! Please [let me know](mailto:mail@nlohmann.me) if I forgot someone. From b14629b0a61bf8a5a8969d7d3d07d7ef900ff9f8 Mon Sep 17 00:00:00 2001 From: Berkus Decker Date: Thu, 23 Mar 2017 14:50:41 +0200 Subject: [PATCH 096/530] Fix typo --- README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.md b/README.md index 4677db56..4bf678ef 100644 --- a/README.md +++ b/README.md @@ -880,7 +880,7 @@ The library is currently used in Apple macOS Sierra and iOS 10. I am not sure wh - Invalid surrogates (e.g., incomplete pairs such as `\uDEAD`) will yield parse errors. - The strings stored in the library are UTF-8 encoded. When using the default string type (`std::string`), note that its length/size functions return the number of stored bytes rather than the number of characters or glyphs. - The code can be compiled without C++ **runtime type identification** features; that is, you can use the `-fno-rtti` compiler flag. -- **Exceptions** are used widly within the library. They can, however, be switched off with either using the compiler flag `-fno-exceptions` or by defining the symbol `JSON_NOEXCEPTION`. In this case, exceptions are replaced by an `abort()` call. +- **Exceptions** are used widely within the library. They can, however, be switched off with either using the compiler flag `-fno-exceptions` or by defining the symbol `JSON_NOEXCEPTION`. In this case, exceptions are replaced by an `abort()` call. - By default, the library does not preserve the **insertion order of object elements**. This is standards-compliant, as the [JSON standard](https://tools.ietf.org/html/rfc7159.html) defines objects as "an unordered collection of zero or more name/value pairs". If you do want to preserve the insertion order, you can specialize the object type with containers like [`tsl::ordered_map`](https://github.com/Tessil/ordered-map) or [`nlohmann::fifo_map`](https://github.com/nlohmann/fifo_map). From cea242673168a4435cbbd987acae4e1c8f627bac Mon Sep 17 00:00:00 2001 From: Ted Lyngmo Date: Fri, 24 Mar 2017 14:56:48 +0100 Subject: [PATCH 097/530] UTF8 encoding enhancement --- src/json.hpp | 6 +++--- src/json.hpp.re2c | 6 +++--- 2 files changed, 6 insertions(+), 6 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 1ccbfa61..6ec045aa 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -10344,20 +10344,20 @@ class basic_json else if (codepoint <= 0x7ff) { // 2-byte characters: 110xxxxx 10xxxxxx - result.append(1, static_cast(0xC0 | ((codepoint >> 6) & 0x1F))); + result.append(1, static_cast(0xC0 | (codepoint >> 6))); result.append(1, static_cast(0x80 | (codepoint & 0x3F))); } else if (codepoint <= 0xffff) { // 3-byte characters: 1110xxxx 10xxxxxx 10xxxxxx - result.append(1, static_cast(0xE0 | ((codepoint >> 12) & 0x0F))); + result.append(1, static_cast(0xE0 | (codepoint >> 12))); result.append(1, static_cast(0x80 | ((codepoint >> 6) & 0x3F))); result.append(1, static_cast(0x80 | (codepoint & 0x3F))); } else if (codepoint <= 0x10ffff) { // 4-byte characters: 11110xxx 10xxxxxx 10xxxxxx 10xxxxxx - result.append(1, static_cast(0xF0 | ((codepoint >> 18) & 0x07))); + result.append(1, static_cast(0xF0 | (codepoint >> 18))); result.append(1, static_cast(0x80 | ((codepoint >> 12) & 0x3F))); result.append(1, static_cast(0x80 | ((codepoint >> 6) & 0x3F))); result.append(1, static_cast(0x80 | (codepoint & 0x3F))); diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 6a1e09e3..7d21f76b 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -10344,20 +10344,20 @@ class basic_json else if (codepoint <= 0x7ff) { // 2-byte characters: 110xxxxx 10xxxxxx - result.append(1, static_cast(0xC0 | ((codepoint >> 6) & 0x1F))); + result.append(1, static_cast(0xC0 | (codepoint >> 6))); result.append(1, static_cast(0x80 | (codepoint & 0x3F))); } else if (codepoint <= 0xffff) { // 3-byte characters: 1110xxxx 10xxxxxx 10xxxxxx - result.append(1, static_cast(0xE0 | ((codepoint >> 12) & 0x0F))); + result.append(1, static_cast(0xE0 | (codepoint >> 12))); result.append(1, static_cast(0x80 | ((codepoint >> 6) & 0x3F))); result.append(1, static_cast(0x80 | (codepoint & 0x3F))); } else if (codepoint <= 0x10ffff) { // 4-byte characters: 11110xxx 10xxxxxx 10xxxxxx 10xxxxxx - result.append(1, static_cast(0xF0 | ((codepoint >> 18) & 0x07))); + result.append(1, static_cast(0xF0 | (codepoint >> 18))); result.append(1, static_cast(0x80 | ((codepoint >> 12) & 0x3F))); result.append(1, static_cast(0x80 | ((codepoint >> 6) & 0x3F))); result.append(1, static_cast(0x80 | (codepoint & 0x3F))); From 40160f482a2cbd96b1f511397d14905e108a9ca1 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Fri, 24 Mar 2017 19:49:02 +0100 Subject: [PATCH 098/530] :construction: manual lexer This commit removed the re2c lexer and replaced it by a manual version. Its integration is not yet complete: number parsing does not respect locales or overflows. Furthermore, parsing does not need to end with EOF. Therefore, a lot of test cases fail. The idea is to push this branch forward so we can conduct performance comparisons. So far, a nice side effect are better diagnosis messages in case of parse errors. --- Makefile | 13 +- errors.txt | 7 + src/json.hpp | 2409 ++---- src/json.hpp.re2c | 12863 ---------------------------- test/src/unit-class_lexer.cpp | 94 +- test/src/unit-class_parser.cpp | 146 +- test/src/unit-deserialization.cpp | 10 +- test/src/unit-regression.cpp | 4 +- test/src/unit-unicode.cpp | 4 +- 9 files changed, 851 insertions(+), 14699 deletions(-) create mode 100644 errors.txt delete mode 100644 src/json.hpp.re2c diff --git a/Makefile b/Makefile index 20857022..38d40eda 100644 --- a/Makefile +++ b/Makefile @@ -1,9 +1,5 @@ .PHONY: pretty clean ChangeLog.md -# used programs -RE2C := $(shell command -v re2c 2> /dev/null) -SED = sed - # main target all: $(MAKE) -C test @@ -183,13 +179,6 @@ clang_sanitize: clean # maintainer targets ########################################################################## -# create scanner with re2c -re2c: src/json.hpp.re2c -ifndef RE2C - $(error "re2c is not available, please install re2c") -endif - $(RE2C) -W --utf-8 --encoding-policy fail --bit-vectors --nested-ifs --no-debug-info $< | $(SED) '1d' > src/json.hpp - # pretty printer pretty: astyle --style=allman --indent=spaces=4 --indent-modifiers \ @@ -197,7 +186,7 @@ pretty: --indent-col1-comments --pad-oper --pad-header --align-pointer=type \ --align-reference=type --add-brackets --convert-tabs --close-templates \ --lineend=linux --preserve-date --suffix=none --formatted \ - src/json.hpp src/json.hpp.re2c test/src/*.cpp \ + src/json.hpp test/src/*.cpp \ benchmarks/benchmarks.cpp doc/examples/*.cpp diff --git a/errors.txt b/errors.txt new file mode 100644 index 00000000..d3a6c6db --- /dev/null +++ b/errors.txt @@ -0,0 +1,7 @@ +- test/test-class_parser + - 617 failed +- test/test-regression + - 11 failed +- test/test-testsuites + - 43 failed + diff --git a/src/json.hpp b/src/json.hpp index 1ccbfa61..b86bee11 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -89,6 +89,10 @@ SOFTWARE. #define JSON_CATCH(exception) if(false) #endif +// manual branch prediction +#define JSON_LIKELY(x) __builtin_expect(!!(x), 1) +#define JSON_UNLIKELY(x) __builtin_expect(!!(x), 0) + /*! @brief namespace for Niels Lohmann @see https://github.com/nlohmann @@ -1968,7 +1972,7 @@ class basic_json default: { - if (t == value_t::null) + if (JSON_UNLIKELY(t == value_t::null)) { JSON_THROW(other_error(500, "961c151d2e87f2686a955a9be24d316f1362bf21 2.1.1")); // LCOV_EXCL_LINE } @@ -10213,12 +10217,195 @@ class basic_json /*! @brief lexical analysis - This class organizes the lexical analysis during JSON deserialization. The - core of it is a scanner generated by [re2c](http://re2c.org) that - processes a buffer and recognizes tokens according to RFC 7159. + This class organizes the lexical analysis during JSON deserialization. */ class lexer { + private: + + /// abstract input adapter interface + class input_adapter + { + public: + virtual int get_character() = 0; + virtual std::string read(size_t offset, size_t length) = 0; + virtual ~input_adapter() {} + }; + + /// input adapter for cached stream input + class cached_input_stream_adapter : public input_adapter + { + public: + cached_input_stream_adapter(std::istream& i) + : is(i), start_position(is.tellg()), + buffer(1024 * 1024, std::char_traits::eof()) + { + // immediately abort if stream is erroneous + if (JSON_UNLIKELY(i.fail())) + { + JSON_THROW(parse_error(111, 0, "bad input stream")); + } + + // initial fill; unfilled buffer charaters remain EOF + is.read(buffer.data(), static_cast(buffer.size())); + + // ignore byte-order mark + if (buffer[0] == '\xEF' and buffer[1] == '\xBB' and buffer[2] == '\xBF') + { + buffer_pos += 3; + processed_chars += 3; + } + } + + ~cached_input_stream_adapter() override + { + // clear stream flags + is.clear(); + // set stream after last processed char + is.seekg(start_position + static_cast(processed_chars - 1)); + } + + int get_character() override + { + // check if refilling is neccessary + if (JSON_UNLIKELY(buffer_pos == buffer.size())) + { + // refill + is.read(reinterpret_cast(buffer.data()), static_cast(buffer.size())); + // set unfilled characters to EOF + std::fill_n(buffer.begin() + is.gcount(), + buffer.size() - static_cast(is.gcount()), + std::char_traits::eof()); + // the buffer is ready + buffer_pos = 0; + } + + ++processed_chars; + return buffer[buffer_pos++]; + } + + std::string read(size_t offset, size_t length) override + { + // create buffer + std::string result(length, '\0'); + + // save stream position + auto current_pos = is.tellg(); + // save stream flags + auto flags = is.rdstate(); + + // clear stream flags + is.clear(); + // set stream position + is.seekg(static_cast(offset)); + // read bytes + is.read(&result[0], static_cast(length)); + + // reset stream position + is.seekg(current_pos); + // reset stream flags + is.setstate(flags); + + return result; + } + + private: + std::istream& is; + + // chars returned via get_character() + size_t processed_chars = 0; + // chars processed in the current buffer + size_t buffer_pos = 0; + + // position of the stream when we started + const std::streampos start_position; + + // internal buffer + std::vector buffer; + }; + + /// input adapter for uncached stream input + class input_stream_adapter : public input_adapter + { + public: + input_stream_adapter(std::istream& i) + : is(i) + { + // immediately abort if stream is erroneous + if (i.fail()) + { + JSON_THROW(parse_error(111, 0, "bad input stream")); + } + } + + int get_character() override + { + return is.get(); + } + + std::string read(size_t offset, size_t length) override + { + // create buffer + std::string result(length, '\0'); + + // save stream position + auto current_pos = is.tellg(); + // save stream flags + auto flags = is.rdstate(); + + // clear stream flags + is.clear(); + // set stream position + is.seekg(offset); + // read bytes + is.read(&result[0], length); + + // reset stream position + is.seekg(current_pos); + // reset stream flags + is.setstate(flags); + + return result; + } + + private: + std::istream& is; + }; + + /// input adapter for buffer input + class input_buffer_adapter : public input_adapter + { + public: + input_buffer_adapter(const char* b, size_t l) + : input_adapter(), cursor(b), limit(b + l), start(b) + {} + + input_buffer_adapter(const input_buffer_adapter&) = delete; + input_buffer_adapter& operator=(input_buffer_adapter&) = delete; + + int get_character() override + { + if (JSON_LIKELY(cursor < limit)) + { + return *cursor++; + } + else + { + return std::char_traits::eof(); + } + } + + std::string read(size_t offset, size_t length) override + { + return std::string(start + offset, length); + } + + private: + const char* cursor; + const char* limit; + const char* start; + }; + public: /// token types for the parser enum class token_type @@ -10241,135 +10428,6 @@ class basic_json end_of_input ///< indicating the end of the input buffer }; - /// the char type to use in the lexer - using lexer_char_t = unsigned char; - - /// a lexer from a buffer with given length - lexer(const lexer_char_t* buff, const size_t len) noexcept - : m_content(buff) - { - assert(m_content != nullptr); - m_start = m_cursor = m_content; - m_limit = m_content + len; - } - - /*! - @brief a lexer from an input stream - @throw parse_error.111 if input stream is in a bad state - */ - explicit lexer(std::istream& s) - : m_stream(&s), m_line_buffer() - { - // immediately abort if stream is erroneous - if (s.fail()) - { - JSON_THROW(parse_error(111, 0, "bad input stream")); - } - - // fill buffer - fill_line_buffer(); - - // skip UTF-8 byte-order mark - if (m_line_buffer.size() >= 3 and m_line_buffer.substr(0, 3) == "\xEF\xBB\xBF") - { - m_line_buffer[0] = ' '; - m_line_buffer[1] = ' '; - m_line_buffer[2] = ' '; - } - } - - // switch off unwanted functions (due to pointer members) - lexer() = delete; - lexer(const lexer&) = delete; - lexer operator=(const lexer&) = delete; - - /*! - @brief create a string from one or two Unicode code points - - There are two cases: (1) @a codepoint1 is in the Basic Multilingual - Plane (U+0000 through U+FFFF) and @a codepoint2 is 0, or (2) - @a codepoint1 and @a codepoint2 are a UTF-16 surrogate pair to - represent a code point above U+FFFF. - - @param[in] codepoint1 the code point (can be high surrogate) - @param[in] codepoint2 the code point (can be low surrogate or 0) - - @return string representation of the code point; the length of the - result string is between 1 and 4 characters. - - @throw parse_error.102 if the low surrogate is invalid; example: - `""missing or wrong low surrogate""` - @throw parse_error.103 if code point is > 0x10ffff; example: `"code - points above 0x10FFFF are invalid"` - - @complexity Constant. - - @see - */ - string_t to_unicode(const std::size_t codepoint1, - const std::size_t codepoint2 = 0) const - { - // calculate the code point from the given code points - std::size_t codepoint = codepoint1; - - // check if codepoint1 is a high surrogate - if (codepoint1 >= 0xD800 and codepoint1 <= 0xDBFF) - { - // check if codepoint2 is a low surrogate - if (codepoint2 >= 0xDC00 and codepoint2 <= 0xDFFF) - { - codepoint = - // high surrogate occupies the most significant 22 bits - (codepoint1 << 10) - // low surrogate occupies the least significant 15 bits - + codepoint2 - // there is still the 0xD800, 0xDC00 and 0x10000 noise - // in the result so we have to subtract with: - // (0xD800 << 10) + DC00 - 0x10000 = 0x35FDC00 - - 0x35FDC00; - } - else - { - JSON_THROW(parse_error(102, get_position(), "missing or wrong low surrogate")); - } - } - - string_t result; - - if (codepoint < 0x80) - { - // 1-byte characters: 0xxxxxxx (ASCII) - result.append(1, static_cast(codepoint)); - } - else if (codepoint <= 0x7ff) - { - // 2-byte characters: 110xxxxx 10xxxxxx - result.append(1, static_cast(0xC0 | ((codepoint >> 6) & 0x1F))); - result.append(1, static_cast(0x80 | (codepoint & 0x3F))); - } - else if (codepoint <= 0xffff) - { - // 3-byte characters: 1110xxxx 10xxxxxx 10xxxxxx - result.append(1, static_cast(0xE0 | ((codepoint >> 12) & 0x0F))); - result.append(1, static_cast(0x80 | ((codepoint >> 6) & 0x3F))); - result.append(1, static_cast(0x80 | (codepoint & 0x3F))); - } - else if (codepoint <= 0x10ffff) - { - // 4-byte characters: 11110xxx 10xxxxxx 10xxxxxx 10xxxxxx - result.append(1, static_cast(0xF0 | ((codepoint >> 18) & 0x07))); - result.append(1, static_cast(0x80 | ((codepoint >> 12) & 0x3F))); - result.append(1, static_cast(0x80 | ((codepoint >> 6) & 0x3F))); - result.append(1, static_cast(0x80 | (codepoint & 0x3F))); - } - else - { - JSON_THROW(parse_error(103, get_position(), "code points above 0x10FFFF are invalid")); - } - - return result; - } - /// return name of values of type token_type (only used for errors) static std::string token_type_name(const token_type t) { @@ -10413,1615 +10471,584 @@ class basic_json } } - /*! - This function implements a scanner for JSON. It is specified using - regular expressions that try to follow RFC 7159 as close as possible. - These regular expressions are then translated into a minimized - deterministic finite automaton (DFA) by the tool - [re2c](http://re2c.org). As a result, the translated code for this - function consists of a large block of code with `goto` jumps. + explicit lexer(std::istream& i) + // : ia(new input_stream_adapter(i)) + : ia(new cached_input_stream_adapter(i)) + {} - @return the class of the next token read from the buffer + lexer(const char* buff, const size_t len) + : ia(new input_buffer_adapter(buff, len)) + {} - @complexity Linear in the length of the input.\n - - Proposition: The loop below will always terminate for finite input.\n - - Proof (by contradiction): Assume a finite input. To loop forever, the - loop must never hit code with a `break` statement. The only code - snippets without a `break` statement is the continue statement for - whitespace. To loop forever, the input must be an infinite sequence - whitespace. This contradicts the assumption of finite input, q.e.d. - */ - token_type scan() + ~lexer() { - while (true) - { - // pointer for backtracking information - m_marker = nullptr; - - // remember the begin of the token - m_start = m_cursor; - assert(m_start != nullptr); - - - { - lexer_char_t yych; - unsigned int yyaccept = 0; - static const unsigned char yybm[] = - { - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 32, 32, 0, 0, 32, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 160, 128, 0, 128, 128, 128, 128, 128, - 128, 128, 128, 128, 128, 128, 128, 128, - 192, 192, 192, 192, 192, 192, 192, 192, - 192, 192, 128, 128, 128, 128, 128, 128, - 128, 128, 128, 128, 128, 128, 128, 128, - 128, 128, 128, 128, 128, 128, 128, 128, - 128, 128, 128, 128, 128, 128, 128, 128, - 128, 128, 128, 128, 0, 128, 128, 128, - 128, 128, 128, 128, 128, 128, 128, 128, - 128, 128, 128, 128, 128, 128, 128, 128, - 128, 128, 128, 128, 128, 128, 128, 128, - 128, 128, 128, 128, 128, 128, 128, 128, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, - }; - if ((m_limit - m_cursor) < 5) - { - fill_line_buffer(5); // LCOV_EXCL_LINE - } - yych = *m_cursor; - if (yybm[0 + yych] & 32) - { - goto basic_json_parser_6; - } - if (yych <= '[') - { - if (yych <= '-') - { - if (yych <= '"') - { - if (yych <= 0x00) - { - goto basic_json_parser_2; - } - if (yych <= '!') - { - goto basic_json_parser_4; - } - goto basic_json_parser_9; - } - else - { - if (yych <= '+') - { - goto basic_json_parser_4; - } - if (yych <= ',') - { - goto basic_json_parser_10; - } - goto basic_json_parser_12; - } - } - else - { - if (yych <= '9') - { - if (yych <= '/') - { - goto basic_json_parser_4; - } - if (yych <= '0') - { - goto basic_json_parser_13; - } - goto basic_json_parser_15; - } - else - { - if (yych <= ':') - { - goto basic_json_parser_17; - } - if (yych <= 'Z') - { - goto basic_json_parser_4; - } - goto basic_json_parser_19; - } - } - } - else - { - if (yych <= 'n') - { - if (yych <= 'e') - { - if (yych == ']') - { - goto basic_json_parser_21; - } - goto basic_json_parser_4; - } - else - { - if (yych <= 'f') - { - goto basic_json_parser_23; - } - if (yych <= 'm') - { - goto basic_json_parser_4; - } - goto basic_json_parser_24; - } - } - else - { - if (yych <= 'z') - { - if (yych == 't') - { - goto basic_json_parser_25; - } - goto basic_json_parser_4; - } - else - { - if (yych <= '{') - { - goto basic_json_parser_26; - } - if (yych == '}') - { - goto basic_json_parser_28; - } - goto basic_json_parser_4; - } - } - } -basic_json_parser_2: - ++m_cursor; - { - last_token_type = token_type::end_of_input; - break; - } -basic_json_parser_4: - ++m_cursor; -basic_json_parser_5: - { - last_token_type = token_type::parse_error; - break; - } -basic_json_parser_6: - ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } - yych = *m_cursor; - if (yybm[0 + yych] & 32) - { - goto basic_json_parser_6; - } - { - position += static_cast((m_cursor - m_start)); - continue; - } -basic_json_parser_9: - yyaccept = 0; - yych = *(m_marker = ++m_cursor); - if (yych <= 0x1F) - { - goto basic_json_parser_5; - } - if (yych <= 0x7F) - { - goto basic_json_parser_31; - } - if (yych <= 0xC1) - { - goto basic_json_parser_5; - } - if (yych <= 0xF4) - { - goto basic_json_parser_31; - } - goto basic_json_parser_5; -basic_json_parser_10: - ++m_cursor; - { - last_token_type = token_type::value_separator; - break; - } -basic_json_parser_12: - yych = *++m_cursor; - if (yych <= '/') - { - goto basic_json_parser_5; - } - if (yych <= '0') - { - goto basic_json_parser_43; - } - if (yych <= '9') - { - goto basic_json_parser_45; - } - goto basic_json_parser_5; -basic_json_parser_13: - yyaccept = 1; - yych = *(m_marker = ++m_cursor); - if (yych <= '9') - { - if (yych == '.') - { - goto basic_json_parser_47; - } - if (yych >= '0') - { - goto basic_json_parser_48; - } - } - else - { - if (yych <= 'E') - { - if (yych >= 'E') - { - goto basic_json_parser_51; - } - } - else - { - if (yych == 'e') - { - goto basic_json_parser_51; - } - } - } -basic_json_parser_14: - { - last_token_type = token_type::value_unsigned; - break; - } -basic_json_parser_15: - yyaccept = 1; - m_marker = ++m_cursor; - if ((m_limit - m_cursor) < 3) - { - fill_line_buffer(3); // LCOV_EXCL_LINE - } - yych = *m_cursor; - if (yybm[0 + yych] & 64) - { - goto basic_json_parser_15; - } - if (yych <= 'D') - { - if (yych == '.') - { - goto basic_json_parser_47; - } - goto basic_json_parser_14; - } - else - { - if (yych <= 'E') - { - goto basic_json_parser_51; - } - if (yych == 'e') - { - goto basic_json_parser_51; - } - goto basic_json_parser_14; - } -basic_json_parser_17: - ++m_cursor; - { - last_token_type = token_type::name_separator; - break; - } -basic_json_parser_19: - ++m_cursor; - { - last_token_type = token_type::begin_array; - break; - } -basic_json_parser_21: - ++m_cursor; - { - last_token_type = token_type::end_array; - break; - } -basic_json_parser_23: - yyaccept = 0; - yych = *(m_marker = ++m_cursor); - if (yych == 'a') - { - goto basic_json_parser_52; - } - goto basic_json_parser_5; -basic_json_parser_24: - yyaccept = 0; - yych = *(m_marker = ++m_cursor); - if (yych == 'u') - { - goto basic_json_parser_53; - } - goto basic_json_parser_5; -basic_json_parser_25: - yyaccept = 0; - yych = *(m_marker = ++m_cursor); - if (yych == 'r') - { - goto basic_json_parser_54; - } - goto basic_json_parser_5; -basic_json_parser_26: - ++m_cursor; - { - last_token_type = token_type::begin_object; - break; - } -basic_json_parser_28: - ++m_cursor; - { - last_token_type = token_type::end_object; - break; - } -basic_json_parser_30: - ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } - yych = *m_cursor; -basic_json_parser_31: - if (yybm[0 + yych] & 128) - { - goto basic_json_parser_30; - } - if (yych <= 0xE0) - { - if (yych <= '\\') - { - if (yych <= 0x1F) - { - goto basic_json_parser_32; - } - if (yych <= '"') - { - goto basic_json_parser_33; - } - goto basic_json_parser_35; - } - else - { - if (yych <= 0xC1) - { - goto basic_json_parser_32; - } - if (yych <= 0xDF) - { - goto basic_json_parser_36; - } - goto basic_json_parser_37; - } - } - else - { - if (yych <= 0xEF) - { - if (yych == 0xED) - { - goto basic_json_parser_39; - } - goto basic_json_parser_38; - } - else - { - if (yych <= 0xF0) - { - goto basic_json_parser_40; - } - if (yych <= 0xF3) - { - goto basic_json_parser_41; - } - if (yych <= 0xF4) - { - goto basic_json_parser_42; - } - } - } -basic_json_parser_32: - m_cursor = m_marker; - if (yyaccept <= 1) - { - if (yyaccept == 0) - { - goto basic_json_parser_5; - } - else - { - goto basic_json_parser_14; - } - } - else - { - if (yyaccept == 2) - { - goto basic_json_parser_44; - } - else - { - goto basic_json_parser_58; - } - } -basic_json_parser_33: - ++m_cursor; - { - last_token_type = token_type::value_string; - break; - } -basic_json_parser_35: - ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } - yych = *m_cursor; - if (yych <= 'e') - { - if (yych <= '/') - { - if (yych == '"') - { - goto basic_json_parser_30; - } - if (yych <= '.') - { - goto basic_json_parser_32; - } - goto basic_json_parser_30; - } - else - { - if (yych <= '\\') - { - if (yych <= '[') - { - goto basic_json_parser_32; - } - goto basic_json_parser_30; - } - else - { - if (yych == 'b') - { - goto basic_json_parser_30; - } - goto basic_json_parser_32; - } - } - } - else - { - if (yych <= 'q') - { - if (yych <= 'f') - { - goto basic_json_parser_30; - } - if (yych == 'n') - { - goto basic_json_parser_30; - } - goto basic_json_parser_32; - } - else - { - if (yych <= 's') - { - if (yych <= 'r') - { - goto basic_json_parser_30; - } - goto basic_json_parser_32; - } - else - { - if (yych <= 't') - { - goto basic_json_parser_30; - } - if (yych <= 'u') - { - goto basic_json_parser_55; - } - goto basic_json_parser_32; - } - } - } -basic_json_parser_36: - ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } - yych = *m_cursor; - if (yych <= 0x7F) - { - goto basic_json_parser_32; - } - if (yych <= 0xBF) - { - goto basic_json_parser_30; - } - goto basic_json_parser_32; -basic_json_parser_37: - ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } - yych = *m_cursor; - if (yych <= 0x9F) - { - goto basic_json_parser_32; - } - if (yych <= 0xBF) - { - goto basic_json_parser_36; - } - goto basic_json_parser_32; -basic_json_parser_38: - ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } - yych = *m_cursor; - if (yych <= 0x7F) - { - goto basic_json_parser_32; - } - if (yych <= 0xBF) - { - goto basic_json_parser_36; - } - goto basic_json_parser_32; -basic_json_parser_39: - ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } - yych = *m_cursor; - if (yych <= 0x7F) - { - goto basic_json_parser_32; - } - if (yych <= 0x9F) - { - goto basic_json_parser_36; - } - goto basic_json_parser_32; -basic_json_parser_40: - ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } - yych = *m_cursor; - if (yych <= 0x8F) - { - goto basic_json_parser_32; - } - if (yych <= 0xBF) - { - goto basic_json_parser_38; - } - goto basic_json_parser_32; -basic_json_parser_41: - ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } - yych = *m_cursor; - if (yych <= 0x7F) - { - goto basic_json_parser_32; - } - if (yych <= 0xBF) - { - goto basic_json_parser_38; - } - goto basic_json_parser_32; -basic_json_parser_42: - ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } - yych = *m_cursor; - if (yych <= 0x7F) - { - goto basic_json_parser_32; - } - if (yych <= 0x8F) - { - goto basic_json_parser_38; - } - goto basic_json_parser_32; -basic_json_parser_43: - yyaccept = 2; - yych = *(m_marker = ++m_cursor); - if (yych <= '9') - { - if (yych == '.') - { - goto basic_json_parser_47; - } - if (yych >= '0') - { - goto basic_json_parser_48; - } - } - else - { - if (yych <= 'E') - { - if (yych >= 'E') - { - goto basic_json_parser_51; - } - } - else - { - if (yych == 'e') - { - goto basic_json_parser_51; - } - } - } -basic_json_parser_44: - { - last_token_type = token_type::value_integer; - break; - } -basic_json_parser_45: - yyaccept = 2; - m_marker = ++m_cursor; - if ((m_limit - m_cursor) < 3) - { - fill_line_buffer(3); // LCOV_EXCL_LINE - } - yych = *m_cursor; - if (yych <= '9') - { - if (yych == '.') - { - goto basic_json_parser_47; - } - if (yych <= '/') - { - goto basic_json_parser_44; - } - goto basic_json_parser_45; - } - else - { - if (yych <= 'E') - { - if (yych <= 'D') - { - goto basic_json_parser_44; - } - goto basic_json_parser_51; - } - else - { - if (yych == 'e') - { - goto basic_json_parser_51; - } - goto basic_json_parser_44; - } - } -basic_json_parser_47: - yych = *++m_cursor; - if (yych <= '/') - { - goto basic_json_parser_32; - } - if (yych <= '9') - { - goto basic_json_parser_56; - } - goto basic_json_parser_32; -basic_json_parser_48: - ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } - yych = *m_cursor; - if (yych <= '/') - { - goto basic_json_parser_50; - } - if (yych <= '9') - { - goto basic_json_parser_48; - } -basic_json_parser_50: - { - last_token_type = token_type::parse_error; - break; - } -basic_json_parser_51: - yych = *++m_cursor; - if (yych <= ',') - { - if (yych == '+') - { - goto basic_json_parser_59; - } - goto basic_json_parser_32; - } - else - { - if (yych <= '-') - { - goto basic_json_parser_59; - } - if (yych <= '/') - { - goto basic_json_parser_32; - } - if (yych <= '9') - { - goto basic_json_parser_60; - } - goto basic_json_parser_32; - } -basic_json_parser_52: - yych = *++m_cursor; - if (yych == 'l') - { - goto basic_json_parser_62; - } - goto basic_json_parser_32; -basic_json_parser_53: - yych = *++m_cursor; - if (yych == 'l') - { - goto basic_json_parser_63; - } - goto basic_json_parser_32; -basic_json_parser_54: - yych = *++m_cursor; - if (yych == 'u') - { - goto basic_json_parser_64; - } - goto basic_json_parser_32; -basic_json_parser_55: - ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } - yych = *m_cursor; - if (yych <= '@') - { - if (yych <= '/') - { - goto basic_json_parser_32; - } - if (yych <= '9') - { - goto basic_json_parser_65; - } - goto basic_json_parser_32; - } - else - { - if (yych <= 'F') - { - goto basic_json_parser_65; - } - if (yych <= '`') - { - goto basic_json_parser_32; - } - if (yych <= 'f') - { - goto basic_json_parser_65; - } - goto basic_json_parser_32; - } -basic_json_parser_56: - yyaccept = 3; - m_marker = ++m_cursor; - if ((m_limit - m_cursor) < 3) - { - fill_line_buffer(3); // LCOV_EXCL_LINE - } - yych = *m_cursor; - if (yych <= 'D') - { - if (yych <= '/') - { - goto basic_json_parser_58; - } - if (yych <= '9') - { - goto basic_json_parser_56; - } - } - else - { - if (yych <= 'E') - { - goto basic_json_parser_51; - } - if (yych == 'e') - { - goto basic_json_parser_51; - } - } -basic_json_parser_58: - { - last_token_type = token_type::value_float; - break; - } -basic_json_parser_59: - yych = *++m_cursor; - if (yych <= '/') - { - goto basic_json_parser_32; - } - if (yych >= ':') - { - goto basic_json_parser_32; - } -basic_json_parser_60: - ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } - yych = *m_cursor; - if (yych <= '/') - { - goto basic_json_parser_58; - } - if (yych <= '9') - { - goto basic_json_parser_60; - } - goto basic_json_parser_58; -basic_json_parser_62: - yych = *++m_cursor; - if (yych == 's') - { - goto basic_json_parser_66; - } - goto basic_json_parser_32; -basic_json_parser_63: - yych = *++m_cursor; - if (yych == 'l') - { - goto basic_json_parser_67; - } - goto basic_json_parser_32; -basic_json_parser_64: - yych = *++m_cursor; - if (yych == 'e') - { - goto basic_json_parser_69; - } - goto basic_json_parser_32; -basic_json_parser_65: - ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } - yych = *m_cursor; - if (yych <= '@') - { - if (yych <= '/') - { - goto basic_json_parser_32; - } - if (yych <= '9') - { - goto basic_json_parser_71; - } - goto basic_json_parser_32; - } - else - { - if (yych <= 'F') - { - goto basic_json_parser_71; - } - if (yych <= '`') - { - goto basic_json_parser_32; - } - if (yych <= 'f') - { - goto basic_json_parser_71; - } - goto basic_json_parser_32; - } -basic_json_parser_66: - yych = *++m_cursor; - if (yych == 'e') - { - goto basic_json_parser_72; - } - goto basic_json_parser_32; -basic_json_parser_67: - ++m_cursor; - { - last_token_type = token_type::literal_null; - break; - } -basic_json_parser_69: - ++m_cursor; - { - last_token_type = token_type::literal_true; - break; - } -basic_json_parser_71: - ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } - yych = *m_cursor; - if (yych <= '@') - { - if (yych <= '/') - { - goto basic_json_parser_32; - } - if (yych <= '9') - { - goto basic_json_parser_74; - } - goto basic_json_parser_32; - } - else - { - if (yych <= 'F') - { - goto basic_json_parser_74; - } - if (yych <= '`') - { - goto basic_json_parser_32; - } - if (yych <= 'f') - { - goto basic_json_parser_74; - } - goto basic_json_parser_32; - } -basic_json_parser_72: - ++m_cursor; - { - last_token_type = token_type::literal_false; - break; - } -basic_json_parser_74: - ++m_cursor; - if (m_limit <= m_cursor) - { - fill_line_buffer(1); // LCOV_EXCL_LINE - } - yych = *m_cursor; - if (yych <= '@') - { - if (yych <= '/') - { - goto basic_json_parser_32; - } - if (yych <= '9') - { - goto basic_json_parser_30; - } - goto basic_json_parser_32; - } - else - { - if (yych <= 'F') - { - goto basic_json_parser_30; - } - if (yych <= '`') - { - goto basic_json_parser_32; - } - if (yych <= 'f') - { - goto basic_json_parser_30; - } - goto basic_json_parser_32; - } - } - - } - - position += static_cast((m_cursor - m_start)); - return last_token_type; + delete ia; } - /*! - @brief append data from the stream to the line buffer + // switch off unwanted functions (due to pointer members) + lexer() = delete; + lexer(const lexer&) = delete; + lexer operator=(const lexer&) = delete; - This function is called by the scan() function when the end of the - buffer (`m_limit`) is reached and the `m_cursor` pointer cannot be - incremented without leaving the limits of the line buffer. Note re2c - decides when to call this function. + private: + ///////////////////// + // scan functions + ///////////////////// - If the lexer reads from contiguous storage, there is no trailing null - byte. Therefore, this function must make sure to add these padding - null bytes. - - If the lexer reads from an input stream, this function reads the next - line of the input. - - @pre - p p p p p p u u u u u x . . . . . . - ^ ^ ^ ^ - m_content m_start | m_limit - m_cursor - - @post - u u u u u x x x x x x x . . . . . . - ^ ^ ^ - | m_cursor m_limit - m_start - m_content - */ - void fill_line_buffer(size_t n = 0) + // must be called after \u was read; returns following xxxx as hex or -1 when error + int get_codepoint() { - // if line buffer is used, m_content points to its data - assert(m_line_buffer.empty() - or m_content == reinterpret_cast(m_line_buffer.data())); - - // if line buffer is used, m_limit is set past the end of its data - assert(m_line_buffer.empty() - or m_limit == m_content + m_line_buffer.size()); - - // pointer relationships - assert(m_content <= m_start); - assert(m_start <= m_cursor); - assert(m_cursor <= m_limit); - assert(m_marker == nullptr or m_marker <= m_limit); - - // number of processed characters (p) - const auto num_processed_chars = static_cast(m_start - m_content); - // offset for m_marker wrt. to m_start - const auto offset_marker = (m_marker == nullptr) ? 0 : m_marker - m_start; - // number of unprocessed characters (u) - const auto offset_cursor = m_cursor - m_start; - - // no stream is used or end of file is reached - if (m_stream == nullptr or m_stream->eof()) + // read xxxx of \uxxxx + std::vector buffer(5, '\0'); + for (size_t i = 0; i < 4; ++i) { - // m_start may or may not be pointing into m_line_buffer at - // this point. We trust the standard library to do the right - // thing. See http://stackoverflow.com/q/28142011/266378 - m_line_buffer.assign(m_start, m_limit); - - // append n characters to make sure that there is sufficient - // space between m_cursor and m_limit - m_line_buffer.append(1, '\x00'); - if (n > 0) + get(); + if (JSON_UNLIKELY(current != std::char_traits::eof())) { - m_line_buffer.append(n - 1, '\x01'); - } - } - else - { - // delete processed characters from line buffer - m_line_buffer.erase(0, num_processed_chars); - // read next line from input stream - m_line_buffer_tmp.clear(); - - // check if stream is still good - if (m_stream->fail()) - { - JSON_THROW(parse_error(111, 0, "bad input stream")); - } - - std::getline(*m_stream, m_line_buffer_tmp, '\n'); - - // add line with newline symbol to the line buffer - m_line_buffer += m_line_buffer_tmp; - m_line_buffer.push_back('\n'); - } - - // set pointers - m_content = reinterpret_cast(m_line_buffer.data()); - assert(m_content != nullptr); - m_start = m_content; - m_marker = m_start + offset_marker; - m_cursor = m_start + offset_cursor; - m_limit = m_start + m_line_buffer.size(); - } - - /// return string representation of last read token - string_t get_token_string() const - { - assert(m_start != nullptr); - return string_t(reinterpret_cast(m_start), - static_cast(m_cursor - m_start)); - } - - /*! - @brief return string value for string tokens - - The function iterates the characters between the opening and closing - quotes of the string value. The complete string is the range - [m_start,m_cursor). Consequently, we iterate from m_start+1 to - m_cursor-1. - - We differentiate two cases: - - 1. Escaped characters. In this case, a new character is constructed - according to the nature of the escape. Some escapes create new - characters (e.g., `"\\n"` is replaced by `"\n"`), some are copied - as is (e.g., `"\\\\"`). Furthermore, Unicode escapes of the shape - `"\\uxxxx"` need special care. In this case, to_unicode takes care - of the construction of the values. - 2. Unescaped characters are copied as is. - - @pre `m_cursor - m_start >= 2`, meaning the length of the last token - is at least 2 bytes which is trivially true for any string (which - consists of at least two quotes). - - " c1 c2 c3 ... " - ^ ^ - m_start m_cursor - - @complexity Linear in the length of the string.\n - - Lemma: The loop body will always terminate.\n - - Proof (by contradiction): Assume the loop body does not terminate. As - the loop body does not contain another loop, one of the called - functions must never return. The called functions are `std::strtoul` - and to_unicode. Neither function can loop forever, so the loop body - will never loop forever which contradicts the assumption that the loop - body does not terminate, q.e.d.\n - - Lemma: The loop condition for the for loop is eventually false.\n - - Proof (by contradiction): Assume the loop does not terminate. Due to - the above lemma, this can only be due to a tautological loop - condition; that is, the loop condition i < m_cursor - 1 must always be - true. Let x be the change of i for any loop iteration. Then - m_start + 1 + x < m_cursor - 1 must hold to loop indefinitely. This - can be rephrased to m_cursor - m_start - 2 > x. With the - precondition, we x <= 0, meaning that the loop condition holds - indefinitely if i is always decreased. However, observe that the value - of i is strictly increasing with each iteration, as it is incremented - by 1 in the iteration expression and never decremented inside the loop - body. Hence, the loop condition will eventually be false which - contradicts the assumption that the loop condition is a tautology, - q.e.d. - - @return string value of current token without opening and closing - quotes - @throw parse_error.102 if to_unicode fails or surrogate error - @throw parse_error.103 if to_unicode fails - */ - string_t get_string() const - { - assert(m_cursor - m_start >= 2); - - string_t result; - result.reserve(static_cast(m_cursor - m_start - 2)); - - // iterate the result between the quotes - for (const lexer_char_t* i = m_start + 1; i < m_cursor - 1; ++i) - { - // find next escape character - auto e = std::find(i, m_cursor - 1, '\\'); - if (e != i) - { - // see https://github.com/nlohmann/json/issues/365#issuecomment-262874705 - for (auto k = i; k < e; k++) - { - result.push_back(static_cast(*k)); - } - i = e - 1; // -1 because of ++i + buffer[i] = static_cast(current); } else { - // processing escaped character - // read next character - ++i; - - switch (*i) - { - // the default escapes - case 't': - { - result += "\t"; - break; - } - case 'b': - { - result += "\b"; - break; - } - case 'f': - { - result += "\f"; - break; - } - case 'n': - { - result += "\n"; - break; - } - case 'r': - { - result += "\r"; - break; - } - case '\\': - { - result += "\\"; - break; - } - case '/': - { - result += "/"; - break; - } - case '"': - { - result += "\""; - break; - } - - // unicode - case 'u': - { - // get code xxxx from uxxxx - auto codepoint = std::strtoul(std::string(reinterpret_cast(i + 1), - 4).c_str(), nullptr, 16); - - // check if codepoint is a high surrogate - if (codepoint >= 0xD800 and codepoint <= 0xDBFF) - { - // make sure there is a subsequent unicode - if ((i + 6 >= m_limit) or * (i + 5) != '\\' or * (i + 6) != 'u') - { - JSON_THROW(parse_error(102, get_position(), "missing low surrogate")); - } - - // get code yyyy from uxxxx\uyyyy - auto codepoint2 = std::strtoul(std::string(reinterpret_cast - (i + 7), 4).c_str(), nullptr, 16); - result += to_unicode(codepoint, codepoint2); - // skip the next 10 characters (xxxx\uyyyy) - i += 10; - } - else if (codepoint >= 0xDC00 and codepoint <= 0xDFFF) - { - // we found a lone low surrogate - JSON_THROW(parse_error(102, get_position(), "missing high surrogate")); - } - else - { - // add unicode character(s) - result += to_unicode(codepoint); - // skip the next four characters (xxxx) - i += 4; - } - break; - } - } + // error message will be created by caller + return -1; } } - return result; + char* endptr; + errno = 0; + + const int codepoint = static_cast(std::strtoul(buffer.data(), &endptr, 16)); + + if (JSON_LIKELY(errno == 0 and endptr == buffer.data() + 4)) + { + return codepoint; + } + else + { + // conversion incomplete or failure + return -1; + } } - - /*! - @brief parse string into a built-in arithmetic type as if the current - locale is POSIX. - - @note in floating-point case strtod may parse past the token's end - - this is not an error - - @note any leading blanks are not handled - */ - struct strtonum + token_type scan_string() { - public: - strtonum(const char* start, const char* end) - : m_start(start), m_end(end) - {} + // reset yytext (ignore opening quote) + reset(); - /*! - @return true iff parsed successfully as number of type T - - @param[in,out] val shall contain parsed value, or undefined value - if could not parse - */ - template::value>::type> - bool to(T& val) const + while (true) { - return parse(val, std::is_integral()); - } + get(); - private: - const char* const m_start = nullptr; - const char* const m_end = nullptr; - - // floating-point conversion - - // overloaded wrappers for strtod/strtof/strtold - // that will be called from parse - static void strtof(float& f, const char* str, char** endptr) - { - f = std::strtof(str, endptr); - } - - static void strtof(double& f, const char* str, char** endptr) - { - f = std::strtod(str, endptr); - } - - static void strtof(long double& f, const char* str, char** endptr) - { - f = std::strtold(str, endptr); - } - - template - bool parse(T& value, /*is_integral=*/std::false_type) const - { - // replace decimal separator with locale-specific version, - // when necessary; data will point to either the original - // string, or buf, or tempstr containing the fixed string. - std::string tempstr; - std::array buf; - const size_t len = static_cast(m_end - m_start); - - // lexer will reject empty numbers - assert(len > 0); - - // since dealing with strtod family of functions, we're - // getting the decimal point char from the C locale facilities - // instead of C++'s numpunct facet of the current std::locale - const auto loc = localeconv(); - assert(loc != nullptr); - const char decimal_point_char = (loc->decimal_point == nullptr) ? '.' : loc->decimal_point[0]; - - const char* data = m_start; - - if (decimal_point_char != '.') + // end of file while parsing string + if (JSON_UNLIKELY(current == std::char_traits::eof())) { - const size_t ds_pos = static_cast(std::find(m_start, m_end, '.') - m_start); + error_message = "invalid string: missing closing quote"; + return token_type::parse_error; + } - if (ds_pos != len) + // control character + if (JSON_UNLIKELY('\x00' <= current and current <= '\x1f')) + { + error_message = "invalid string: control characters (U+0000 through U+001f) must be escaped"; + return token_type::parse_error; + } + + switch (current) + { + // closing quote + case '\"': { - // copy the data into the local buffer or tempstr, if - // buffer is too small; replace decimal separator, and - // update data to point to the modified bytes - if ((len + 1) < buf.size()) + add('\0'); + --yylen; + return token_type::value_string; + } + + // escape sequence + case '\\': + { + switch (get()) { - std::copy(m_start, m_end, buf.begin()); - buf[len] = 0; - buf[ds_pos] = decimal_point_char; - data = buf.data(); - } - else - { - tempstr.assign(m_start, m_end); - tempstr[ds_pos] = decimal_point_char; - data = tempstr.c_str(); + // quotation mark + case '\"': + add('\"'); + break; + // reverse solidus + case '\\': + add('\\'); + break; + // solidus + case '/': + add('/'); + break; + // backspace + case 'b': + add('\b'); + break; + // form feed + case 'f': + add('\f'); + break; + // line feed + case 'n': + add('\n'); + break; + // carriage return + case 'r': + add('\r'); + break; + // tab + case 't': + add('\t'); + break; + + // unicode escapes + case 'u': + { + int codepoint; + int codepoint1 = get_codepoint(); + + if (JSON_UNLIKELY(codepoint1 == -1)) + { + error_message = "invalid string: '\\u' must be followed by 4 hex digits"; + return token_type::parse_error; + } + + // check if code point is a high surrogate + if (codepoint1 >= 0xD800 and codepoint1 <= 0xDBFF) + { + // expect next \uxxxx entry + if (JSON_LIKELY(get() == '\\' and get() == 'u')) + { + int codepoint2 = get_codepoint(); + + if (JSON_UNLIKELY(codepoint2 == -1)) + { + error_message = "invalid string: '\\u' must be followed by 4 hex digits"; + return token_type::parse_error; + } + + // check if codepoint2 is a low surrogate + if (codepoint2 >= 0xDC00 and codepoint2 <= 0xDFFF) + { + codepoint = + // high surrogate occupies the most significant 22 bits + (codepoint1 << 10) + // low surrogate occupies the least significant 15 bits + + codepoint2 + // there is still the 0xD800, 0xDC00 and 0x10000 noise + // in the result so we have to subtract with: + // (0xD800 << 10) + DC00 - 0x10000 = 0x35FDC00 + - 0x35FDC00; + } + else + { + error_message = "invalid string: invalid low surrogate"; + return token_type::parse_error; + } + } + else + { + error_message = "invalid string: missing low surrogate"; + return token_type::parse_error; + } + } + else + { + if (JSON_UNLIKELY(codepoint1 >= 0xDC00 and codepoint1 <= 0xDFFF)) + { + error_message = "invalid string: missing high surrogate"; + return token_type::parse_error; + } + + // only work with first code point + codepoint = codepoint1; + } + + // translate code point to bytes + if (codepoint < 0x80) + { + // 1-byte characters: 0xxxxxxx (ASCII) + add(codepoint); + } + else if (codepoint <= 0x7ff) + { + // 2-byte characters: 110xxxxx 10xxxxxx + add(0xC0 | (codepoint >> 6)); + add(0x80 | (codepoint & 0x3F)); + } + else if (codepoint <= 0xffff) + { + // 3-byte characters: 1110xxxx 10xxxxxx 10xxxxxx + add(0xE0 | (codepoint >> 12)); + add(0x80 | ((codepoint >> 6) & 0x3F)); + add(0x80 | (codepoint & 0x3F)); + } + else if (codepoint <= 0x10ffff) + { + // 4-byte characters: 11110xxx 10xxxxxx 10xxxxxx 10xxxxxx + add(0xF0 | (codepoint >> 18)); + add(0x80 | ((codepoint >> 12) & 0x3F)); + add(0x80 | ((codepoint >> 6) & 0x3F)); + add(0x80 | (codepoint & 0x3F)); + } + else + { + error_message = "invalid string: code points above U+10FFFF are invalid"; + return token_type::parse_error; + } + + break; + } + + // other characters after escape + default: + error_message = "invalid string: forbidden character after backspace"; + return token_type::parse_error; } + + break; + } + + // any other character + default: + { + add(current); + break; } } + } + } - char* endptr = nullptr; - value = 0; - // this calls appropriate overload depending on T - strtof(value, data, &endptr); + token_type scan_number() + { + static unsigned char lookup[9][256] = + { + {10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 1, 10, 10, 3, 2, 2, 2, 2, 2, 2, 2, 2, 2, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10}, + {10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 3, 2, 2, 2, 2, 2, 2, 2, 2, 2, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10}, + {9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 4, 9, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 5, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 5, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9}, + {9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 4, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 5, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 5, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9}, + {10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 5, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 5, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10}, + {10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 8, 10, 8, 10, 10, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10}, + {9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 5, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 5, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9}, + {9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9}, + {10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10} + }; - // parsing was successful iff strtof parsed exactly the number - // of characters determined by the lexer (len) - const bool ok = (endptr == (data + len)); + reset(); - if (ok and (value == static_cast(0.0)) and (*data == '-')) + bool has_sign = false; + bool has_exp = false; + bool has_point = false; + + int state = lookup[0][static_cast(current)]; + int old_state = 0; + + while (state != 9) + { + has_sign = has_sign or (state == 1); + has_point = has_point or (state == 4); + has_exp = has_exp or (state == 5); + + if (JSON_UNLIKELY(state == 10)) { - // some implementations forget to negate the zero - value = -0.0; + // create error message based on previous state + switch (old_state) + { + case 0: + error_message = "invalid number; expected '-' or digit"; + break; + case 1: + error_message = "invalid number; expected digit after '-'"; + break; + case 4: + error_message = "invalid number; expected digit after '.'"; + break; + case 5: + error_message = "invalid number; expected '+', '-', or digit after exponent"; + break; + case 8: + error_message = "invalid number; expected digit after exponent sign"; + break; + default: + assert(false); // no error in the other states + break; + } + return token_type::parse_error; } - return ok; + add(current); + get(); + old_state = state; + state = lookup[state][static_cast(current)]; } - // integral conversion + // unget the character after the number + unget(); - signed long long parse_integral(char** endptr, /*is_signed*/std::true_type) const + // terminate token + add('\0'); + --yylen; + + if (has_exp or has_point) { - return std::strtoll(m_start, endptr, 10); + value_float = std::strtod(yytext.data(), nullptr); + return token_type::value_float; } - unsigned long long parse_integral(char** endptr, /*is_signed*/std::false_type) const - { - return std::strtoull(m_start, endptr, 10); - } - - template - bool parse(T& value, /*is_integral=*/std::true_type) const + if (has_sign) { char* endptr = nullptr; - errno = 0; // these are thread-local - const auto x = parse_integral(&endptr, std::is_signed()); - - // called right overload? - static_assert(std::is_signed() == std::is_signed(), ""); - - value = static_cast(x); - - return (x == static_cast(value)) // x fits into destination T - and (x < 0) == (value < 0) // preserved sign - //and ((x != 0) or is_integral()) // strto[u]ll did nto fail - and (errno == 0) // strto[u]ll did not overflow - and (m_start < m_end) // token was not empty - and (endptr == m_end); // parsed entire token exactly + value_integer = std::strtoll(yytext.data(), &endptr, 10); + return token_type::value_integer; } - }; + else + { + char* endptr = nullptr; + value_unsigned = std::strtoull(yytext.data(), &endptr, 10); + return token_type::value_unsigned; + } + } - /*! - @brief return number value for number tokens + token_type scan_true() + { + if (JSON_LIKELY((get() == 'r' and get() == 'u' and get() == 'e'))) + { + return token_type::literal_true; + } - This function translates the last token into the most appropriate - number type (either integer, unsigned integer or floating point), - which is passed back to the caller via the result parameter. + error_message = "invalid literal; expected 'true'"; + return token_type::parse_error; + } - integral numbers that don't fit into the the range of the respective - type are parsed as number_float_t + token_type scan_false() + { + if (JSON_LIKELY((get() == 'a' and get() == 'l' and get() == 's' and get() == 'e'))) + { + return token_type::literal_false; + } - floating-point values do not satisfy std::isfinite predicate - are converted to value_t::null + error_message = "invalid literal; expected 'false'"; + return token_type::parse_error; + } - throws if the entire string [m_start .. m_cursor) cannot be - interpreted as a number + token_type scan_null() + { + if (JSON_LIKELY((get() == 'u' and get() == 'l' and get() == 'l'))) + { + return token_type::literal_null; + } + + error_message = "invalid literal; expected 'null'"; + return token_type::parse_error; + } + + ///////////////////// + // input management + ///////////////////// + + void reset() + { + yylen = 0; + start_pos = chars_read - 1; + } + + // get a character from the input + int get() + { + ++chars_read; + + if (JSON_UNLIKELY(next_unget)) + { + next_unget = false; + } + else + { + current = ia->get_character(); + } + + return current; + } + + // unget a character to the input + void unget() + { + --chars_read; + next_unget = true; + } + + // add a character to yytext + void add(int c) + { + // resize yytext if necessary + if (JSON_UNLIKELY((yylen + 1 > yytext.capacity()))) + { + yytext.resize(2 * yytext.capacity(), '\0'); + } + yytext[yylen++] = static_cast(c); + } + + public: + constexpr size_t get_position() const + { + return chars_read; + } + + const std::string get_string() + { + return std::string(yytext.data(), yylen); + } + + std::string get_token_string() const + { + std::string s = ia->read(start_pos, chars_read - start_pos); + std::stringstream ss; + + for (auto c : s) + { + if (c == '\0' or c == std::char_traits::eof()) + { + continue; + } + else if ('\x00' <= c and c <= '\x1f') + { + ss << ""; + } + else + { + ss << c; + } + } + + return ss.str(); + } + + const std::string& get_error_message() const + { + return error_message; + } - @param[out] result @ref basic_json object to receive the number. - @param[in] token the type of the number token - */ bool get_number(basic_json& result, const token_type token) const { - assert(m_start != nullptr); - assert(m_start < m_cursor); - assert((token == token_type::value_unsigned) or - (token == token_type::value_integer) or - (token == token_type::value_float)); - - strtonum num_converter(reinterpret_cast(m_start), - reinterpret_cast(m_cursor)); - switch (token) { case lexer::token_type::value_unsigned: { - number_unsigned_t val; - if (num_converter.to(val)) - { - // parsing successful - result.m_type = value_t::number_unsigned; - result.m_value = val; - return true; - } - break; + result.m_type = value_t::number_unsigned; + result.m_value = static_cast(value_unsigned); + return true; } case lexer::token_type::value_integer: { - number_integer_t val; - if (num_converter.to(val)) - { - // parsing successful - result.m_type = value_t::number_integer; - result.m_value = val; - return true; - } - break; + result.m_type = value_t::number_integer; + result.m_value = static_cast(value_integer); + return true; + } + + case lexer::token_type::value_float: + { + result.m_type = value_t::number_float; + result.m_value = static_cast(value_float); + return true; } default: { - break; + return false; } } - - // parse float (either explicitly or because a previous conversion - // failed) - number_float_t val; - if (num_converter.to(val)) - { - // parsing successful - result.m_type = value_t::number_float; - result.m_value = val; - - // throw in case of infinity or NAN - if (not std::isfinite(result.m_value.number_float)) - { - JSON_THROW(out_of_range(406, "number overflow parsing '" + get_token_string() + "'")); - } - - return true; - } - - // couldn't parse number in any format - return false; } - constexpr size_t get_position() const + token_type scan() { - return position; + // read next character and ignore whitespace + do + { + get(); + } + while (current == ' ' or current == '\t' or current == '\n' or current == '\r'); + + switch (current) + { + // structural characters + case '[': + return token_type::begin_array; + case ']': + return token_type::end_array; + case '{': + return token_type::begin_object; + case '}': + return token_type::end_object; + case ':': + return token_type::name_separator; + case ',': + return token_type::value_separator; + + // literals + case 't': + return scan_true(); + case 'f': + return scan_false(); + case 'n': + return scan_null(); + + // string + case '\"': + return scan_string(); + + // number + case '-': + case '0': + case '1': + case '2': + case '3': + case '4': + case '5': + case '6': + case '7': + case '8': + case '9': + return scan_number(); + + // end of input + case '\0': + case std::char_traits::eof(): + return token_type::end_of_input; + + // error + default: + error_message = "invalid literal"; + return token_type::parse_error; + } } private: - /// optional input stream - std::istream* m_stream = nullptr; - /// line buffer buffer for m_stream - string_t m_line_buffer {}; - /// used for filling m_line_buffer - string_t m_line_buffer_tmp {}; - /// the buffer pointer - const lexer_char_t* m_content = nullptr; - /// pointer to the beginning of the current symbol - const lexer_char_t* m_start = nullptr; - /// pointer for backtracking information - const lexer_char_t* m_marker = nullptr; - /// pointer to the current symbol - const lexer_char_t* m_cursor = nullptr; - /// pointer to the end of the buffer - const lexer_char_t* m_limit = nullptr; - /// the last token type - token_type last_token_type = token_type::end_of_input; - /// current position in the input (read bytes) - size_t position = 0; + /// input adapter + input_adapter* ia = nullptr; + + /// the current character + int current = std::char_traits::eof(); + + /// whether get() should return the last character again + bool next_unget = false; + + /// the number of characters read + size_t chars_read = 0; + /// the start position of the current token + size_t start_pos = 0; + + /// buffer for variable-length tokens (numbers, strings) + std::vector yytext = std::vector(1024, '\0'); + /// current index in yytext + size_t yylen = 0; + + /// a description of occurred lexer errors + std::string error_message = ""; + + // number values + long long value_integer = 0; + unsigned long long value_unsigned = 0; + double value_float = 0; }; /*! @@ -12035,7 +11062,7 @@ basic_json_parser_74: /// a parser reading from a string literal parser(const char* buff, const parser_callback_t cb = nullptr) : callback(cb), - m_lexer(reinterpret_cast(buff), std::strlen(buff)) + m_lexer(buff, std::strlen(buff)) {} /*! @@ -12053,7 +11080,7 @@ basic_json_parser_74: = 0> parser(IteratorType first, IteratorType last, const parser_callback_t cb = nullptr) : callback(cb), - m_lexer(reinterpret_cast(&(*first)), + m_lexer(reinterpret_cast(&(*first)), static_cast(std::distance(first, last))) {} @@ -12071,7 +11098,8 @@ basic_json_parser_74: basic_json result = parse_internal(true); result.assert_invariant(); - expect(lexer::token_type::end_of_input); + // FIXME: this is new behavior + //expect(lexer::token_type::end_of_input); // return parser result and replace it with null in case the // top-level value was discarded by the callback function @@ -12129,6 +11157,8 @@ basic_json_parser_74: // store key expect(lexer::token_type::value_string); + // FIXME get_string returns const char*; maybe we can + // avoid this copy in the future const auto key = m_lexer.get_string(); bool keep_tag = false; @@ -12235,9 +11265,8 @@ basic_json_parser_74: case lexer::token_type::value_string: { - const auto s = m_lexer.get_string(); + result = basic_json(m_lexer.get_string()); get_token(); - result = basic_json(s); break; } @@ -12294,10 +11323,16 @@ basic_json_parser_74: { if (t != last_token) { - std::string error_msg = "parse error - unexpected "; - error_msg += (last_token == lexer::token_type::parse_error ? ("'" + m_lexer.get_token_string() + - "'") : - lexer::token_type_name(last_token)); + std::string error_msg = "syntax error - "; + if (last_token == lexer::token_type::parse_error) + { + error_msg += m_lexer.get_error_message() + "; last read: '" + m_lexer.get_token_string() + "'"; + } + else + { + error_msg += "unexpected " + lexer::token_type_name(last_token); + } + error_msg += "; expected " + lexer::token_type_name(t); JSON_THROW(parse_error(101, m_lexer.get_position(), error_msg)); } @@ -12310,10 +11345,16 @@ basic_json_parser_74: { if (t == last_token) { - std::string error_msg = "parse error - unexpected "; - error_msg += (last_token == lexer::token_type::parse_error ? ("'" + m_lexer.get_token_string() + - "'") : - lexer::token_type_name(last_token)); + std::string error_msg = "syntax error - "; + if (last_token == lexer::token_type::parse_error) + { + error_msg += m_lexer.get_error_message() + "; last read '" + m_lexer.get_token_string() + "'"; + } + else + { + error_msg += "unexpected " + lexer::token_type_name(last_token); + } + JSON_THROW(parse_error(101, m_lexer.get_position(), error_msg)); } } @@ -13826,5 +12867,7 @@ inline nlohmann::json::json_pointer operator "" _json_pointer(const char* s, std #undef JSON_CATCH #undef JSON_THROW #undef JSON_TRY +#undef JSON_LIKELY +#undef JSON_UNLIKELY #endif diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c deleted file mode 100644 index 6a1e09e3..00000000 --- a/src/json.hpp.re2c +++ /dev/null @@ -1,12863 +0,0 @@ -/* - __ _____ _____ _____ - __| | __| | | | JSON for Modern C++ -| | |__ | | | | | | version 2.1.1 -|_____|_____|_____|_|___| https://github.com/nlohmann/json - -Licensed under the MIT License . -Copyright (c) 2013-2017 Niels Lohmann . - -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. -*/ - -#ifndef NLOHMANN_JSON_HPP -#define NLOHMANN_JSON_HPP - -#include // all_of, copy, fill, find, for_each, none_of, remove, reverse, transform -#include // array -#include // assert -#include // and, not, or -#include // lconv, localeconv -#include // isfinite, labs, ldexp, signbit -#include // nullptr_t, ptrdiff_t, size_t -#include // int64_t, uint64_t -#include // abort, strtod, strtof, strtold, strtoul, strtoll, strtoull -#include // strlen -#include // forward_list -#include // function, hash, less -#include // initializer_list -#include // istream, ostream -#include // advance, begin, back_inserter, bidirectional_iterator_tag, distance, end, inserter, iterator, iterator_traits, next, random_access_iterator_tag, reverse_iterator -#include // numeric_limits -#include // locale -#include // map -#include // addressof, allocator, allocator_traits, unique_ptr -#include // accumulate -#include // stringstream -#include // getline, stoi, string, to_string -#include // add_pointer, conditional, decay, enable_if, false_type, integral_constant, is_arithmetic, is_base_of, is_const, is_constructible, is_convertible, is_default_constructible, is_enum, is_floating_point, is_integral, is_nothrow_move_assignable, is_nothrow_move_constructible, is_pointer, is_reference, is_same, is_scalar, is_signed, remove_const, remove_cv, remove_pointer, remove_reference, true_type, underlying_type -#include // declval, forward, make_pair, move, pair, swap -#include // vector - -// exclude unsupported compilers -#if defined(__clang__) - #if (__clang_major__ * 10000 + __clang_minor__ * 100 + __clang_patchlevel__) < 30400 - #error "unsupported Clang version - see https://github.com/nlohmann/json#supported-compilers" - #endif -#elif defined(__GNUC__) - #if (__GNUC__ * 10000 + __GNUC_MINOR__ * 100 + __GNUC_PATCHLEVEL__) < 40900 - #error "unsupported GCC version - see https://github.com/nlohmann/json#supported-compilers" - #endif -#endif - -// disable float-equal warnings on GCC/clang -#if defined(__clang__) || defined(__GNUC__) || defined(__GNUG__) - #pragma GCC diagnostic push - #pragma GCC diagnostic ignored "-Wfloat-equal" -#endif - -// disable documentation warnings on clang -#if defined(__clang__) - #pragma GCC diagnostic push - #pragma GCC diagnostic ignored "-Wdocumentation" -#endif - -// allow to disable exceptions -#if (defined(__cpp_exceptions) || defined(__EXCEPTIONS) || defined(_CPPUNWIND)) && not defined(JSON_NOEXCEPTION) - #define JSON_THROW(exception) throw exception - #define JSON_TRY try - #define JSON_CATCH(exception) catch(exception) -#else - #define JSON_THROW(exception) std::abort() - #define JSON_TRY if(true) - #define JSON_CATCH(exception) if(false) -#endif - -/*! -@brief namespace for Niels Lohmann -@see https://github.com/nlohmann -@since version 1.0.0 -*/ -namespace nlohmann -{ - -/*! -@brief unnamed namespace with internal helper functions - -This namespace collects some functions that could not be defined inside the -@ref basic_json class. - -@since version 2.1.0 -*/ -namespace detail -{ -//////////////// -// exceptions // -//////////////// - -/*! -@brief general exception of the @ref basic_json class - -Extension of std::exception objects with a member @a id for exception ids. - -@since version 3.0.0 -*/ -class exception : public std::exception -{ - public: - /// create exception with id an explanatory string - exception(int id_, const std::string& ename, const std::string& what_arg_) - : id(id_), - what_arg("[json.exception." + ename + "." + std::to_string(id_) + "] " + what_arg_) - {} - - /// returns the explanatory string - virtual const char* what() const noexcept override - { - return what_arg.c_str(); - } - - /// the id of the exception - const int id; - - private: - /// the explanatory string - const std::string what_arg; -}; - -/*! -@brief exception indicating a parse error - -This excpetion is thrown by the library when a parse error occurs. Parse -errors can occur during the deserialization of JSON text as well as when -using JSON Patch. - -Member @a byte holds the byte index of the last read character in the input -file. - -@note For an input with n bytes, 1 is the index of the first character - and n+1 is the index of the terminating null byte or the end of - file. This also holds true when reading a byte vector (CBOR or - MessagePack). - -Exceptions have ids 1xx. - -name / id | example massage | description ------------------------------- | --------------- | ------------------------- -json.exception.parse_error.101 | parse error at 2: unexpected end of input; expected string literal | This error indicates a syntax error while deserializing a JSON text. The error message describes that an unexpected token (character) was encountered, and the member @a byte indicates the error position. -json.exception.parse_error.102 | parse error at 14: missing or wrong low surrogate | JSON uses the `\uxxxx` format to describe Unicode characters. Code points above above 0xFFFF are split into two `\uxxxx` entries ("surrogate pairs"). This error indicates that the surrogate pair is incomplete or contains an invalid code point. -json.exception.parse_error.103 | parse error: code points above 0x10FFFF are invalid | Unicode supports code points up to 0x10FFFF. Code points above 0x10FFFF are invalid. -json.exception.parse_error.104 | parse error: JSON patch must be an array of objects | [RFC 6902](https://tools.ietf.org/html/rfc6902) requires a JSON Patch document to be a JSON document that represents an array of objects. -json.exception.parse_error.105 | parse error: operation must have string member 'op' | An operation of a JSON Patch document must contain exactly one "op" member, whose value indicates the operation to perform. Its value must be one of "add", "remove", "replace", "move", "copy", or "test"; other values are errors. -json.exception.parse_error.106 | parse error: array index '01' must not begin with '0' | An array index in a JSON Pointer ([RFC 6901](https://tools.ietf.org/html/rfc6901)) may be `0` or any number wihtout a leading `0`. -json.exception.parse_error.107 | parse error: JSON pointer must be empty or begin with '/' - was: 'foo' | A JSON Pointer must be a Unicode string containing a sequence of zero or more reference tokens, each prefixed by a `/` character. -json.exception.parse_error.108 | parse error: escape character '~' must be followed with '0' or '1' | In a JSON Pointer, only `~0` and `~1` are valid escape sequences. -json.exception.parse_error.109 | parse error: array index 'one' is not a number | A JSON Pointer array index must be a number. -json.exception.parse_error.110 | parse error at 1: cannot read 2 bytes from vector | When parsing CBOR or MessagePack, the byte vector ends before the complete value has been read. -json.exception.parse_error.111 | parse error: bad input stream | Parsing CBOR or MessagePack from an input stream where the [`badbit` or `failbit`](http://en.cppreference.com/w/cpp/io/ios_base/iostate) is set. -json.exception.parse_error.112 | parse error at 1: error reading CBOR; last byte: 0xf8 | Not all types of CBOR or MessagePack are supported. This exception occurs if an unsupported byte was read. -json.exception.parse_error.113 | parse error at 2: expected a CBOR string; last byte: 0x98 | While parsing a map key, a value that is not a string has been read. - -@since version 3.0.0 -*/ -class parse_error : public exception -{ - public: - /*! - @brief create a parse error exception - @param[in] id_ the id of the exception - @param[in] byte_ the byte index where the error occured (or 0 if - the position cannot be determined) - @param[in] what_arg_ the explanatory string - */ - parse_error(int id_, size_t byte_, const std::string& what_arg_) - : exception(id_, "parse_error", "parse error" + - (byte_ != 0 ? (" at " + std::to_string(byte_)) : "") + - ": " + what_arg_), - byte(byte_) - {} - - /*! - @brief byte index of the parse error - - The byte index of the last read character in the input file. - - @note For an input with n bytes, 1 is the index of the first character - and n+1 is the index of the terminating null byte or the end of - file. This also holds true when reading a byte vector (CBOR or - MessagePack). - */ - const size_t byte; -}; - -/*! -@brief exception indicating errors with iterators - -Exceptions have ids 2xx. - -name / id | example massage | description ------------------------------------ | --------------- | ------------------------- -json.exception.invalid_iterator.201 | iterators are not compatible | The iterators passed to constructor @ref basic_json(InputIT first, InputIT last) are not compatible, meaning they do not belong to the same container. Therefore, the range (@a first, @a last) is invalid. -json.exception.invalid_iterator.202 | iterator does not fit current value | In an erase or insert function, the passed iterator @a pos does not belong to the JSON value for which the function was called. It hence does not define a valid position for the deletion/insertion. -json.exception.invalid_iterator.203 | iterators do not fit current value | Either iterator passed to function @ref erase(IteratorType first, IteratorType last) does not belong to the JSON value from which values shall be erased. It hence does not define a valid range to delete values from. -json.exception.invalid_iterator.204 | iterators out of range | When an iterator range for a primitive type (number, boolean, or string) is passed to a constructor or an erase function, this range has to be exactly (@ref begin(), @ref end()), because this is the only way the single stored value is expressed. All other ranges are invalid. -json.exception.invalid_iterator.205 | iterator out of range | When an iterator for a primitive type (number, boolean, or string) is passed to an erase function, the iterator has to be the @ref begin() iterator, because it is the only way to address the stored value. All other iterators are invalid. -json.exception.invalid_iterator.206 | cannot construct with iterators from null | The iterators passed to constructor @ref basic_json(InputIT first, InputIT last) belong to a JSON null value and hence to not define a valid range. -json.exception.invalid_iterator.207 | cannot use key() for non-object iterators | The key() member function can only be used on iterators belonging to a JSON object, because other types do not have a concept of a key. -json.exception.invalid_iterator.208 | cannot use operator[] for object iterators | The operator[] to specify a concrete offset cannot be used on iterators belonging to a JSON object, because JSON objects are unordered. -json.exception.invalid_iterator.209 | cannot use offsets with object iterators | The offset operators (+, -, +=, -=) cannot be used on iterators belonging to a JSON object, because JSON objects are unordered. -json.exception.invalid_iterator.210 | iterators do not fit | The iterator range passed to the insert function are not compatible, meaning they do not belong to the same container. Therefore, the range (@a first, @a last) is invalid. -json.exception.invalid_iterator.211 | passed iterators may not belong to container | The iterator range passed to the insert function must not be a subrange of the container to insert to. -json.exception.invalid_iterator.212 | cannot compare iterators of different containers | When two iterators are compared, they must belong to the same container. -json.exception.invalid_iterator.213 | cannot compare order of object iterators | The order of object iterators cannot be compated, because JSON objects are unordered. -json.exception.invalid_iterator.214 | cannot get value | Cannot get value for iterator: Either the iterator belongs to a null value or it is an iterator to a primitive type (number, boolean, or string), but the iterator is different to @ref begin(). - -@since version 3.0.0 -*/ -class invalid_iterator : public exception -{ - public: - invalid_iterator(int id_, const std::string& what_arg_) - : exception(id_, "invalid_iterator", what_arg_) - {} -}; - -/*! -@brief exception indicating executing a member function with a wrong type - -Exceptions have ids 3xx. - -name / id | example massage | description ------------------------------ | --------------- | ------------------------- -json.exception.type_error.301 | cannot create object from initializer list | To create an object from an initializer list, the initializer list must consist only of a list of pairs whose first element is a string. When this constraint is violated, an array is created instead. -json.exception.type_error.302 | type must be object, but is array | During implicit or explicit value conversion, the JSON type must be compatible to the target type. For instance, a JSON string can only be converted into string types, but not into numbers or boolean types. -json.exception.type_error.303 | incompatible ReferenceType for get_ref, actual type is object | To retrieve a reference to a value stored in a @ref basic_json object with @ref get_ref, the type of the reference must match the value type. For instance, for a JSON array, the @a ReferenceType must be @ref array_t&. -json.exception.type_error.304 | cannot use at() with string | The @ref at() member functions can only be executed for certain JSON types. -json.exception.type_error.305 | cannot use operator[] with string | The @ref operator[] member functions can only be executed for certain JSON types. -json.exception.type_error.306 | cannot use value() with string | The @ref value() member functions can only be executed for certain JSON types. -json.exception.type_error.307 | cannot use erase() with string | The @ref erase() member functions can only be executed for certain JSON types. -json.exception.type_error.308 | cannot use push_back() with string | The @ref push_back() and @ref operator+= member functions can only be executed for certain JSON types. -json.exception.type_error.309 | cannot use insert() with | The @ref insert() member functions can only be executed for certain JSON types. -json.exception.type_error.310 | cannot use swap() with number | The @ref swap() member functions can only be executed for certain JSON types. -json.exception.type_error.311 | cannot use emplace_back() with string | The @ref emplace_back() member function can only be executed for certain JSON types. -json.exception.type_error.313 | invalid value to unflatten | The @ref unflatten function converts an object whose keys are JSON Pointers back into an arbitrary nested JSON value. The JSON Pointers must not overlap, because then the resulting value would not be well defined. -json.exception.type_error.314 | only objects can be unflattened | The @ref unflatten function only works for an object whose keys are JSON Pointers. -json.exception.type_error.315 | values in object must be primitive | The @ref unflatten function only works for an object whose keys are JSON Pointers and whose values are primitive. - -@since version 3.0.0 -*/ -class type_error : public exception -{ - public: - type_error(int id_, const std::string& what_arg_) - : exception(id_, "type_error", what_arg_) - {} -}; - -/*! -@brief exception indicating access out of the defined range - -Exceptions have ids 4xx. - -name / id | example massage | description -------------------------------- | --------------- | ------------------------- -json.exception.out_of_range.401 | array index 3 is out of range | The provided array index @a i is larger than @a size-1. -json.exception.out_of_range.402 | array index '-' (3) is out of range | The special array index `-` in a JSON Pointer never describes a valid element of the array, but the index past the end. That is, it can only be used to add elements at this position, but not to read it. -json.exception.out_of_range.403 | key 'foo' not found | The provided key was not found in the JSON object. -json.exception.out_of_range.404 | unresolved reference token 'foo' | A reference token in a JSON Pointer could not be resolved. -json.exception.out_of_range.405 | JSON pointer has no parent | The JSON Patch operations 'remove' and 'add' can not be applied to the root element of the JSON value. -json.exception.out_of_range.406 | number overflow parsing '10E1000' | A parsed number could not be stored as without changing it to NaN or INF. - -@since version 3.0.0 -*/ -class out_of_range : public exception -{ - public: - out_of_range(int id_, const std::string& what_arg_) - : exception(id_, "out_of_range", what_arg_) - {} -}; - -/*! -@brief exception indicating other errors - -Exceptions have ids 5xx. - -name / id | example massage | description ------------------------------- | --------------- | ------------------------- -json.exception.other_error.501 | unsuccessful: {"op":"test","path":"/baz", "value":"bar"} | A JSON Patch operation 'test' failed. The unsuccessful operation is also printed. - -@since version 3.0.0 -*/ -class other_error : public exception -{ - public: - other_error(int id_, const std::string& what_arg_) - : exception(id_, "other_error", what_arg_) - {} -}; - - - -/////////////////////////// -// JSON type enumeration // -/////////////////////////// - -/*! -@brief the JSON type enumeration - -This enumeration collects the different JSON types. It is internally used to -distinguish the stored values, and the functions @ref basic_json::is_null(), -@ref basic_json::is_object(), @ref basic_json::is_array(), -@ref basic_json::is_string(), @ref basic_json::is_boolean(), -@ref basic_json::is_number() (with @ref basic_json::is_number_integer(), -@ref basic_json::is_number_unsigned(), and @ref basic_json::is_number_float()), -@ref basic_json::is_discarded(), @ref basic_json::is_primitive(), and -@ref basic_json::is_structured() rely on it. - -@note There are three enumeration entries (number_integer, number_unsigned, and -number_float), because the library distinguishes these three types for numbers: -@ref basic_json::number_unsigned_t is used for unsigned integers, -@ref basic_json::number_integer_t is used for signed integers, and -@ref basic_json::number_float_t is used for floating-point numbers or to -approximate integers which do not fit in the limits of their respective type. - -@sa @ref basic_json::basic_json(const value_t value_type) -- create a JSON -value with the default value for a given type - -@since version 1.0.0 -*/ -enum class value_t : uint8_t -{ - null, ///< null value - object, ///< object (unordered set of name/value pairs) - array, ///< array (ordered collection of values) - string, ///< string value - boolean, ///< boolean value - number_integer, ///< number value (signed integer) - number_unsigned, ///< number value (unsigned integer) - number_float, ///< number value (floating-point) - discarded ///< discarded by the the parser callback function -}; - -/*! -@brief comparison operator for JSON types - -Returns an ordering that is similar to Python: -- order: null < boolean < number < object < array < string -- furthermore, each type is not smaller than itself - -@since version 1.0.0 -*/ -inline bool operator<(const value_t lhs, const value_t rhs) noexcept -{ - static constexpr std::array order = {{ - 0, // null - 3, // object - 4, // array - 5, // string - 1, // boolean - 2, // integer - 2, // unsigned - 2, // float - } - }; - - // discarded values are not comparable - if (lhs == value_t::discarded or rhs == value_t::discarded) - { - return false; - } - - return order[static_cast(lhs)] < - order[static_cast(rhs)]; -} - - -///////////// -// helpers // -///////////// - -// alias templates to reduce boilerplate -template -using enable_if_t = typename std::enable_if::type; - -template -using uncvref_t = typename std::remove_cv::type>::type; - -// taken from http://stackoverflow.com/a/26936864/266378 -template -using is_unscoped_enum = - std::integral_constant::value and - std::is_enum::value>; - -/* -Implementation of two C++17 constructs: conjunction, negation. This is needed -to avoid evaluating all the traits in a condition - -For example: not std::is_same::value and has_value_type::value -will not compile when T = void (on MSVC at least). Whereas -conjunction>, has_value_type>::value will -stop evaluating if negation<...>::value == false - -Please note that those constructs must be used with caution, since symbols can -become very long quickly (which can slow down compilation and cause MSVC -internal compiler errors). Only use it when you have to (see example ahead). -*/ -template struct conjunction : std::true_type {}; -template struct conjunction : B1 {}; -template -struct conjunction : std::conditional, B1>::type {}; - -template struct negation : std::integral_constant < bool, !B::value > {}; - -// dispatch utility (taken from ranges-v3) -template struct priority_tag : priority_tag < N - 1 > {}; -template<> struct priority_tag<0> {}; - - -////////////////// -// constructors // -////////////////// - -template struct external_constructor; - -template<> -struct external_constructor -{ - template - static void construct(BasicJsonType& j, typename BasicJsonType::boolean_t b) noexcept - { - j.m_type = value_t::boolean; - j.m_value = b; - j.assert_invariant(); - } -}; - -template<> -struct external_constructor -{ - template - static void construct(BasicJsonType& j, const typename BasicJsonType::string_t& s) - { - j.m_type = value_t::string; - j.m_value = s; - j.assert_invariant(); - } -}; - -template<> -struct external_constructor -{ - template - static void construct(BasicJsonType& j, typename BasicJsonType::number_float_t val) noexcept - { - j.m_type = value_t::number_float; - j.m_value = val; - j.assert_invariant(); - } -}; - -template<> -struct external_constructor -{ - template - static void construct(BasicJsonType& j, typename BasicJsonType::number_unsigned_t val) noexcept - { - j.m_type = value_t::number_unsigned; - j.m_value = val; - j.assert_invariant(); - } -}; - -template<> -struct external_constructor -{ - template - static void construct(BasicJsonType& j, typename BasicJsonType::number_integer_t val) noexcept - { - j.m_type = value_t::number_integer; - j.m_value = val; - j.assert_invariant(); - } -}; - -template<> -struct external_constructor -{ - template - static void construct(BasicJsonType& j, const typename BasicJsonType::array_t& arr) - { - j.m_type = value_t::array; - j.m_value = arr; - j.assert_invariant(); - } - - template::value, - int> = 0> - static void construct(BasicJsonType& j, const CompatibleArrayType& arr) - { - using std::begin; - using std::end; - j.m_type = value_t::array; - j.m_value.array = j.template create(begin(arr), end(arr)); - j.assert_invariant(); - } - - template - static void construct(BasicJsonType& j, const std::vector& arr) - { - j.m_type = value_t::array; - j.m_value = value_t::array; - j.m_value.array->reserve(arr.size()); - for (bool x : arr) - { - j.m_value.array->push_back(x); - } - j.assert_invariant(); - } -}; - -template<> -struct external_constructor -{ - template - static void construct(BasicJsonType& j, const typename BasicJsonType::object_t& obj) - { - j.m_type = value_t::object; - j.m_value = obj; - j.assert_invariant(); - } - - template::value, - int> = 0> - static void construct(BasicJsonType& j, const CompatibleObjectType& obj) - { - using std::begin; - using std::end; - - j.m_type = value_t::object; - j.m_value.object = j.template create(begin(obj), end(obj)); - j.assert_invariant(); - } -}; - - -//////////////////////// -// has_/is_ functions // -//////////////////////// - -/*! -@brief Helper to determine whether there's a key_type for T. - -This helper is used to tell associative containers apart from other containers -such as sequence containers. For instance, `std::map` passes the test as it -contains a `mapped_type`, whereas `std::vector` fails the test. - -@sa http://stackoverflow.com/a/7728728/266378 -@since version 1.0.0, overworked in version 2.0.6 -*/ -#define NLOHMANN_JSON_HAS_HELPER(type) \ - template struct has_##type { \ - private: \ - template \ - static int detect(U &&); \ - static void detect(...); \ - public: \ - static constexpr bool value = \ - std::is_integral()))>::value; \ - } - -NLOHMANN_JSON_HAS_HELPER(mapped_type); -NLOHMANN_JSON_HAS_HELPER(key_type); -NLOHMANN_JSON_HAS_HELPER(value_type); -NLOHMANN_JSON_HAS_HELPER(iterator); - -#undef NLOHMANN_JSON_HAS_HELPER - - -template -struct is_compatible_object_type_impl : std::false_type {}; - -template -struct is_compatible_object_type_impl -{ - static constexpr auto value = - std::is_constructible::value and - std::is_constructible::value; -}; - -template -struct is_compatible_object_type -{ - static auto constexpr value = is_compatible_object_type_impl < - conjunction>, - has_mapped_type, - has_key_type>::value, - typename BasicJsonType::object_t, CompatibleObjectType >::value; -}; - -template -struct is_basic_json_nested_type -{ - static auto constexpr value = std::is_same::value or - std::is_same::value or - std::is_same::value or - std::is_same::value or - std::is_same::value; -}; - -template -struct is_compatible_array_type -{ - static auto constexpr value = - conjunction>, - negation>, - negation>, - negation>, - has_value_type, - has_iterator>::value; -}; - -template -struct is_compatible_integer_type_impl : std::false_type {}; - -template -struct is_compatible_integer_type_impl -{ - // is there an assert somewhere on overflows? - using RealLimits = std::numeric_limits; - using CompatibleLimits = std::numeric_limits; - - static constexpr auto value = - std::is_constructible::value and - CompatibleLimits::is_integer and - RealLimits::is_signed == CompatibleLimits::is_signed; -}; - -template -struct is_compatible_integer_type -{ - static constexpr auto value = - is_compatible_integer_type_impl < - std::is_integral::value and - not std::is_same::value, - RealIntegerType, CompatibleNumberIntegerType > ::value; -}; - - -// trait checking if JSONSerializer::from_json(json const&, udt&) exists -template -struct has_from_json -{ - private: - // also check the return type of from_json - template::from_json( - std::declval(), std::declval()))>::value>> - static int detect(U&&); - static void detect(...); - - public: - static constexpr bool value = std::is_integral>()))>::value; -}; - -// This trait checks if JSONSerializer::from_json(json const&) exists -// this overload is used for non-default-constructible user-defined-types -template -struct has_non_default_from_json -{ - private: - template < - typename U, - typename = enable_if_t::from_json(std::declval()))>::value >> - static int detect(U&&); - static void detect(...); - - public: - static constexpr bool value = std::is_integral>()))>::value; -}; - -// This trait checks if BasicJsonType::json_serializer::to_json exists -template -struct has_to_json -{ - private: - template::to_json( - std::declval(), std::declval()))> - static int detect(U&&); - static void detect(...); - - public: - static constexpr bool value = std::is_integral>()))>::value; -}; - - -///////////// -// to_json // -///////////// - -template::value, int> = 0> -void to_json(BasicJsonType& j, T b) noexcept -{ - external_constructor::construct(j, b); -} - -template::value, int> = 0> -void to_json(BasicJsonType& j, const CompatibleString& s) -{ - external_constructor::construct(j, s); -} - -template::value, int> = 0> -void to_json(BasicJsonType& j, FloatType val) noexcept -{ - external_constructor::construct(j, static_cast(val)); -} - -template < - typename BasicJsonType, typename CompatibleNumberUnsignedType, - enable_if_t::value, int> = 0 > -void to_json(BasicJsonType& j, CompatibleNumberUnsignedType val) noexcept -{ - external_constructor::construct(j, static_cast(val)); -} - -template < - typename BasicJsonType, typename CompatibleNumberIntegerType, - enable_if_t::value, int> = 0 > -void to_json(BasicJsonType& j, CompatibleNumberIntegerType val) noexcept -{ - external_constructor::construct(j, static_cast(val)); -} - -template::value, int> = 0> -void to_json(BasicJsonType& j, UnscopedEnumType e) noexcept -{ - external_constructor::construct(j, e); -} - -template -void to_json(BasicJsonType& j, const std::vector& e) -{ - external_constructor::construct(j, e); -} - -template < - typename BasicJsonType, typename CompatibleArrayType, - enable_if_t < - is_compatible_array_type::value or - std::is_same::value, - int > = 0 > -void to_json(BasicJsonType& j, const CompatibleArrayType& arr) -{ - external_constructor::construct(j, arr); -} - -template < - typename BasicJsonType, typename CompatibleObjectType, - enable_if_t::value, - int> = 0 > -void to_json(BasicJsonType& j, const CompatibleObjectType& arr) -{ - external_constructor::construct(j, arr); -} - -template ::value, - int> = 0> -void to_json(BasicJsonType& j, T (&arr)[N]) -{ - external_constructor::construct(j, arr); -} - -/////////////// -// from_json // -/////////////// - -// overloads for basic_json template parameters -template::value and - not std::is_same::value, - int> = 0> -void get_arithmetic_value(const BasicJsonType& j, ArithmeticType& val) -{ - switch (static_cast(j)) - { - case value_t::number_unsigned: - { - val = static_cast( - *j.template get_ptr()); - break; - } - case value_t::number_integer: - { - val = static_cast( - *j.template get_ptr()); - break; - } - case value_t::number_float: - { - val = static_cast( - *j.template get_ptr()); - break; - } - default: - { - JSON_THROW(type_error(302, "type must be number, but is " + j.type_name())); - } - } -} - -template -void from_json(const BasicJsonType& j, typename BasicJsonType::boolean_t& b) -{ - if (not j.is_boolean()) - { - JSON_THROW(type_error(302, "type must be boolean, but is " + j.type_name())); - } - b = *j.template get_ptr(); -} - -template -void from_json(const BasicJsonType& j, typename BasicJsonType::string_t& s) -{ - if (not j.is_string()) - { - JSON_THROW(type_error(302, "type must be string, but is " + j.type_name())); - } - s = *j.template get_ptr(); -} - -template -void from_json(const BasicJsonType& j, typename BasicJsonType::number_float_t& val) -{ - get_arithmetic_value(j, val); -} - -template -void from_json(const BasicJsonType& j, typename BasicJsonType::number_unsigned_t& val) -{ - get_arithmetic_value(j, val); -} - -template -void from_json(const BasicJsonType& j, typename BasicJsonType::number_integer_t& val) -{ - get_arithmetic_value(j, val); -} - -template::value, int> = 0> -void from_json(const BasicJsonType& j, UnscopedEnumType& e) -{ - typename std::underlying_type::type val; - get_arithmetic_value(j, val); - e = static_cast(val); -} - -template -void from_json(const BasicJsonType& j, typename BasicJsonType::array_t& arr) -{ - if (not j.is_array()) - { - JSON_THROW(type_error(302, "type must be array, but is " + j.type_name())); - } - arr = *j.template get_ptr(); -} - -// forward_list doesn't have an insert method -template::value, int> = 0> -void from_json(const BasicJsonType& j, std::forward_list& l) -{ - if (not j.is_array()) - { - JSON_THROW(type_error(302, "type must be array, but is " + j.type_name())); - } - - for (auto it = j.rbegin(), end = j.rend(); it != end; ++it) - { - l.push_front(it->template get()); - } -} - -template -void from_json_array_impl(const BasicJsonType& j, CompatibleArrayType& arr, priority_tag<0>) -{ - using std::begin; - using std::end; - - std::transform(j.begin(), j.end(), - std::inserter(arr, end(arr)), [](const BasicJsonType & i) - { - // get() returns *this, this won't call a from_json - // method when value_type is BasicJsonType - return i.template get(); - }); -} - -template -auto from_json_array_impl(const BasicJsonType& j, CompatibleArrayType& arr, priority_tag<1>) --> decltype( - arr.reserve(std::declval()), - void()) -{ - using std::begin; - using std::end; - - arr.reserve(j.size()); - std::transform(j.begin(), j.end(), - std::inserter(arr, end(arr)), [](const BasicJsonType & i) - { - // get() returns *this, this won't call a from_json - // method when value_type is BasicJsonType - return i.template get(); - }); -} - -template::value and - std::is_convertible::value and - not std::is_same::value, int> = 0> -void from_json(const BasicJsonType& j, CompatibleArrayType& arr) -{ - if (not j.is_array()) - { - JSON_THROW(type_error(302, "type must be array, but is " + j.type_name())); - } - - from_json_array_impl(j, arr, priority_tag<1> {}); -} - -template::value, int> = 0> -void from_json(const BasicJsonType& j, CompatibleObjectType& obj) -{ - if (not j.is_object()) - { - JSON_THROW(type_error(302, "type must be object, but is " + j.type_name())); - } - - auto inner_object = j.template get_ptr(); - using std::begin; - using std::end; - // we could avoid the assignment, but this might require a for loop, which - // might be less efficient than the container constructor for some - // containers (would it?) - obj = CompatibleObjectType(begin(*inner_object), end(*inner_object)); -} - -// overload for arithmetic types, not chosen for basic_json template arguments -// (BooleanType, etc..); note: Is it really necessary to provide explicit -// overloads for boolean_t etc. in case of a custom BooleanType which is not -// an arithmetic type? -template::value and - not std::is_same::value and - not std::is_same::value and - not std::is_same::value and - not std::is_same::value, - int> = 0> -void from_json(const BasicJsonType& j, ArithmeticType& val) -{ - switch (static_cast(j)) - { - case value_t::number_unsigned: - { - val = static_cast(*j.template get_ptr()); - break; - } - case value_t::number_integer: - { - val = static_cast(*j.template get_ptr()); - break; - } - case value_t::number_float: - { - val = static_cast(*j.template get_ptr()); - break; - } - case value_t::boolean: - { - val = static_cast(*j.template get_ptr()); - break; - } - default: - { - JSON_THROW(type_error(302, "type must be number, but is " + j.type_name())); - } - } -} - -struct to_json_fn -{ - private: - template - auto call(BasicJsonType& j, T&& val, priority_tag<1>) const noexcept(noexcept(to_json(j, std::forward(val)))) - -> decltype(to_json(j, std::forward(val)), void()) - { - return to_json(j, std::forward(val)); - } - - template - void call(BasicJsonType&, T&&, priority_tag<0>) const noexcept - { - static_assert(sizeof(BasicJsonType) == 0, - "could not find to_json() method in T's namespace"); - } - - public: - template - void operator()(BasicJsonType& j, T&& val) const - noexcept(noexcept(std::declval().call(j, std::forward(val), priority_tag<1> {}))) - { - return call(j, std::forward(val), priority_tag<1> {}); - } -}; - -struct from_json_fn -{ - private: - template - auto call(const BasicJsonType& j, T& val, priority_tag<1>) const - noexcept(noexcept(from_json(j, val))) - -> decltype(from_json(j, val), void()) - { - return from_json(j, val); - } - - template - void call(const BasicJsonType&, T&, priority_tag<0>) const noexcept - { - static_assert(sizeof(BasicJsonType) == 0, - "could not find from_json() method in T's namespace"); - } - - public: - template - void operator()(const BasicJsonType& j, T& val) const - noexcept(noexcept(std::declval().call(j, val, priority_tag<1> {}))) - { - return call(j, val, priority_tag<1> {}); - } -}; - -// taken from ranges-v3 -template -struct static_const -{ - static constexpr T value{}; -}; - -template -constexpr T static_const::value; -} // namespace detail - - -/// namespace to hold default `to_json` / `from_json` functions -namespace -{ -constexpr const auto& to_json = detail::static_const::value; -constexpr const auto& from_json = detail::static_const::value; -} - - -/*! -@brief default JSONSerializer template argument - -This serializer ignores the template arguments and uses ADL -([argument-dependent lookup](http://en.cppreference.com/w/cpp/language/adl)) -for serialization. -*/ -template -struct adl_serializer -{ - /*! - @brief convert a JSON value to any value type - - This function is usually called by the `get()` function of the - @ref basic_json class (either explicit or via conversion operators). - - @param[in] j JSON value to read from - @param[in,out] val value to write to - */ - template - static void from_json(BasicJsonType&& j, ValueType& val) noexcept( - noexcept(::nlohmann::from_json(std::forward(j), val))) - { - ::nlohmann::from_json(std::forward(j), val); - } - - /*! - @brief convert any value type to a JSON value - - This function is usually called by the constructors of the @ref basic_json - class. - - @param[in,out] j JSON value to write to - @param[in] val value to read from - */ - template - static void to_json(BasicJsonType& j, ValueType&& val) noexcept( - noexcept(::nlohmann::to_json(j, std::forward(val)))) - { - ::nlohmann::to_json(j, std::forward(val)); - } -}; - - -/*! -@brief a class to store JSON values - -@tparam ObjectType type for JSON objects (`std::map` by default; will be used -in @ref object_t) -@tparam ArrayType type for JSON arrays (`std::vector` by default; will be used -in @ref array_t) -@tparam StringType type for JSON strings and object keys (`std::string` by -default; will be used in @ref string_t) -@tparam BooleanType type for JSON booleans (`bool` by default; will be used -in @ref boolean_t) -@tparam NumberIntegerType type for JSON integer numbers (`int64_t` by -default; will be used in @ref number_integer_t) -@tparam NumberUnsignedType type for JSON unsigned integer numbers (@c -`uint64_t` by default; will be used in @ref number_unsigned_t) -@tparam NumberFloatType type for JSON floating-point numbers (`double` by -default; will be used in @ref number_float_t) -@tparam AllocatorType type of the allocator to use (`std::allocator` by -default) -@tparam JSONSerializer the serializer to resolve internal calls to `to_json()` -and `from_json()` (@ref adl_serializer by default) - -@requirement The class satisfies the following concept requirements: -- Basic - - [DefaultConstructible](http://en.cppreference.com/w/cpp/concept/DefaultConstructible): - JSON values can be default constructed. The result will be a JSON null - value. - - [MoveConstructible](http://en.cppreference.com/w/cpp/concept/MoveConstructible): - A JSON value can be constructed from an rvalue argument. - - [CopyConstructible](http://en.cppreference.com/w/cpp/concept/CopyConstructible): - A JSON value can be copy-constructed from an lvalue expression. - - [MoveAssignable](http://en.cppreference.com/w/cpp/concept/MoveAssignable): - A JSON value van be assigned from an rvalue argument. - - [CopyAssignable](http://en.cppreference.com/w/cpp/concept/CopyAssignable): - A JSON value can be copy-assigned from an lvalue expression. - - [Destructible](http://en.cppreference.com/w/cpp/concept/Destructible): - JSON values can be destructed. -- Layout - - [StandardLayoutType](http://en.cppreference.com/w/cpp/concept/StandardLayoutType): - JSON values have - [standard layout](http://en.cppreference.com/w/cpp/language/data_members#Standard_layout): - All non-static data members are private and standard layout types, the - class has no virtual functions or (virtual) base classes. -- Library-wide - - [EqualityComparable](http://en.cppreference.com/w/cpp/concept/EqualityComparable): - JSON values can be compared with `==`, see @ref - operator==(const_reference,const_reference). - - [LessThanComparable](http://en.cppreference.com/w/cpp/concept/LessThanComparable): - JSON values can be compared with `<`, see @ref - operator<(const_reference,const_reference). - - [Swappable](http://en.cppreference.com/w/cpp/concept/Swappable): - Any JSON lvalue or rvalue of can be swapped with any lvalue or rvalue of - other compatible types, using unqualified function call @ref swap(). - - [NullablePointer](http://en.cppreference.com/w/cpp/concept/NullablePointer): - JSON values can be compared against `std::nullptr_t` objects which are used - to model the `null` value. -- Container - - [Container](http://en.cppreference.com/w/cpp/concept/Container): - JSON values can be used like STL containers and provide iterator access. - - [ReversibleContainer](http://en.cppreference.com/w/cpp/concept/ReversibleContainer); - JSON values can be used like STL containers and provide reverse iterator - access. - -@invariant The member variables @a m_value and @a m_type have the following -relationship: -- If `m_type == value_t::object`, then `m_value.object != nullptr`. -- If `m_type == value_t::array`, then `m_value.array != nullptr`. -- If `m_type == value_t::string`, then `m_value.string != nullptr`. -The invariants are checked by member function assert_invariant(). - -@internal -@note ObjectType trick from http://stackoverflow.com/a/9860911 -@endinternal - -@see [RFC 7159: The JavaScript Object Notation (JSON) Data Interchange -Format](http://rfc7159.net/rfc7159) - -@since version 1.0.0 - -@nosubgrouping -*/ -template < - template class ObjectType = std::map, - template class ArrayType = std::vector, - class StringType = std::string, - class BooleanType = bool, - class NumberIntegerType = std::int64_t, - class NumberUnsignedType = std::uint64_t, - class NumberFloatType = double, - template class AllocatorType = std::allocator, - template class JSONSerializer = adl_serializer - > -class basic_json -{ - private: - template friend struct detail::external_constructor; - /// workaround type for MSVC - using basic_json_t = basic_json; - - public: - using value_t = detail::value_t; - // forward declarations - template class iter_impl; - template class json_reverse_iterator; - class json_pointer; - template - using json_serializer = JSONSerializer; - - - //////////////// - // exceptions // - //////////////// - - /// @name exceptions - /// Classes to implement user-defined exceptions. - /// @{ - - /// @copydoc detail::exception - using exception = detail::exception; - /// @copydoc detail::parse_error - using parse_error = detail::parse_error; - /// @copydoc detail::invalid_iterator - using invalid_iterator = detail::invalid_iterator; - /// @copydoc detail::type_error - using type_error = detail::type_error; - /// @copydoc detail::out_of_range - using out_of_range = detail::out_of_range; - /// @copydoc detail::other_error - using other_error = detail::other_error; - - /// @} - - - ///////////////////// - // container types // - ///////////////////// - - /// @name container types - /// The canonic container types to use @ref basic_json like any other STL - /// container. - /// @{ - - /// the type of elements in a basic_json container - using value_type = basic_json; - - /// the type of an element reference - using reference = value_type&; - /// the type of an element const reference - using const_reference = const value_type&; - - /// a type to represent differences between iterators - using difference_type = std::ptrdiff_t; - /// a type to represent container sizes - using size_type = std::size_t; - - /// the allocator type - using allocator_type = AllocatorType; - - /// the type of an element pointer - using pointer = typename std::allocator_traits::pointer; - /// the type of an element const pointer - using const_pointer = typename std::allocator_traits::const_pointer; - - /// an iterator for a basic_json container - using iterator = iter_impl; - /// a const iterator for a basic_json container - using const_iterator = iter_impl; - /// a reverse iterator for a basic_json container - using reverse_iterator = json_reverse_iterator; - /// a const reverse iterator for a basic_json container - using const_reverse_iterator = json_reverse_iterator; - - /// @} - - - /*! - @brief returns the allocator associated with the container - */ - static allocator_type get_allocator() - { - return allocator_type(); - } - - /*! - @brief returns version information on the library - - This function returns a JSON object with information about the library, - including the version number and information on the platform and compiler. - - @return JSON object holding version information - key | description - ----------- | --------------- - `compiler` | Information on the used compiler. It is an object with the following keys: `c++` (the used C++ standard), `family` (the compiler family; possible values are `clang`, `icc`, `gcc`, `ilecpp`, `msvc`, `pgcpp`, `sunpro`, and `unknown`), and `version` (the compiler version). - `copyright` | The copyright line for the library as string. - `name` | The name of the library as string. - `platform` | The used platform as string. Possible values are `win32`, `linux`, `apple`, `unix`, and `unknown`. - `url` | The URL of the project as string. - `version` | The version of the library. It is an object with the following keys: `major`, `minor`, and `patch` as defined by [Semantic Versioning](http://semver.org), and `string` (the version string). - - @liveexample{The following code shows an example output of the `meta()` - function.,meta} - - @complexity Constant. - - @since 2.1.0 - */ - static basic_json meta() - { - basic_json result; - - result["copyright"] = "(C) 2013-2017 Niels Lohmann"; - result["name"] = "JSON for Modern C++"; - result["url"] = "https://github.com/nlohmann/json"; - result["version"] = - { - {"string", "2.1.1"}, {"major", 2}, {"minor", 1}, {"patch", 1} - }; - -#ifdef _WIN32 - result["platform"] = "win32"; -#elif defined __linux__ - result["platform"] = "linux"; -#elif defined __APPLE__ - result["platform"] = "apple"; -#elif defined __unix__ - result["platform"] = "unix"; -#else - result["platform"] = "unknown"; -#endif - -#if defined(__clang__) - result["compiler"] = {{"family", "clang"}, {"version", __clang_version__}}; -#elif defined(__ICC) || defined(__INTEL_COMPILER) - result["compiler"] = {{"family", "icc"}, {"version", __INTEL_COMPILER}}; -#elif defined(__GNUC__) || defined(__GNUG__) - result["compiler"] = {{"family", "gcc"}, {"version", std::to_string(__GNUC__) + "." + std::to_string(__GNUC_MINOR__) + "." + std::to_string(__GNUC_PATCHLEVEL__)}}; -#elif defined(__HP_cc) || defined(__HP_aCC) - result["compiler"] = "hp" -#elif defined(__IBMCPP__) - result["compiler"] = {{"family", "ilecpp"}, {"version", __IBMCPP__}}; -#elif defined(_MSC_VER) - result["compiler"] = {{"family", "msvc"}, {"version", _MSC_VER}}; -#elif defined(__PGI) - result["compiler"] = {{"family", "pgcpp"}, {"version", __PGI}}; -#elif defined(__SUNPRO_CC) - result["compiler"] = {{"family", "sunpro"}, {"version", __SUNPRO_CC}}; -#else - result["compiler"] = {{"family", "unknown"}, {"version", "unknown"}}; -#endif - -#ifdef __cplusplus - result["compiler"]["c++"] = std::to_string(__cplusplus); -#else - result["compiler"]["c++"] = "unknown"; -#endif - return result; - } - - - /////////////////////////// - // JSON value data types // - /////////////////////////// - - /// @name JSON value data types - /// The data types to store a JSON value. These types are derived from - /// the template arguments passed to class @ref basic_json. - /// @{ - - /*! - @brief a type for an object - - [RFC 7159](http://rfc7159.net/rfc7159) describes JSON objects as follows: - > An object is an unordered collection of zero or more name/value pairs, - > where a name is a string and a value is a string, number, boolean, null, - > object, or array. - - To store objects in C++, a type is defined by the template parameters - described below. - - @tparam ObjectType the container to store objects (e.g., `std::map` or - `std::unordered_map`) - @tparam StringType the type of the keys or names (e.g., `std::string`). - The comparison function `std::less` is used to order elements - inside the container. - @tparam AllocatorType the allocator to use for objects (e.g., - `std::allocator`) - - #### Default type - - With the default values for @a ObjectType (`std::map`), @a StringType - (`std::string`), and @a AllocatorType (`std::allocator`), the default - value for @a object_t is: - - @code {.cpp} - std::map< - std::string, // key_type - basic_json, // value_type - std::less, // key_compare - std::allocator> // allocator_type - > - @endcode - - #### Behavior - - The choice of @a object_t influences the behavior of the JSON class. With - the default type, objects have the following behavior: - - - When all names are unique, objects will be interoperable in the sense - that all software implementations receiving that object will agree on - the name-value mappings. - - When the names within an object are not unique, later stored name/value - pairs overwrite previously stored name/value pairs, leaving the used - names unique. For instance, `{"key": 1}` and `{"key": 2, "key": 1}` will - be treated as equal and both stored as `{"key": 1}`. - - Internally, name/value pairs are stored in lexicographical order of the - names. Objects will also be serialized (see @ref dump) in this order. - For instance, `{"b": 1, "a": 2}` and `{"a": 2, "b": 1}` will be stored - and serialized as `{"a": 2, "b": 1}`. - - When comparing objects, the order of the name/value pairs is irrelevant. - This makes objects interoperable in the sense that they will not be - affected by these differences. For instance, `{"b": 1, "a": 2}` and - `{"a": 2, "b": 1}` will be treated as equal. - - #### Limits - - [RFC 7159](http://rfc7159.net/rfc7159) specifies: - > An implementation may set limits on the maximum depth of nesting. - - In this class, the object's limit of nesting is not constraint explicitly. - However, a maximum depth of nesting may be introduced by the compiler or - runtime environment. A theoretical limit can be queried by calling the - @ref max_size function of a JSON object. - - #### Storage - - Objects are stored as pointers in a @ref basic_json type. That is, for any - access to object values, a pointer of type `object_t*` must be - dereferenced. - - @sa @ref array_t -- type for an array value - - @since version 1.0.0 - - @note The order name/value pairs are added to the object is *not* - preserved by the library. Therefore, iterating an object may return - name/value pairs in a different order than they were originally stored. In - fact, keys will be traversed in alphabetical order as `std::map` with - `std::less` is used by default. Please note this behavior conforms to [RFC - 7159](http://rfc7159.net/rfc7159), because any order implements the - specified "unordered" nature of JSON objects. - */ - using object_t = ObjectType, - AllocatorType>>; - - /*! - @brief a type for an array - - [RFC 7159](http://rfc7159.net/rfc7159) describes JSON arrays as follows: - > An array is an ordered sequence of zero or more values. - - To store objects in C++, a type is defined by the template parameters - explained below. - - @tparam ArrayType container type to store arrays (e.g., `std::vector` or - `std::list`) - @tparam AllocatorType allocator to use for arrays (e.g., `std::allocator`) - - #### Default type - - With the default values for @a ArrayType (`std::vector`) and @a - AllocatorType (`std::allocator`), the default value for @a array_t is: - - @code {.cpp} - std::vector< - basic_json, // value_type - std::allocator // allocator_type - > - @endcode - - #### Limits - - [RFC 7159](http://rfc7159.net/rfc7159) specifies: - > An implementation may set limits on the maximum depth of nesting. - - In this class, the array's limit of nesting is not constraint explicitly. - However, a maximum depth of nesting may be introduced by the compiler or - runtime environment. A theoretical limit can be queried by calling the - @ref max_size function of a JSON array. - - #### Storage - - Arrays are stored as pointers in a @ref basic_json type. That is, for any - access to array values, a pointer of type `array_t*` must be dereferenced. - - @sa @ref object_t -- type for an object value - - @since version 1.0.0 - */ - using array_t = ArrayType>; - - /*! - @brief a type for a string - - [RFC 7159](http://rfc7159.net/rfc7159) describes JSON strings as follows: - > A string is a sequence of zero or more Unicode characters. - - To store objects in C++, a type is defined by the template parameter - described below. Unicode values are split by the JSON class into - byte-sized characters during deserialization. - - @tparam StringType the container to store strings (e.g., `std::string`). - Note this container is used for keys/names in objects, see @ref object_t. - - #### Default type - - With the default values for @a StringType (`std::string`), the default - value for @a string_t is: - - @code {.cpp} - std::string - @endcode - - #### Encoding - - Strings are stored in UTF-8 encoding. Therefore, functions like - `std::string::size()` or `std::string::length()` return the number of - bytes in the string rather than the number of characters or glyphs. - - #### String comparison - - [RFC 7159](http://rfc7159.net/rfc7159) states: - > Software implementations are typically required to test names of object - > members for equality. Implementations that transform the textual - > representation into sequences of Unicode code units and then perform the - > comparison numerically, code unit by code unit, are interoperable in the - > sense that implementations will agree in all cases on equality or - > inequality of two strings. For example, implementations that compare - > strings with escaped characters unconverted may incorrectly find that - > `"a\\b"` and `"a\u005Cb"` are not equal. - - This implementation is interoperable as it does compare strings code unit - by code unit. - - #### Storage - - String values are stored as pointers in a @ref basic_json type. That is, - for any access to string values, a pointer of type `string_t*` must be - dereferenced. - - @since version 1.0.0 - */ - using string_t = StringType; - - /*! - @brief a type for a boolean - - [RFC 7159](http://rfc7159.net/rfc7159) implicitly describes a boolean as a - type which differentiates the two literals `true` and `false`. - - To store objects in C++, a type is defined by the template parameter @a - BooleanType which chooses the type to use. - - #### Default type - - With the default values for @a BooleanType (`bool`), the default value for - @a boolean_t is: - - @code {.cpp} - bool - @endcode - - #### Storage - - Boolean values are stored directly inside a @ref basic_json type. - - @since version 1.0.0 - */ - using boolean_t = BooleanType; - - /*! - @brief a type for a number (integer) - - [RFC 7159](http://rfc7159.net/rfc7159) describes numbers as follows: - > The representation of numbers is similar to that used in most - > programming languages. A number is represented in base 10 using decimal - > digits. It contains an integer component that may be prefixed with an - > optional minus sign, which may be followed by a fraction part and/or an - > exponent part. Leading zeros are not allowed. (...) Numeric values that - > cannot be represented in the grammar below (such as Infinity and NaN) - > are not permitted. - - This description includes both integer and floating-point numbers. - However, C++ allows more precise storage if it is known whether the number - is a signed integer, an unsigned integer or a floating-point number. - Therefore, three different types, @ref number_integer_t, @ref - number_unsigned_t and @ref number_float_t are used. - - To store integer numbers in C++, a type is defined by the template - parameter @a NumberIntegerType which chooses the type to use. - - #### Default type - - With the default values for @a NumberIntegerType (`int64_t`), the default - value for @a number_integer_t is: - - @code {.cpp} - int64_t - @endcode - - #### Default behavior - - - The restrictions about leading zeros is not enforced in C++. Instead, - leading zeros in integer literals lead to an interpretation as octal - number. Internally, the value will be stored as decimal number. For - instance, the C++ integer literal `010` will be serialized to `8`. - During deserialization, leading zeros yield an error. - - Not-a-number (NaN) values will be serialized to `null`. - - #### Limits - - [RFC 7159](http://rfc7159.net/rfc7159) specifies: - > An implementation may set limits on the range and precision of numbers. - - When the default type is used, the maximal integer number that can be - stored is `9223372036854775807` (INT64_MAX) and the minimal integer number - that can be stored is `-9223372036854775808` (INT64_MIN). Integer numbers - that are out of range will yield over/underflow when used in a - constructor. During deserialization, too large or small integer numbers - will be automatically be stored as @ref number_unsigned_t or @ref - number_float_t. - - [RFC 7159](http://rfc7159.net/rfc7159) further states: - > Note that when such software is used, numbers that are integers and are - > in the range \f$[-2^{53}+1, 2^{53}-1]\f$ are interoperable in the sense - > that implementations will agree exactly on their numeric values. - - As this range is a subrange of the exactly supported range [INT64_MIN, - INT64_MAX], this class's integer type is interoperable. - - #### Storage - - Integer number values are stored directly inside a @ref basic_json type. - - @sa @ref number_float_t -- type for number values (floating-point) - - @sa @ref number_unsigned_t -- type for number values (unsigned integer) - - @since version 1.0.0 - */ - using number_integer_t = NumberIntegerType; - - /*! - @brief a type for a number (unsigned) - - [RFC 7159](http://rfc7159.net/rfc7159) describes numbers as follows: - > The representation of numbers is similar to that used in most - > programming languages. A number is represented in base 10 using decimal - > digits. It contains an integer component that may be prefixed with an - > optional minus sign, which may be followed by a fraction part and/or an - > exponent part. Leading zeros are not allowed. (...) Numeric values that - > cannot be represented in the grammar below (such as Infinity and NaN) - > are not permitted. - - This description includes both integer and floating-point numbers. - However, C++ allows more precise storage if it is known whether the number - is a signed integer, an unsigned integer or a floating-point number. - Therefore, three different types, @ref number_integer_t, @ref - number_unsigned_t and @ref number_float_t are used. - - To store unsigned integer numbers in C++, a type is defined by the - template parameter @a NumberUnsignedType which chooses the type to use. - - #### Default type - - With the default values for @a NumberUnsignedType (`uint64_t`), the - default value for @a number_unsigned_t is: - - @code {.cpp} - uint64_t - @endcode - - #### Default behavior - - - The restrictions about leading zeros is not enforced in C++. Instead, - leading zeros in integer literals lead to an interpretation as octal - number. Internally, the value will be stored as decimal number. For - instance, the C++ integer literal `010` will be serialized to `8`. - During deserialization, leading zeros yield an error. - - Not-a-number (NaN) values will be serialized to `null`. - - #### Limits - - [RFC 7159](http://rfc7159.net/rfc7159) specifies: - > An implementation may set limits on the range and precision of numbers. - - When the default type is used, the maximal integer number that can be - stored is `18446744073709551615` (UINT64_MAX) and the minimal integer - number that can be stored is `0`. Integer numbers that are out of range - will yield over/underflow when used in a constructor. During - deserialization, too large or small integer numbers will be automatically - be stored as @ref number_integer_t or @ref number_float_t. - - [RFC 7159](http://rfc7159.net/rfc7159) further states: - > Note that when such software is used, numbers that are integers and are - > in the range \f$[-2^{53}+1, 2^{53}-1]\f$ are interoperable in the sense - > that implementations will agree exactly on their numeric values. - - As this range is a subrange (when considered in conjunction with the - number_integer_t type) of the exactly supported range [0, UINT64_MAX], - this class's integer type is interoperable. - - #### Storage - - Integer number values are stored directly inside a @ref basic_json type. - - @sa @ref number_float_t -- type for number values (floating-point) - @sa @ref number_integer_t -- type for number values (integer) - - @since version 2.0.0 - */ - using number_unsigned_t = NumberUnsignedType; - - /*! - @brief a type for a number (floating-point) - - [RFC 7159](http://rfc7159.net/rfc7159) describes numbers as follows: - > The representation of numbers is similar to that used in most - > programming languages. A number is represented in base 10 using decimal - > digits. It contains an integer component that may be prefixed with an - > optional minus sign, which may be followed by a fraction part and/or an - > exponent part. Leading zeros are not allowed. (...) Numeric values that - > cannot be represented in the grammar below (such as Infinity and NaN) - > are not permitted. - - This description includes both integer and floating-point numbers. - However, C++ allows more precise storage if it is known whether the number - is a signed integer, an unsigned integer or a floating-point number. - Therefore, three different types, @ref number_integer_t, @ref - number_unsigned_t and @ref number_float_t are used. - - To store floating-point numbers in C++, a type is defined by the template - parameter @a NumberFloatType which chooses the type to use. - - #### Default type - - With the default values for @a NumberFloatType (`double`), the default - value for @a number_float_t is: - - @code {.cpp} - double - @endcode - - #### Default behavior - - - The restrictions about leading zeros is not enforced in C++. Instead, - leading zeros in floating-point literals will be ignored. Internally, - the value will be stored as decimal number. For instance, the C++ - floating-point literal `01.2` will be serialized to `1.2`. During - deserialization, leading zeros yield an error. - - Not-a-number (NaN) values will be serialized to `null`. - - #### Limits - - [RFC 7159](http://rfc7159.net/rfc7159) states: - > This specification allows implementations to set limits on the range and - > precision of numbers accepted. Since software that implements IEEE - > 754-2008 binary64 (double precision) numbers is generally available and - > widely used, good interoperability can be achieved by implementations - > that expect no more precision or range than these provide, in the sense - > that implementations will approximate JSON numbers within the expected - > precision. - - This implementation does exactly follow this approach, as it uses double - precision floating-point numbers. Note values smaller than - `-1.79769313486232e+308` and values greater than `1.79769313486232e+308` - will be stored as NaN internally and be serialized to `null`. - - #### Storage - - Floating-point number values are stored directly inside a @ref basic_json - type. - - @sa @ref number_integer_t -- type for number values (integer) - - @sa @ref number_unsigned_t -- type for number values (unsigned integer) - - @since version 1.0.0 - */ - using number_float_t = NumberFloatType; - - /// @} - - private: - - /// helper for exception-safe object creation - template - static T* create(Args&& ... args) - { - AllocatorType alloc; - auto deleter = [&](T * object) - { - alloc.deallocate(object, 1); - }; - std::unique_ptr object(alloc.allocate(1), deleter); - alloc.construct(object.get(), std::forward(args)...); - assert(object != nullptr); - return object.release(); - } - - //////////////////////// - // JSON value storage // - //////////////////////// - - /*! - @brief a JSON value - - The actual storage for a JSON value of the @ref basic_json class. This - union combines the different storage types for the JSON value types - defined in @ref value_t. - - JSON type | value_t type | used type - --------- | --------------- | ------------------------ - object | object | pointer to @ref object_t - array | array | pointer to @ref array_t - string | string | pointer to @ref string_t - boolean | boolean | @ref boolean_t - number | number_integer | @ref number_integer_t - number | number_unsigned | @ref number_unsigned_t - number | number_float | @ref number_float_t - null | null | *no value is stored* - - @note Variable-length types (objects, arrays, and strings) are stored as - pointers. The size of the union should not exceed 64 bits if the default - value types are used. - - @since version 1.0.0 - */ - union json_value - { - /// object (stored with pointer to save storage) - object_t* object; - /// array (stored with pointer to save storage) - array_t* array; - /// string (stored with pointer to save storage) - string_t* string; - /// boolean - boolean_t boolean; - /// number (integer) - number_integer_t number_integer; - /// number (unsigned integer) - number_unsigned_t number_unsigned; - /// number (floating-point) - number_float_t number_float; - - /// default constructor (for null values) - json_value() = default; - /// constructor for booleans - json_value(boolean_t v) noexcept : boolean(v) {} - /// constructor for numbers (integer) - json_value(number_integer_t v) noexcept : number_integer(v) {} - /// constructor for numbers (unsigned) - json_value(number_unsigned_t v) noexcept : number_unsigned(v) {} - /// constructor for numbers (floating-point) - json_value(number_float_t v) noexcept : number_float(v) {} - /// constructor for empty values of a given type - json_value(value_t t) - { - switch (t) - { - case value_t::object: - { - object = create(); - break; - } - - case value_t::array: - { - array = create(); - break; - } - - case value_t::string: - { - string = create(""); - break; - } - - case value_t::boolean: - { - boolean = boolean_t(false); - break; - } - - case value_t::number_integer: - { - number_integer = number_integer_t(0); - break; - } - - case value_t::number_unsigned: - { - number_unsigned = number_unsigned_t(0); - break; - } - - case value_t::number_float: - { - number_float = number_float_t(0.0); - break; - } - - case value_t::null: - { - break; - } - - default: - { - if (t == value_t::null) - { - JSON_THROW(other_error(500, "961c151d2e87f2686a955a9be24d316f1362bf21 2.1.1")); // LCOV_EXCL_LINE - } - break; - } - } - } - - /// constructor for strings - json_value(const string_t& value) - { - string = create(value); - } - - /// constructor for objects - json_value(const object_t& value) - { - object = create(value); - } - - /// constructor for arrays - json_value(const array_t& value) - { - array = create(value); - } - }; - - /*! - @brief checks the class invariants - - This function asserts the class invariants. It needs to be called at the - end of every constructor to make sure that created objects respect the - invariant. Furthermore, it has to be called each time the type of a JSON - value is changed, because the invariant expresses a relationship between - @a m_type and @a m_value. - */ - void assert_invariant() const - { - assert(m_type != value_t::object or m_value.object != nullptr); - assert(m_type != value_t::array or m_value.array != nullptr); - assert(m_type != value_t::string or m_value.string != nullptr); - } - - public: - ////////////////////////// - // JSON parser callback // - ////////////////////////// - - /*! - @brief JSON callback events - - This enumeration lists the parser events that can trigger calling a - callback function of type @ref parser_callback_t during parsing. - - @image html callback_events.png "Example when certain parse events are triggered" - - @since version 1.0.0 - */ - enum class parse_event_t : uint8_t - { - /// the parser read `{` and started to process a JSON object - object_start, - /// the parser read `}` and finished processing a JSON object - object_end, - /// the parser read `[` and started to process a JSON array - array_start, - /// the parser read `]` and finished processing a JSON array - array_end, - /// the parser read a key of a value in an object - key, - /// the parser finished reading a JSON value - value - }; - - /*! - @brief per-element parser callback type - - With a parser callback function, the result of parsing a JSON text can be - influenced. When passed to @ref parse(std::istream&, const - parser_callback_t) or @ref parse(const CharT, const parser_callback_t), - it is called on certain events (passed as @ref parse_event_t via parameter - @a event) with a set recursion depth @a depth and context JSON value - @a parsed. The return value of the callback function is a boolean - indicating whether the element that emitted the callback shall be kept or - not. - - We distinguish six scenarios (determined by the event type) in which the - callback function can be called. The following table describes the values - of the parameters @a depth, @a event, and @a parsed. - - parameter @a event | description | parameter @a depth | parameter @a parsed - ------------------ | ----------- | ------------------ | ------------------- - parse_event_t::object_start | the parser read `{` and started to process a JSON object | depth of the parent of the JSON object | a JSON value with type discarded - parse_event_t::key | the parser read a key of a value in an object | depth of the currently parsed JSON object | a JSON string containing the key - parse_event_t::object_end | the parser read `}` and finished processing a JSON object | depth of the parent of the JSON object | the parsed JSON object - parse_event_t::array_start | the parser read `[` and started to process a JSON array | depth of the parent of the JSON array | a JSON value with type discarded - parse_event_t::array_end | the parser read `]` and finished processing a JSON array | depth of the parent of the JSON array | the parsed JSON array - parse_event_t::value | the parser finished reading a JSON value | depth of the value | the parsed JSON value - - @image html callback_events.png "Example when certain parse events are triggered" - - Discarding a value (i.e., returning `false`) has different effects - depending on the context in which function was called: - - - Discarded values in structured types are skipped. That is, the parser - will behave as if the discarded value was never read. - - In case a value outside a structured type is skipped, it is replaced - with `null`. This case happens if the top-level element is skipped. - - @param[in] depth the depth of the recursion during parsing - - @param[in] event an event of type parse_event_t indicating the context in - the callback function has been called - - @param[in,out] parsed the current intermediate parse result; note that - writing to this value has no effect for parse_event_t::key events - - @return Whether the JSON value which called the function during parsing - should be kept (`true`) or not (`false`). In the latter case, it is either - skipped completely or replaced by an empty discarded object. - - @sa @ref parse(std::istream&, parser_callback_t) or - @ref parse(const CharT, const parser_callback_t) for examples - - @since version 1.0.0 - */ - using parser_callback_t = std::function; - - - ////////////////// - // constructors // - ////////////////// - - /// @name constructors and destructors - /// Constructors of class @ref basic_json, copy/move constructor, copy - /// assignment, static functions creating objects, and the destructor. - /// @{ - - /*! - @brief create an empty value with a given type - - Create an empty JSON value with a given type. The value will be default - initialized with an empty value which depends on the type: - - Value type | initial value - ----------- | ------------- - null | `null` - boolean | `false` - string | `""` - number | `0` - object | `{}` - array | `[]` - - @param[in] value_type the type of the value to create - - @complexity Constant. - - @liveexample{The following code shows the constructor for different @ref - value_t values,basic_json__value_t} - - @since version 1.0.0 - */ - basic_json(const value_t value_type) - : m_type(value_type), m_value(value_type) - { - assert_invariant(); - } - - /*! - @brief create a null object - - Create a `null` JSON value. It either takes a null pointer as parameter - (explicitly creating `null`) or no parameter (implicitly creating `null`). - The passed null pointer itself is not read -- it is only used to choose - the right constructor. - - @complexity Constant. - - @exceptionsafety No-throw guarantee: this constructor never throws - exceptions. - - @liveexample{The following code shows the constructor with and without a - null pointer parameter.,basic_json__nullptr_t} - - @since version 1.0.0 - */ - basic_json(std::nullptr_t = nullptr) noexcept - : basic_json(value_t::null) - { - assert_invariant(); - } - - /*! - @brief create a JSON value - - This is a "catch all" constructor for all compatible JSON types; that is, - types for which a `to_json()` method exsits. The constructor forwards the - parameter @a val to that method (to `json_serializer::to_json` method - with `U = uncvref_t`, to be exact). - - Template type @a CompatibleType includes, but is not limited to, the - following types: - - **arrays**: @ref array_t and all kinds of compatible containers such as - `std::vector`, `std::deque`, `std::list`, `std::forward_list`, - `std::array`, `std::set`, `std::unordered_set`, `std::multiset`, and - `unordered_multiset` with a `value_type` from which a @ref basic_json - value can be constructed. - - **objects**: @ref object_t and all kinds of compatible associative - containers such as `std::map`, `std::unordered_map`, `std::multimap`, - and `std::unordered_multimap` with a `key_type` compatible to - @ref string_t and a `value_type` from which a @ref basic_json value can - be constructed. - - **strings**: @ref string_t, string literals, and all compatible string - containers can be used. - - **numbers**: @ref number_integer_t, @ref number_unsigned_t, - @ref number_float_t, and all convertible number types such as `int`, - `size_t`, `int64_t`, `float` or `double` can be used. - - **boolean**: @ref boolean_t / `bool` can be used. - - See the examples below. - - @tparam CompatibleType a type such that: - - @a CompatibleType is not derived from `std::istream`, - - @a CompatibleType is not @ref basic_json (to avoid hijacking copy/move - constructors), - - @a CompatibleType is not a @ref basic_json nested type (e.g., - @ref json_pointer, @ref iterator, etc ...) - - @ref @ref json_serializer has a - `to_json(basic_json_t&, CompatibleType&&)` method - - @tparam U = `uncvref_t` - - @param[in] val the value to be forwarded - - @complexity Usually linear in the size of the passed @a val, also - depending on the implementation of the called `to_json()` - method. - - @throw what `json_serializer::to_json()` throws - - @liveexample{The following code shows the constructor with several - compatible types.,basic_json__CompatibleType} - - @since version 2.1.0 - */ - template, - detail::enable_if_t::value and - not std::is_same::value and - not detail::is_basic_json_nested_type< - basic_json_t, U>::value and - detail::has_to_json::value, - int> = 0> - basic_json(CompatibleType && val) noexcept(noexcept(JSONSerializer::to_json( - std::declval(), std::forward(val)))) - { - JSONSerializer::to_json(*this, std::forward(val)); - assert_invariant(); - } - - /*! - @brief create a container (array or object) from an initializer list - - Creates a JSON value of type array or object from the passed initializer - list @a init. In case @a type_deduction is `true` (default), the type of - the JSON value to be created is deducted from the initializer list @a init - according to the following rules: - - 1. If the list is empty, an empty JSON object value `{}` is created. - 2. If the list consists of pairs whose first element is a string, a JSON - object value is created where the first elements of the pairs are - treated as keys and the second elements are as values. - 3. In all other cases, an array is created. - - The rules aim to create the best fit between a C++ initializer list and - JSON values. The rationale is as follows: - - 1. The empty initializer list is written as `{}` which is exactly an empty - JSON object. - 2. C++ has now way of describing mapped types other than to list a list of - pairs. As JSON requires that keys must be of type string, rule 2 is the - weakest constraint one can pose on initializer lists to interpret them - as an object. - 3. In all other cases, the initializer list could not be interpreted as - JSON object type, so interpreting it as JSON array type is safe. - - With the rules described above, the following JSON values cannot be - expressed by an initializer list: - - - the empty array (`[]`): use @ref array(std::initializer_list) - with an empty initializer list in this case - - arrays whose elements satisfy rule 2: use @ref - array(std::initializer_list) with the same initializer list - in this case - - @note When used without parentheses around an empty initializer list, @ref - basic_json() is called instead of this function, yielding the JSON null - value. - - @param[in] init initializer list with JSON values - - @param[in] type_deduction internal parameter; when set to `true`, the type - of the JSON value is deducted from the initializer list @a init; when set - to `false`, the type provided via @a manual_type is forced. This mode is - used by the functions @ref array(std::initializer_list) and - @ref object(std::initializer_list). - - @param[in] manual_type internal parameter; when @a type_deduction is set - to `false`, the created JSON value will use the provided type (only @ref - value_t::array and @ref value_t::object are valid); when @a type_deduction - is set to `true`, this parameter has no effect - - @throw type_error.301 if @a type_deduction is `false`, @a manual_type is - `value_t::object`, but @a init contains an element which is not a pair - whose first element is a string. In this case, the constructor could not - create an object. If @a type_deduction would have be `true`, an array - would have been created. See @ref object(std::initializer_list) - for an example. - - @complexity Linear in the size of the initializer list @a init. - - @liveexample{The example below shows how JSON values are created from - initializer lists.,basic_json__list_init_t} - - @sa @ref array(std::initializer_list) -- create a JSON array - value from an initializer list - @sa @ref object(std::initializer_list) -- create a JSON object - value from an initializer list - - @since version 1.0.0 - */ - basic_json(std::initializer_list init, - bool type_deduction = true, - value_t manual_type = value_t::array) - { - // check if each element is an array with two elements whose first - // element is a string - bool is_an_object = std::all_of(init.begin(), init.end(), - [](const basic_json & element) - { - return element.is_array() and element.size() == 2 and element[0].is_string(); - }); - - // adjust type if type deduction is not wanted - if (not type_deduction) - { - // if array is wanted, do not create an object though possible - if (manual_type == value_t::array) - { - is_an_object = false; - } - - // if object is wanted but impossible, throw an exception - if (manual_type == value_t::object and not is_an_object) - { - JSON_THROW(type_error(301, "cannot create object from initializer list")); - } - } - - if (is_an_object) - { - // the initializer list is a list of pairs -> create object - m_type = value_t::object; - m_value = value_t::object; - - std::for_each(init.begin(), init.end(), [this](const basic_json & element) - { - m_value.object->emplace(*(element[0].m_value.string), element[1]); - }); - } - else - { - // the initializer list describes an array -> create array - m_type = value_t::array; - m_value.array = create(init); - } - - assert_invariant(); - } - - /*! - @brief explicitly create an array from an initializer list - - Creates a JSON array value from a given initializer list. That is, given a - list of values `a, b, c`, creates the JSON value `[a, b, c]`. If the - initializer list is empty, the empty array `[]` is created. - - @note This function is only needed to express two edge cases that cannot - be realized with the initializer list constructor (@ref - basic_json(std::initializer_list, bool, value_t)). These cases - are: - 1. creating an array whose elements are all pairs whose first element is a - string -- in this case, the initializer list constructor would create an - object, taking the first elements as keys - 2. creating an empty array -- passing the empty initializer list to the - initializer list constructor yields an empty object - - @param[in] init initializer list with JSON values to create an array from - (optional) - - @return JSON array value - - @complexity Linear in the size of @a init. - - @liveexample{The following code shows an example for the `array` - function.,array} - - @sa @ref basic_json(std::initializer_list, bool, value_t) -- - create a JSON value from an initializer list - @sa @ref object(std::initializer_list) -- create a JSON object - value from an initializer list - - @since version 1.0.0 - */ - static basic_json array(std::initializer_list init = - std::initializer_list()) - { - return basic_json(init, false, value_t::array); - } - - /*! - @brief explicitly create an object from an initializer list - - Creates a JSON object value from a given initializer list. The initializer - lists elements must be pairs, and their first elements must be strings. If - the initializer list is empty, the empty object `{}` is created. - - @note This function is only added for symmetry reasons. In contrast to the - related function @ref array(std::initializer_list), there are - no cases which can only be expressed by this function. That is, any - initializer list @a init can also be passed to the initializer list - constructor @ref basic_json(std::initializer_list, bool, value_t). - - @param[in] init initializer list to create an object from (optional) - - @return JSON object value - - @throw type_error.301 if @a init is not a list of pairs whose first - elements are strings. In this case, no object can be created. When such a - value is passed to @ref basic_json(std::initializer_list, bool, value_t), - an array would have been created from the passed initializer list @a init. - See example below. - - @complexity Linear in the size of @a init. - - @liveexample{The following code shows an example for the `object` - function.,object} - - @sa @ref basic_json(std::initializer_list, bool, value_t) -- - create a JSON value from an initializer list - @sa @ref array(std::initializer_list) -- create a JSON array - value from an initializer list - - @since version 1.0.0 - */ - static basic_json object(std::initializer_list init = - std::initializer_list()) - { - return basic_json(init, false, value_t::object); - } - - /*! - @brief construct an array with count copies of given value - - Constructs a JSON array value by creating @a cnt copies of a passed value. - In case @a cnt is `0`, an empty array is created. As postcondition, - `std::distance(begin(),end()) == cnt` holds. - - @param[in] cnt the number of JSON copies of @a val to create - @param[in] val the JSON value to copy - - @complexity Linear in @a cnt. - - @liveexample{The following code shows examples for the @ref - basic_json(size_type\, const basic_json&) - constructor.,basic_json__size_type_basic_json} - - @since version 1.0.0 - */ - basic_json(size_type cnt, const basic_json& val) - : m_type(value_t::array) - { - m_value.array = create(cnt, val); - assert_invariant(); - } - - /*! - @brief construct a JSON container given an iterator range - - Constructs the JSON value with the contents of the range `[first, last)`. - The semantics depends on the different types a JSON value can have: - - In case of primitive types (number, boolean, or string), @a first must - be `begin()` and @a last must be `end()`. In this case, the value is - copied. Otherwise, invalid_iterator.204 is thrown. - - In case of structured types (array, object), the constructor behaves as - similar versions for `std::vector`. - - In case of a null type, invalid_iterator.206 is thrown. - - @tparam InputIT an input iterator type (@ref iterator or @ref - const_iterator) - - @param[in] first begin of the range to copy from (included) - @param[in] last end of the range to copy from (excluded) - - @pre Iterators @a first and @a last must be initialized. **This - precondition is enforced with an assertion.** - - @pre Range `[first, last)` is valid. Usually, this precondition cannot be - checked efficiently. Only certain edge cases are detected; see the - description of the exceptions below. - - @throw invalid_iterator.201 if iterators @a first and @a last are not - compatible (i.e., do not belong to the same JSON value). In this case, - the range `[first, last)` is undefined. - @throw invalid_iterator.204 if iterators @a first and @a last belong to a - primitive type (number, boolean, or string), but @a first does not point - to the first element any more. In this case, the range `[first, last)` is - undefined. See example code below. - @throw invalid_iterator.206 if iterators @a first and @a last belong to a - null value. In this case, the range `[first, last)` is undefined. - - @complexity Linear in distance between @a first and @a last. - - @liveexample{The example below shows several ways to create JSON values by - specifying a subrange with iterators.,basic_json__InputIt_InputIt} - - @since version 1.0.0 - */ - template::value or - std::is_same::value, int>::type = 0> - basic_json(InputIT first, InputIT last) - { - assert(first.m_object != nullptr); - assert(last.m_object != nullptr); - - // make sure iterator fits the current value - if (first.m_object != last.m_object) - { - JSON_THROW(invalid_iterator(201, "iterators are not compatible")); - } - - // copy type from first iterator - m_type = first.m_object->m_type; - - // check if iterator range is complete for primitive values - switch (m_type) - { - case value_t::boolean: - case value_t::number_float: - case value_t::number_integer: - case value_t::number_unsigned: - case value_t::string: - { - if (not first.m_it.primitive_iterator.is_begin() or not last.m_it.primitive_iterator.is_end()) - { - JSON_THROW(invalid_iterator(204, "iterators out of range")); - } - break; - } - - default: - { - break; - } - } - - switch (m_type) - { - case value_t::number_integer: - { - m_value.number_integer = first.m_object->m_value.number_integer; - break; - } - - case value_t::number_unsigned: - { - m_value.number_unsigned = first.m_object->m_value.number_unsigned; - break; - } - - case value_t::number_float: - { - m_value.number_float = first.m_object->m_value.number_float; - break; - } - - case value_t::boolean: - { - m_value.boolean = first.m_object->m_value.boolean; - break; - } - - case value_t::string: - { - m_value = *first.m_object->m_value.string; - break; - } - - case value_t::object: - { - m_value.object = create(first.m_it.object_iterator, - last.m_it.object_iterator); - break; - } - - case value_t::array: - { - m_value.array = create(first.m_it.array_iterator, - last.m_it.array_iterator); - break; - } - - default: - { - JSON_THROW(invalid_iterator(206, "cannot construct with iterators from " + - first.m_object->type_name())); - } - } - - assert_invariant(); - } - - - /////////////////////////////////////// - // other constructors and destructor // - /////////////////////////////////////// - - /*! - @brief copy constructor - - Creates a copy of a given JSON value. - - @param[in] other the JSON value to copy - - @complexity Linear in the size of @a other. - - @requirement This function helps `basic_json` satisfying the - [Container](http://en.cppreference.com/w/cpp/concept/Container) - requirements: - - The complexity is linear. - - As postcondition, it holds: `other == basic_json(other)`. - - @liveexample{The following code shows an example for the copy - constructor.,basic_json__basic_json} - - @since version 1.0.0 - */ - basic_json(const basic_json& other) - : m_type(other.m_type) - { - // check of passed value is valid - other.assert_invariant(); - - switch (m_type) - { - case value_t::object: - { - m_value = *other.m_value.object; - break; - } - - case value_t::array: - { - m_value = *other.m_value.array; - break; - } - - case value_t::string: - { - m_value = *other.m_value.string; - break; - } - - case value_t::boolean: - { - m_value = other.m_value.boolean; - break; - } - - case value_t::number_integer: - { - m_value = other.m_value.number_integer; - break; - } - - case value_t::number_unsigned: - { - m_value = other.m_value.number_unsigned; - break; - } - - case value_t::number_float: - { - m_value = other.m_value.number_float; - break; - } - - default: - { - break; - } - } - - assert_invariant(); - } - - /*! - @brief move constructor - - Move constructor. Constructs a JSON value with the contents of the given - value @a other using move semantics. It "steals" the resources from @a - other and leaves it as JSON null value. - - @param[in,out] other value to move to this object - - @post @a other is a JSON null value - - @complexity Constant. - - @liveexample{The code below shows the move constructor explicitly called - via std::move.,basic_json__moveconstructor} - - @since version 1.0.0 - */ - basic_json(basic_json&& other) noexcept - : m_type(std::move(other.m_type)), - m_value(std::move(other.m_value)) - { - // check that passed value is valid - other.assert_invariant(); - - // invalidate payload - other.m_type = value_t::null; - other.m_value = {}; - - assert_invariant(); - } - - /*! - @brief copy assignment - - Copy assignment operator. Copies a JSON value via the "copy and swap" - strategy: It is expressed in terms of the copy constructor, destructor, - and the swap() member function. - - @param[in] other value to copy from - - @complexity Linear. - - @requirement This function helps `basic_json` satisfying the - [Container](http://en.cppreference.com/w/cpp/concept/Container) - requirements: - - The complexity is linear. - - @liveexample{The code below shows and example for the copy assignment. It - creates a copy of value `a` which is then swapped with `b`. Finally\, the - copy of `a` (which is the null value after the swap) is - destroyed.,basic_json__copyassignment} - - @since version 1.0.0 - */ - reference& operator=(basic_json other) noexcept ( - std::is_nothrow_move_constructible::value and - std::is_nothrow_move_assignable::value and - std::is_nothrow_move_constructible::value and - std::is_nothrow_move_assignable::value - ) - { - // check that passed value is valid - other.assert_invariant(); - - using std::swap; - swap(m_type, other.m_type); - swap(m_value, other.m_value); - - assert_invariant(); - return *this; - } - - /*! - @brief destructor - - Destroys the JSON value and frees all allocated memory. - - @complexity Linear. - - @requirement This function helps `basic_json` satisfying the - [Container](http://en.cppreference.com/w/cpp/concept/Container) - requirements: - - The complexity is linear. - - All stored elements are destroyed and all memory is freed. - - @since version 1.0.0 - */ - ~basic_json() - { - assert_invariant(); - - switch (m_type) - { - case value_t::object: - { - AllocatorType alloc; - alloc.destroy(m_value.object); - alloc.deallocate(m_value.object, 1); - break; - } - - case value_t::array: - { - AllocatorType alloc; - alloc.destroy(m_value.array); - alloc.deallocate(m_value.array, 1); - break; - } - - case value_t::string: - { - AllocatorType alloc; - alloc.destroy(m_value.string); - alloc.deallocate(m_value.string, 1); - break; - } - - default: - { - // all other types need no specific destructor - break; - } - } - } - - /// @} - - public: - /////////////////////// - // object inspection // - /////////////////////// - - /// @name object inspection - /// Functions to inspect the type of a JSON value. - /// @{ - - /*! - @brief serialization - - Serialization function for JSON values. The function tries to mimic - Python's `json.dumps()` function, and currently supports its @a indent - parameter. - - @param[in] indent If indent is nonnegative, then array elements and object - members will be pretty-printed with that indent level. An indent level of - `0` will only insert newlines. `-1` (the default) selects the most compact - representation. - - @return string containing the serialization of the JSON value - - @complexity Linear. - - @liveexample{The following example shows the effect of different @a indent - parameters to the result of the serialization.,dump} - - @see https://docs.python.org/2/library/json.html#json.dump - - @since version 1.0.0 - */ - string_t dump(const int indent = -1) const - { - std::stringstream ss; - serializer s(ss); - - if (indent >= 0) - { - s.dump(*this, true, static_cast(indent)); - } - else - { - s.dump(*this, false, 0); - } - - return ss.str(); - } - - /*! - @brief return the type of the JSON value (explicit) - - Return the type of the JSON value as a value from the @ref value_t - enumeration. - - @return the type of the JSON value - - @complexity Constant. - - @exceptionsafety No-throw guarantee: this member function never throws - exceptions. - - @liveexample{The following code exemplifies `type()` for all JSON - types.,type} - - @since version 1.0.0 - */ - constexpr value_t type() const noexcept - { - return m_type; - } - - /*! - @brief return whether type is primitive - - This function returns true iff the JSON type is primitive (string, number, - boolean, or null). - - @return `true` if type is primitive (string, number, boolean, or null), - `false` otherwise. - - @complexity Constant. - - @exceptionsafety No-throw guarantee: this member function never throws - exceptions. - - @liveexample{The following code exemplifies `is_primitive()` for all JSON - types.,is_primitive} - - @sa @ref is_structured() -- returns whether JSON value is structured - @sa @ref is_null() -- returns whether JSON value is `null` - @sa @ref is_string() -- returns whether JSON value is a string - @sa @ref is_boolean() -- returns whether JSON value is a boolean - @sa @ref is_number() -- returns whether JSON value is a number - - @since version 1.0.0 - */ - constexpr bool is_primitive() const noexcept - { - return is_null() or is_string() or is_boolean() or is_number(); - } - - /*! - @brief return whether type is structured - - This function returns true iff the JSON type is structured (array or - object). - - @return `true` if type is structured (array or object), `false` otherwise. - - @complexity Constant. - - @exceptionsafety No-throw guarantee: this member function never throws - exceptions. - - @liveexample{The following code exemplifies `is_structured()` for all JSON - types.,is_structured} - - @sa @ref is_primitive() -- returns whether value is primitive - @sa @ref is_array() -- returns whether value is an array - @sa @ref is_object() -- returns whether value is an object - - @since version 1.0.0 - */ - constexpr bool is_structured() const noexcept - { - return is_array() or is_object(); - } - - /*! - @brief return whether value is null - - This function returns true iff the JSON value is null. - - @return `true` if type is null, `false` otherwise. - - @complexity Constant. - - @exceptionsafety No-throw guarantee: this member function never throws - exceptions. - - @liveexample{The following code exemplifies `is_null()` for all JSON - types.,is_null} - - @since version 1.0.0 - */ - constexpr bool is_null() const noexcept - { - return m_type == value_t::null; - } - - /*! - @brief return whether value is a boolean - - This function returns true iff the JSON value is a boolean. - - @return `true` if type is boolean, `false` otherwise. - - @complexity Constant. - - @exceptionsafety No-throw guarantee: this member function never throws - exceptions. - - @liveexample{The following code exemplifies `is_boolean()` for all JSON - types.,is_boolean} - - @since version 1.0.0 - */ - constexpr bool is_boolean() const noexcept - { - return m_type == value_t::boolean; - } - - /*! - @brief return whether value is a number - - This function returns true iff the JSON value is a number. This includes - both integer and floating-point values. - - @return `true` if type is number (regardless whether integer, unsigned - integer or floating-type), `false` otherwise. - - @complexity Constant. - - @exceptionsafety No-throw guarantee: this member function never throws - exceptions. - - @liveexample{The following code exemplifies `is_number()` for all JSON - types.,is_number} - - @sa @ref is_number_integer() -- check if value is an integer or unsigned - integer number - @sa @ref is_number_unsigned() -- check if value is an unsigned integer - number - @sa @ref is_number_float() -- check if value is a floating-point number - - @since version 1.0.0 - */ - constexpr bool is_number() const noexcept - { - return is_number_integer() or is_number_float(); - } - - /*! - @brief return whether value is an integer number - - This function returns true iff the JSON value is an integer or unsigned - integer number. This excludes floating-point values. - - @return `true` if type is an integer or unsigned integer number, `false` - otherwise. - - @complexity Constant. - - @exceptionsafety No-throw guarantee: this member function never throws - exceptions. - - @liveexample{The following code exemplifies `is_number_integer()` for all - JSON types.,is_number_integer} - - @sa @ref is_number() -- check if value is a number - @sa @ref is_number_unsigned() -- check if value is an unsigned integer - number - @sa @ref is_number_float() -- check if value is a floating-point number - - @since version 1.0.0 - */ - constexpr bool is_number_integer() const noexcept - { - return m_type == value_t::number_integer or m_type == value_t::number_unsigned; - } - - /*! - @brief return whether value is an unsigned integer number - - This function returns true iff the JSON value is an unsigned integer - number. This excludes floating-point and (signed) integer values. - - @return `true` if type is an unsigned integer number, `false` otherwise. - - @complexity Constant. - - @exceptionsafety No-throw guarantee: this member function never throws - exceptions. - - @liveexample{The following code exemplifies `is_number_unsigned()` for all - JSON types.,is_number_unsigned} - - @sa @ref is_number() -- check if value is a number - @sa @ref is_number_integer() -- check if value is an integer or unsigned - integer number - @sa @ref is_number_float() -- check if value is a floating-point number - - @since version 2.0.0 - */ - constexpr bool is_number_unsigned() const noexcept - { - return m_type == value_t::number_unsigned; - } - - /*! - @brief return whether value is a floating-point number - - This function returns true iff the JSON value is a floating-point number. - This excludes integer and unsigned integer values. - - @return `true` if type is a floating-point number, `false` otherwise. - - @complexity Constant. - - @exceptionsafety No-throw guarantee: this member function never throws - exceptions. - - @liveexample{The following code exemplifies `is_number_float()` for all - JSON types.,is_number_float} - - @sa @ref is_number() -- check if value is number - @sa @ref is_number_integer() -- check if value is an integer number - @sa @ref is_number_unsigned() -- check if value is an unsigned integer - number - - @since version 1.0.0 - */ - constexpr bool is_number_float() const noexcept - { - return m_type == value_t::number_float; - } - - /*! - @brief return whether value is an object - - This function returns true iff the JSON value is an object. - - @return `true` if type is object, `false` otherwise. - - @complexity Constant. - - @exceptionsafety No-throw guarantee: this member function never throws - exceptions. - - @liveexample{The following code exemplifies `is_object()` for all JSON - types.,is_object} - - @since version 1.0.0 - */ - constexpr bool is_object() const noexcept - { - return m_type == value_t::object; - } - - /*! - @brief return whether value is an array - - This function returns true iff the JSON value is an array. - - @return `true` if type is array, `false` otherwise. - - @complexity Constant. - - @exceptionsafety No-throw guarantee: this member function never throws - exceptions. - - @liveexample{The following code exemplifies `is_array()` for all JSON - types.,is_array} - - @since version 1.0.0 - */ - constexpr bool is_array() const noexcept - { - return m_type == value_t::array; - } - - /*! - @brief return whether value is a string - - This function returns true iff the JSON value is a string. - - @return `true` if type is string, `false` otherwise. - - @complexity Constant. - - @exceptionsafety No-throw guarantee: this member function never throws - exceptions. - - @liveexample{The following code exemplifies `is_string()` for all JSON - types.,is_string} - - @since version 1.0.0 - */ - constexpr bool is_string() const noexcept - { - return m_type == value_t::string; - } - - /*! - @brief return whether value is discarded - - This function returns true iff the JSON value was discarded during parsing - with a callback function (see @ref parser_callback_t). - - @note This function will always be `false` for JSON values after parsing. - That is, discarded values can only occur during parsing, but will be - removed when inside a structured value or replaced by null in other cases. - - @return `true` if type is discarded, `false` otherwise. - - @complexity Constant. - - @exceptionsafety No-throw guarantee: this member function never throws - exceptions. - - @liveexample{The following code exemplifies `is_discarded()` for all JSON - types.,is_discarded} - - @since version 1.0.0 - */ - constexpr bool is_discarded() const noexcept - { - return m_type == value_t::discarded; - } - - /*! - @brief return the type of the JSON value (implicit) - - Implicitly return the type of the JSON value as a value from the @ref - value_t enumeration. - - @return the type of the JSON value - - @complexity Constant. - - @exceptionsafety No-throw guarantee: this member function never throws - exceptions. - - @liveexample{The following code exemplifies the @ref value_t operator for - all JSON types.,operator__value_t} - - @since version 1.0.0 - */ - constexpr operator value_t() const noexcept - { - return m_type; - } - - /// @} - - private: - ////////////////// - // value access // - ////////////////// - - /// get a boolean (explicit) - boolean_t get_impl(boolean_t* /*unused*/) const - { - if (is_boolean()) - { - return m_value.boolean; - } - - JSON_THROW(type_error(302, "type must be boolean, but is " + type_name())); - } - - /// get a pointer to the value (object) - object_t* get_impl_ptr(object_t* /*unused*/) noexcept - { - return is_object() ? m_value.object : nullptr; - } - - /// get a pointer to the value (object) - constexpr const object_t* get_impl_ptr(const object_t* /*unused*/) const noexcept - { - return is_object() ? m_value.object : nullptr; - } - - /// get a pointer to the value (array) - array_t* get_impl_ptr(array_t* /*unused*/) noexcept - { - return is_array() ? m_value.array : nullptr; - } - - /// get a pointer to the value (array) - constexpr const array_t* get_impl_ptr(const array_t* /*unused*/) const noexcept - { - return is_array() ? m_value.array : nullptr; - } - - /// get a pointer to the value (string) - string_t* get_impl_ptr(string_t* /*unused*/) noexcept - { - return is_string() ? m_value.string : nullptr; - } - - /// get a pointer to the value (string) - constexpr const string_t* get_impl_ptr(const string_t* /*unused*/) const noexcept - { - return is_string() ? m_value.string : nullptr; - } - - /// get a pointer to the value (boolean) - boolean_t* get_impl_ptr(boolean_t* /*unused*/) noexcept - { - return is_boolean() ? &m_value.boolean : nullptr; - } - - /// get a pointer to the value (boolean) - constexpr const boolean_t* get_impl_ptr(const boolean_t* /*unused*/) const noexcept - { - return is_boolean() ? &m_value.boolean : nullptr; - } - - /// get a pointer to the value (integer number) - number_integer_t* get_impl_ptr(number_integer_t* /*unused*/) noexcept - { - return is_number_integer() ? &m_value.number_integer : nullptr; - } - - /// get a pointer to the value (integer number) - constexpr const number_integer_t* get_impl_ptr(const number_integer_t* /*unused*/) const noexcept - { - return is_number_integer() ? &m_value.number_integer : nullptr; - } - - /// get a pointer to the value (unsigned number) - number_unsigned_t* get_impl_ptr(number_unsigned_t* /*unused*/) noexcept - { - return is_number_unsigned() ? &m_value.number_unsigned : nullptr; - } - - /// get a pointer to the value (unsigned number) - constexpr const number_unsigned_t* get_impl_ptr(const number_unsigned_t* /*unused*/) const noexcept - { - return is_number_unsigned() ? &m_value.number_unsigned : nullptr; - } - - /// get a pointer to the value (floating-point number) - number_float_t* get_impl_ptr(number_float_t* /*unused*/) noexcept - { - return is_number_float() ? &m_value.number_float : nullptr; - } - - /// get a pointer to the value (floating-point number) - constexpr const number_float_t* get_impl_ptr(const number_float_t* /*unused*/) const noexcept - { - return is_number_float() ? &m_value.number_float : nullptr; - } - - /*! - @brief helper function to implement get_ref() - - This funcion helps to implement get_ref() without code duplication for - const and non-const overloads - - @tparam ThisType will be deduced as `basic_json` or `const basic_json` - - @throw type_error.303 if ReferenceType does not match underlying value - type of the current JSON - */ - template - static ReferenceType get_ref_impl(ThisType& obj) - { - // helper type - using PointerType = typename std::add_pointer::type; - - // delegate the call to get_ptr<>() - auto ptr = obj.template get_ptr(); - - if (ptr != nullptr) - { - return *ptr; - } - - JSON_THROW(type_error(303, "incompatible ReferenceType for get_ref, actual type is " + obj.type_name())); - } - - public: - /// @name value access - /// Direct access to the stored value of a JSON value. - /// @{ - - /*! - @brief get special-case overload - - This overloads avoids a lot of template boilerplate, it can be seen as the - identity method - - @tparam BasicJsonType == @ref basic_json - - @return a copy of *this - - @complexity Constant. - - @since version 2.1.0 - */ - template < - typename BasicJsonType, - detail::enable_if_t::type, - basic_json_t>::value, - int> = 0 > - basic_json get() const - { - return *this; - } - - /*! - @brief get a value (explicit) - - Explicit type conversion between the JSON value and a compatible value - which is [CopyConstructible](http://en.cppreference.com/w/cpp/concept/CopyConstructible) - and [DefaultConstructible](http://en.cppreference.com/w/cpp/concept/DefaultConstructible). - The value is converted by calling the @ref json_serializer - `from_json()` method. - - The function is equivalent to executing - @code {.cpp} - ValueType ret; - JSONSerializer::from_json(*this, ret); - return ret; - @endcode - - This overloads is chosen if: - - @a ValueType is not @ref basic_json, - - @ref json_serializer has a `from_json()` method of the form - `void from_json(const @ref basic_json&, ValueType&)`, and - - @ref json_serializer does not have a `from_json()` method of - the form `ValueType from_json(const @ref basic_json&)` - - @tparam ValueTypeCV the provided value type - @tparam ValueType the returned value type - - @return copy of the JSON value, converted to @a ValueType - - @throw what @ref json_serializer `from_json()` method throws - - @liveexample{The example below shows several conversions from JSON values - to other types. There a few things to note: (1) Floating-point numbers can - be converted to integers\, (2) A JSON array can be converted to a standard - `std::vector`\, (3) A JSON object can be converted to C++ - associative containers such as `std::unordered_map`.,get__ValueType_const} - - @since version 2.1.0 - */ - template < - typename ValueTypeCV, - typename ValueType = detail::uncvref_t, - detail::enable_if_t < - not std::is_same::value and - detail::has_from_json::value and - not detail::has_non_default_from_json::value, - int > = 0 > - ValueType get() const noexcept(noexcept( - JSONSerializer::from_json(std::declval(), std::declval()))) - { - // we cannot static_assert on ValueTypeCV being non-const, because - // there is support for get(), which is why we - // still need the uncvref - static_assert(not std::is_reference::value, - "get() cannot be used with reference types, you might want to use get_ref()"); - static_assert(std::is_default_constructible::value, - "types must be DefaultConstructible when used with get()"); - - ValueType ret; - JSONSerializer::from_json(*this, ret); - return ret; - } - - /*! - @brief get a value (explicit); special case - - Explicit type conversion between the JSON value and a compatible value - which is **not** [CopyConstructible](http://en.cppreference.com/w/cpp/concept/CopyConstructible) - and **not** [DefaultConstructible](http://en.cppreference.com/w/cpp/concept/DefaultConstructible). - The value is converted by calling the @ref json_serializer - `from_json()` method. - - The function is equivalent to executing - @code {.cpp} - return JSONSerializer::from_json(*this); - @endcode - - This overloads is chosen if: - - @a ValueType is not @ref basic_json and - - @ref json_serializer has a `from_json()` method of the form - `ValueType from_json(const @ref basic_json&)` - - @note If @ref json_serializer has both overloads of - `from_json()`, this one is chosen. - - @tparam ValueTypeCV the provided value type - @tparam ValueType the returned value type - - @return copy of the JSON value, converted to @a ValueType - - @throw what @ref json_serializer `from_json()` method throws - - @since version 2.1.0 - */ - template < - typename ValueTypeCV, - typename ValueType = detail::uncvref_t, - detail::enable_if_t::value and - detail::has_non_default_from_json::value, int> = 0 > - ValueType get() const noexcept(noexcept( - JSONSerializer::from_json(std::declval()))) - { - static_assert(not std::is_reference::value, - "get() cannot be used with reference types, you might want to use get_ref()"); - return JSONSerializer::from_json(*this); - } - - /*! - @brief get a pointer value (explicit) - - Explicit pointer access to the internally stored JSON value. No copies are - made. - - @warning The pointer becomes invalid if the underlying JSON object - changes. - - @tparam PointerType pointer type; must be a pointer to @ref array_t, @ref - object_t, @ref string_t, @ref boolean_t, @ref number_integer_t, - @ref number_unsigned_t, or @ref number_float_t. - - @return pointer to the internally stored JSON value if the requested - pointer type @a PointerType fits to the JSON value; `nullptr` otherwise - - @complexity Constant. - - @liveexample{The example below shows how pointers to internal values of a - JSON value can be requested. Note that no type conversions are made and a - `nullptr` is returned if the value and the requested pointer type does not - match.,get__PointerType} - - @sa @ref get_ptr() for explicit pointer-member access - - @since version 1.0.0 - */ - template::value, int>::type = 0> - PointerType get() noexcept - { - // delegate the call to get_ptr - return get_ptr(); - } - - /*! - @brief get a pointer value (explicit) - @copydoc get() - */ - template::value, int>::type = 0> - constexpr const PointerType get() const noexcept - { - // delegate the call to get_ptr - return get_ptr(); - } - - /*! - @brief get a pointer value (implicit) - - Implicit pointer access to the internally stored JSON value. No copies are - made. - - @warning Writing data to the pointee of the result yields an undefined - state. - - @tparam PointerType pointer type; must be a pointer to @ref array_t, @ref - object_t, @ref string_t, @ref boolean_t, @ref number_integer_t, - @ref number_unsigned_t, or @ref number_float_t. Enforced by a static - assertion. - - @return pointer to the internally stored JSON value if the requested - pointer type @a PointerType fits to the JSON value; `nullptr` otherwise - - @complexity Constant. - - @liveexample{The example below shows how pointers to internal values of a - JSON value can be requested. Note that no type conversions are made and a - `nullptr` is returned if the value and the requested pointer type does not - match.,get_ptr} - - @since version 1.0.0 - */ - template::value, int>::type = 0> - PointerType get_ptr() noexcept - { - // get the type of the PointerType (remove pointer and const) - using pointee_t = typename std::remove_const::type>::type>::type; - // make sure the type matches the allowed types - static_assert( - std::is_same::value - or std::is_same::value - or std::is_same::value - or std::is_same::value - or std::is_same::value - or std::is_same::value - or std::is_same::value - , "incompatible pointer type"); - - // delegate the call to get_impl_ptr<>() - return get_impl_ptr(static_cast(nullptr)); - } - - /*! - @brief get a pointer value (implicit) - @copydoc get_ptr() - */ - template::value and - std::is_const::type>::value, int>::type = 0> - constexpr const PointerType get_ptr() const noexcept - { - // get the type of the PointerType (remove pointer and const) - using pointee_t = typename std::remove_const::type>::type>::type; - // make sure the type matches the allowed types - static_assert( - std::is_same::value - or std::is_same::value - or std::is_same::value - or std::is_same::value - or std::is_same::value - or std::is_same::value - or std::is_same::value - , "incompatible pointer type"); - - // delegate the call to get_impl_ptr<>() const - return get_impl_ptr(static_cast(nullptr)); - } - - /*! - @brief get a reference value (implicit) - - Implicit reference access to the internally stored JSON value. No copies - are made. - - @warning Writing data to the referee of the result yields an undefined - state. - - @tparam ReferenceType reference type; must be a reference to @ref array_t, - @ref object_t, @ref string_t, @ref boolean_t, @ref number_integer_t, or - @ref number_float_t. Enforced by static assertion. - - @return reference to the internally stored JSON value if the requested - reference type @a ReferenceType fits to the JSON value; throws - type_error.303 otherwise - - @throw type_error.303 in case passed type @a ReferenceType is incompatible - with the stored JSON value; see example below - - @complexity Constant. - - @liveexample{The example shows several calls to `get_ref()`.,get_ref} - - @since version 1.1.0 - */ - template::value, int>::type = 0> - ReferenceType get_ref() - { - // delegate call to get_ref_impl - return get_ref_impl(*this); - } - - /*! - @brief get a reference value (implicit) - @copydoc get_ref() - */ - template::value and - std::is_const::type>::value, int>::type = 0> - ReferenceType get_ref() const - { - // delegate call to get_ref_impl - return get_ref_impl(*this); - } - - /*! - @brief get a value (implicit) - - Implicit type conversion between the JSON value and a compatible value. - The call is realized by calling @ref get() const. - - @tparam ValueType non-pointer type compatible to the JSON value, for - instance `int` for JSON integer numbers, `bool` for JSON booleans, or - `std::vector` types for JSON arrays. The character type of @ref string_t - as well as an initializer list of this type is excluded to avoid - ambiguities as these types implicitly convert to `std::string`. - - @return copy of the JSON value, converted to type @a ValueType - - @throw type_error.302 in case passed type @a ValueType is incompatible - to the JSON value type (e.g., the JSON value is of type boolean, but a - string is requested); see example below - - @complexity Linear in the size of the JSON value. - - @liveexample{The example below shows several conversions from JSON values - to other types. There a few things to note: (1) Floating-point numbers can - be converted to integers\, (2) A JSON array can be converted to a standard - `std::vector`\, (3) A JSON object can be converted to C++ - associative containers such as `std::unordered_map`.,operator__ValueType} - - @since version 1.0.0 - */ - template < typename ValueType, typename std::enable_if < - not std::is_pointer::value and - not std::is_same::value -#ifndef _MSC_VER // fix for issue #167 operator<< ambiguity under VS2015 - and not std::is_same>::value -#endif - , int >::type = 0 > - operator ValueType() const - { - // delegate the call to get<>() const - return get(); - } - - /// @} - - - //////////////////// - // element access // - //////////////////// - - /// @name element access - /// Access to the JSON value. - /// @{ - - /*! - @brief access specified array element with bounds checking - - Returns a reference to the element at specified location @a idx, with - bounds checking. - - @param[in] idx index of the element to access - - @return reference to the element at index @a idx - - @throw type_error.304 if the JSON value is not an array; in this case, - calling `at` with an index makes no sense. See example below. - @throw out_of_range.401 if the index @a idx is out of range of the array; - that is, `idx >= size()`. See example below. - - @exceptionsafety Strong guarantee: if an exception is thrown, there are no - changes in the JSON value. - - @complexity Constant. - - @since version 1.0.0 - - @liveexample{The example below shows how array elements can be read and - written using `at()`. It also demonstrates the different exceptions that - can be thrown.,at__size_type} - */ - reference at(size_type idx) - { - // at only works for arrays - if (is_array()) - { - JSON_TRY - { - return m_value.array->at(idx); - } - JSON_CATCH (std::out_of_range&) - { - // create better exception explanation - JSON_THROW(out_of_range(401, "array index " + std::to_string(idx) + " is out of range")); - } - } - else - { - JSON_THROW(type_error(304, "cannot use at() with " + type_name())); - } - } - - /*! - @brief access specified array element with bounds checking - - Returns a const reference to the element at specified location @a idx, - with bounds checking. - - @param[in] idx index of the element to access - - @return const reference to the element at index @a idx - - @throw type_error.304 if the JSON value is not an array; in this case, - calling `at` with an index makes no sense. See example below. - @throw out_of_range.401 if the index @a idx is out of range of the array; - that is, `idx >= size()`. See example below. - - @exceptionsafety Strong guarantee: if an exception is thrown, there are no - changes in the JSON value. - - @complexity Constant. - - @since version 1.0.0 - - @liveexample{The example below shows how array elements can be read using - `at()`. It also demonstrates the different exceptions that can be thrown., - at__size_type_const} - */ - const_reference at(size_type idx) const - { - // at only works for arrays - if (is_array()) - { - JSON_TRY - { - return m_value.array->at(idx); - } - JSON_CATCH (std::out_of_range&) - { - // create better exception explanation - JSON_THROW(out_of_range(401, "array index " + std::to_string(idx) + " is out of range")); - } - } - else - { - JSON_THROW(type_error(304, "cannot use at() with " + type_name())); - } - } - - /*! - @brief access specified object element with bounds checking - - Returns a reference to the element at with specified key @a key, with - bounds checking. - - @param[in] key key of the element to access - - @return reference to the element at key @a key - - @throw type_error.304 if the JSON value is not an object; in this case, - calling `at` with a key makes no sense. See example below. - @throw out_of_range.403 if the key @a key is is not stored in the object; - that is, `find(key) == end()`. See example below. - - @exceptionsafety Strong guarantee: if an exception is thrown, there are no - changes in the JSON value. - - @complexity Logarithmic in the size of the container. - - @sa @ref operator[](const typename object_t::key_type&) for unchecked - access by reference - @sa @ref value() for access by value with a default value - - @since version 1.0.0 - - @liveexample{The example below shows how object elements can be read and - written using `at()`. It also demonstrates the different exceptions that - can be thrown.,at__object_t_key_type} - */ - reference at(const typename object_t::key_type& key) - { - // at only works for objects - if (is_object()) - { - JSON_TRY - { - return m_value.object->at(key); - } - JSON_CATCH (std::out_of_range&) - { - // create better exception explanation - JSON_THROW(out_of_range(403, "key '" + key + "' not found")); - } - } - else - { - JSON_THROW(type_error(304, "cannot use at() with " + type_name())); - } - } - - /*! - @brief access specified object element with bounds checking - - Returns a const reference to the element at with specified key @a key, - with bounds checking. - - @param[in] key key of the element to access - - @return const reference to the element at key @a key - - @throw type_error.304 if the JSON value is not an object; in this case, - calling `at` with a key makes no sense. See example below. - @throw out_of_range.403 if the key @a key is is not stored in the object; - that is, `find(key) == end()`. See example below. - - @exceptionsafety Strong guarantee: if an exception is thrown, there are no - changes in the JSON value. - - @complexity Logarithmic in the size of the container. - - @sa @ref operator[](const typename object_t::key_type&) for unchecked - access by reference - @sa @ref value() for access by value with a default value - - @since version 1.0.0 - - @liveexample{The example below shows how object elements can be read using - `at()`. It also demonstrates the different exceptions that can be thrown., - at__object_t_key_type_const} - */ - const_reference at(const typename object_t::key_type& key) const - { - // at only works for objects - if (is_object()) - { - JSON_TRY - { - return m_value.object->at(key); - } - JSON_CATCH (std::out_of_range&) - { - // create better exception explanation - JSON_THROW(out_of_range(403, "key '" + key + "' not found")); - } - } - else - { - JSON_THROW(type_error(304, "cannot use at() with " + type_name())); - } - } - - /*! - @brief access specified array element - - Returns a reference to the element at specified location @a idx. - - @note If @a idx is beyond the range of the array (i.e., `idx >= size()`), - then the array is silently filled up with `null` values to make `idx` a - valid reference to the last stored element. - - @param[in] idx index of the element to access - - @return reference to the element at index @a idx - - @throw type_error.305 if the JSON value is not an array or null; in that - cases, using the [] operator with an index makes no sense. - - @complexity Constant if @a idx is in the range of the array. Otherwise - linear in `idx - size()`. - - @liveexample{The example below shows how array elements can be read and - written using `[]` operator. Note the addition of `null` - values.,operatorarray__size_type} - - @since version 1.0.0 - */ - reference operator[](size_type idx) - { - // implicitly convert null value to an empty array - if (is_null()) - { - m_type = value_t::array; - m_value.array = create(); - assert_invariant(); - } - - // operator[] only works for arrays - if (is_array()) - { - // fill up array with null values if given idx is outside range - if (idx >= m_value.array->size()) - { - m_value.array->insert(m_value.array->end(), - idx - m_value.array->size() + 1, - basic_json()); - } - - return m_value.array->operator[](idx); - } - - JSON_THROW(type_error(305, "cannot use operator[] with " + type_name())); - } - - /*! - @brief access specified array element - - Returns a const reference to the element at specified location @a idx. - - @param[in] idx index of the element to access - - @return const reference to the element at index @a idx - - @throw type_error.305 if the JSON value is not an array; in that cases, - using the [] operator with an index makes no sense. - - @complexity Constant. - - @liveexample{The example below shows how array elements can be read using - the `[]` operator.,operatorarray__size_type_const} - - @since version 1.0.0 - */ - const_reference operator[](size_type idx) const - { - // const operator[] only works for arrays - if (is_array()) - { - return m_value.array->operator[](idx); - } - - JSON_THROW(type_error(305, "cannot use operator[] with " + type_name())); - } - - /*! - @brief access specified object element - - Returns a reference to the element at with specified key @a key. - - @note If @a key is not found in the object, then it is silently added to - the object and filled with a `null` value to make `key` a valid reference. - In case the value was `null` before, it is converted to an object. - - @param[in] key key of the element to access - - @return reference to the element at key @a key - - @throw type_error.305 if the JSON value is not an object or null; in that - cases, using the [] operator with a key makes no sense. - - @complexity Logarithmic in the size of the container. - - @liveexample{The example below shows how object elements can be read and - written using the `[]` operator.,operatorarray__key_type} - - @sa @ref at(const typename object_t::key_type&) for access by reference - with range checking - @sa @ref value() for access by value with a default value - - @since version 1.0.0 - */ - reference operator[](const typename object_t::key_type& key) - { - // implicitly convert null value to an empty object - if (is_null()) - { - m_type = value_t::object; - m_value.object = create(); - assert_invariant(); - } - - // operator[] only works for objects - if (is_object()) - { - return m_value.object->operator[](key); - } - - JSON_THROW(type_error(305, "cannot use operator[] with " + type_name())); - } - - /*! - @brief read-only access specified object element - - Returns a const reference to the element at with specified key @a key. No - bounds checking is performed. - - @warning If the element with key @a key does not exist, the behavior is - undefined. - - @param[in] key key of the element to access - - @return const reference to the element at key @a key - - @pre The element with key @a key must exist. **This precondition is - enforced with an assertion.** - - @throw type_error.305 if the JSON value is not an object; in that cases, - using the [] operator with a key makes no sense. - - @complexity Logarithmic in the size of the container. - - @liveexample{The example below shows how object elements can be read using - the `[]` operator.,operatorarray__key_type_const} - - @sa @ref at(const typename object_t::key_type&) for access by reference - with range checking - @sa @ref value() for access by value with a default value - - @since version 1.0.0 - */ - const_reference operator[](const typename object_t::key_type& key) const - { - // const operator[] only works for objects - if (is_object()) - { - assert(m_value.object->find(key) != m_value.object->end()); - return m_value.object->find(key)->second; - } - - JSON_THROW(type_error(305, "cannot use operator[] with " + type_name())); - } - - /*! - @brief access specified object element - - Returns a reference to the element at with specified key @a key. - - @note If @a key is not found in the object, then it is silently added to - the object and filled with a `null` value to make `key` a valid reference. - In case the value was `null` before, it is converted to an object. - - @param[in] key key of the element to access - - @return reference to the element at key @a key - - @throw type_error.305 if the JSON value is not an object or null; in that - cases, using the [] operator with a key makes no sense. - - @complexity Logarithmic in the size of the container. - - @liveexample{The example below shows how object elements can be read and - written using the `[]` operator.,operatorarray__key_type} - - @sa @ref at(const typename object_t::key_type&) for access by reference - with range checking - @sa @ref value() for access by value with a default value - - @since version 1.0.0 - */ - template - reference operator[](T * (&key)[n]) - { - return operator[](static_cast(key)); - } - - /*! - @brief read-only access specified object element - - Returns a const reference to the element at with specified key @a key. No - bounds checking is performed. - - @warning If the element with key @a key does not exist, the behavior is - undefined. - - @note This function is required for compatibility reasons with Clang. - - @param[in] key key of the element to access - - @return const reference to the element at key @a key - - @throw type_error.305 if the JSON value is not an object; in that cases, - using the [] operator with a key makes no sense. - - @complexity Logarithmic in the size of the container. - - @liveexample{The example below shows how object elements can be read using - the `[]` operator.,operatorarray__key_type_const} - - @sa @ref at(const typename object_t::key_type&) for access by reference - with range checking - @sa @ref value() for access by value with a default value - - @since version 1.0.0 - */ - template - const_reference operator[](T * (&key)[n]) const - { - return operator[](static_cast(key)); - } - - /*! - @brief access specified object element - - Returns a reference to the element at with specified key @a key. - - @note If @a key is not found in the object, then it is silently added to - the object and filled with a `null` value to make `key` a valid reference. - In case the value was `null` before, it is converted to an object. - - @param[in] key key of the element to access - - @return reference to the element at key @a key - - @throw type_error.305 if the JSON value is not an object or null; in that - cases, using the [] operator with a key makes no sense. - - @complexity Logarithmic in the size of the container. - - @liveexample{The example below shows how object elements can be read and - written using the `[]` operator.,operatorarray__key_type} - - @sa @ref at(const typename object_t::key_type&) for access by reference - with range checking - @sa @ref value() for access by value with a default value - - @since version 1.1.0 - */ - template - reference operator[](T* key) - { - // implicitly convert null to object - if (is_null()) - { - m_type = value_t::object; - m_value = value_t::object; - assert_invariant(); - } - - // at only works for objects - if (is_object()) - { - return m_value.object->operator[](key); - } - - JSON_THROW(type_error(305, "cannot use operator[] with " + type_name())); - } - - /*! - @brief read-only access specified object element - - Returns a const reference to the element at with specified key @a key. No - bounds checking is performed. - - @warning If the element with key @a key does not exist, the behavior is - undefined. - - @param[in] key key of the element to access - - @return const reference to the element at key @a key - - @pre The element with key @a key must exist. **This precondition is - enforced with an assertion.** - - @throw type_error.305 if the JSON value is not an object; in that cases, - using the [] operator with a key makes no sense. - - @complexity Logarithmic in the size of the container. - - @liveexample{The example below shows how object elements can be read using - the `[]` operator.,operatorarray__key_type_const} - - @sa @ref at(const typename object_t::key_type&) for access by reference - with range checking - @sa @ref value() for access by value with a default value - - @since version 1.1.0 - */ - template - const_reference operator[](T* key) const - { - // at only works for objects - if (is_object()) - { - assert(m_value.object->find(key) != m_value.object->end()); - return m_value.object->find(key)->second; - } - - JSON_THROW(type_error(305, "cannot use operator[] with " + type_name())); - } - - /*! - @brief access specified object element with default value - - Returns either a copy of an object's element at the specified key @a key - or a given default value if no element with key @a key exists. - - The function is basically equivalent to executing - @code {.cpp} - try { - return at(key); - } catch(out_of_range) { - return default_value; - } - @endcode - - @note Unlike @ref at(const typename object_t::key_type&), this function - does not throw if the given key @a key was not found. - - @note Unlike @ref operator[](const typename object_t::key_type& key), this - function does not implicitly add an element to the position defined by @a - key. This function is furthermore also applicable to const objects. - - @param[in] key key of the element to access - @param[in] default_value the value to return if @a key is not found - - @tparam ValueType type compatible to JSON values, for instance `int` for - JSON integer numbers, `bool` for JSON booleans, or `std::vector` types for - JSON arrays. Note the type of the expected value at @a key and the default - value @a default_value must be compatible. - - @return copy of the element at key @a key or @a default_value if @a key - is not found - - @throw type_error.306 if the JSON value is not an objec; in that cases, - using `value()` with a key makes no sense. - - @complexity Logarithmic in the size of the container. - - @liveexample{The example below shows how object elements can be queried - with a default value.,basic_json__value} - - @sa @ref at(const typename object_t::key_type&) for access by reference - with range checking - @sa @ref operator[](const typename object_t::key_type&) for unchecked - access by reference - - @since version 1.0.0 - */ - template::value, int>::type = 0> - ValueType value(const typename object_t::key_type& key, ValueType default_value) const - { - // at only works for objects - if (is_object()) - { - // if key is found, return value and given default value otherwise - const auto it = find(key); - if (it != end()) - { - return *it; - } - - return default_value; - } - else - { - JSON_THROW(type_error(306, "cannot use value() with " + type_name())); - } - } - - /*! - @brief overload for a default value of type const char* - @copydoc basic_json::value(const typename object_t::key_type&, ValueType) const - */ - string_t value(const typename object_t::key_type& key, const char* default_value) const - { - return value(key, string_t(default_value)); - } - - /*! - @brief access specified object element via JSON Pointer with default value - - Returns either a copy of an object's element at the specified key @a key - or a given default value if no element with key @a key exists. - - The function is basically equivalent to executing - @code {.cpp} - try { - return at(ptr); - } catch(out_of_range) { - return default_value; - } - @endcode - - @note Unlike @ref at(const json_pointer&), this function does not throw - if the given key @a key was not found. - - @param[in] ptr a JSON pointer to the element to access - @param[in] default_value the value to return if @a ptr found no value - - @tparam ValueType type compatible to JSON values, for instance `int` for - JSON integer numbers, `bool` for JSON booleans, or `std::vector` types for - JSON arrays. Note the type of the expected value at @a key and the default - value @a default_value must be compatible. - - @return copy of the element at key @a key or @a default_value if @a key - is not found - - @throw type_error.306 if the JSON value is not an objec; in that cases, - using `value()` with a key makes no sense. - - @complexity Logarithmic in the size of the container. - - @liveexample{The example below shows how object elements can be queried - with a default value.,basic_json__value_ptr} - - @sa @ref operator[](const json_pointer&) for unchecked access by reference - - @since version 2.0.2 - */ - template::value, int>::type = 0> - ValueType value(const json_pointer& ptr, ValueType default_value) const - { - // at only works for objects - if (is_object()) - { - // if pointer resolves a value, return it or use default value - JSON_TRY - { - return ptr.get_checked(this); - } - JSON_CATCH (out_of_range&) - { - return default_value; - } - } - - JSON_THROW(type_error(306, "cannot use value() with " + type_name())); - } - - /*! - @brief overload for a default value of type const char* - @copydoc basic_json::value(const json_pointer&, ValueType) const - */ - string_t value(const json_pointer& ptr, const char* default_value) const - { - return value(ptr, string_t(default_value)); - } - - /*! - @brief access the first element - - Returns a reference to the first element in the container. For a JSON - container `c`, the expression `c.front()` is equivalent to `*c.begin()`. - - @return In case of a structured type (array or object), a reference to the - first element is returned. In case of number, string, or boolean values, a - reference to the value is returned. - - @complexity Constant. - - @pre The JSON value must not be `null` (would throw `std::out_of_range`) - or an empty array or object (undefined behavior, **guarded by - assertions**). - @post The JSON value remains unchanged. - - @throw invalid_iterator.214 when called on `null` value - - @liveexample{The following code shows an example for `front()`.,front} - - @sa @ref back() -- access the last element - - @since version 1.0.0 - */ - reference front() - { - return *begin(); - } - - /*! - @copydoc basic_json::front() - */ - const_reference front() const - { - return *cbegin(); - } - - /*! - @brief access the last element - - Returns a reference to the last element in the container. For a JSON - container `c`, the expression `c.back()` is equivalent to - @code {.cpp} - auto tmp = c.end(); - --tmp; - return *tmp; - @endcode - - @return In case of a structured type (array or object), a reference to the - last element is returned. In case of number, string, or boolean values, a - reference to the value is returned. - - @complexity Constant. - - @pre The JSON value must not be `null` (would throw `std::out_of_range`) - or an empty array or object (undefined behavior, **guarded by - assertions**). - @post The JSON value remains unchanged. - - @throw invalid_iterator.214 when called on a `null` value. See example - below. - - @liveexample{The following code shows an example for `back()`.,back} - - @sa @ref front() -- access the first element - - @since version 1.0.0 - */ - reference back() - { - auto tmp = end(); - --tmp; - return *tmp; - } - - /*! - @copydoc basic_json::back() - */ - const_reference back() const - { - auto tmp = cend(); - --tmp; - return *tmp; - } - - /*! - @brief remove element given an iterator - - Removes the element specified by iterator @a pos. The iterator @a pos must - be valid and dereferenceable. Thus the `end()` iterator (which is valid, - but is not dereferenceable) cannot be used as a value for @a pos. - - If called on a primitive type other than `null`, the resulting JSON value - will be `null`. - - @param[in] pos iterator to the element to remove - @return Iterator following the last removed element. If the iterator @a - pos refers to the last element, the `end()` iterator is returned. - - @tparam IteratorType an @ref iterator or @ref const_iterator - - @post Invalidates iterators and references at or after the point of the - erase, including the `end()` iterator. - - @throw type_error.307 if called on a `null` value; example: `"cannot use - erase() with null"` - @throw invalid_iterator.202 if called on an iterator which does not belong - to the current JSON value; example: `"iterator does not fit current - value"` - @throw invalid_iterator.205 if called on a primitive type with invalid - iterator (i.e., any iterator which is not `begin()`); example: `"iterator - out of range"` - - @complexity The complexity depends on the type: - - objects: amortized constant - - arrays: linear in distance between @a pos and the end of the container - - strings: linear in the length of the string - - other types: constant - - @liveexample{The example shows the result of `erase()` for different JSON - types.,erase__IteratorType} - - @sa @ref erase(IteratorType, IteratorType) -- removes the elements in - the given range - @sa @ref erase(const typename object_t::key_type&) -- removes the element - from an object at the given key - @sa @ref erase(const size_type) -- removes the element from an array at - the given index - - @since version 1.0.0 - */ - template::value or - std::is_same::value, int>::type - = 0> - IteratorType erase(IteratorType pos) - { - // make sure iterator fits the current value - if (this != pos.m_object) - { - JSON_THROW(invalid_iterator(202, "iterator does not fit current value")); - } - - IteratorType result = end(); - - switch (m_type) - { - case value_t::boolean: - case value_t::number_float: - case value_t::number_integer: - case value_t::number_unsigned: - case value_t::string: - { - if (not pos.m_it.primitive_iterator.is_begin()) - { - JSON_THROW(invalid_iterator(205, "iterator out of range")); - } - - if (is_string()) - { - AllocatorType alloc; - alloc.destroy(m_value.string); - alloc.deallocate(m_value.string, 1); - m_value.string = nullptr; - } - - m_type = value_t::null; - assert_invariant(); - break; - } - - case value_t::object: - { - result.m_it.object_iterator = m_value.object->erase(pos.m_it.object_iterator); - break; - } - - case value_t::array: - { - result.m_it.array_iterator = m_value.array->erase(pos.m_it.array_iterator); - break; - } - - default: - { - JSON_THROW(type_error(307, "cannot use erase() with " + type_name())); - } - } - - return result; - } - - /*! - @brief remove elements given an iterator range - - Removes the element specified by the range `[first; last)`. The iterator - @a first does not need to be dereferenceable if `first == last`: erasing - an empty range is a no-op. - - If called on a primitive type other than `null`, the resulting JSON value - will be `null`. - - @param[in] first iterator to the beginning of the range to remove - @param[in] last iterator past the end of the range to remove - @return Iterator following the last removed element. If the iterator @a - second refers to the last element, the `end()` iterator is returned. - - @tparam IteratorType an @ref iterator or @ref const_iterator - - @post Invalidates iterators and references at or after the point of the - erase, including the `end()` iterator. - - @throw type_error.307 if called on a `null` value; example: `"cannot use - erase() with null"` - @throw invalid_iterator.203 if called on iterators which does not belong - to the current JSON value; example: `"iterators do not fit current value"` - @throw invalid_iterator.204 if called on a primitive type with invalid - iterators (i.e., if `first != begin()` and `last != end()`); example: - `"iterators out of range"` - - @complexity The complexity depends on the type: - - objects: `log(size()) + std::distance(first, last)` - - arrays: linear in the distance between @a first and @a last, plus linear - in the distance between @a last and end of the container - - strings: linear in the length of the string - - other types: constant - - @liveexample{The example shows the result of `erase()` for different JSON - types.,erase__IteratorType_IteratorType} - - @sa @ref erase(IteratorType) -- removes the element at a given position - @sa @ref erase(const typename object_t::key_type&) -- removes the element - from an object at the given key - @sa @ref erase(const size_type) -- removes the element from an array at - the given index - - @since version 1.0.0 - */ - template::value or - std::is_same::value, int>::type - = 0> - IteratorType erase(IteratorType first, IteratorType last) - { - // make sure iterator fits the current value - if (this != first.m_object or this != last.m_object) - { - JSON_THROW(invalid_iterator(203, "iterators do not fit current value")); - } - - IteratorType result = end(); - - switch (m_type) - { - case value_t::boolean: - case value_t::number_float: - case value_t::number_integer: - case value_t::number_unsigned: - case value_t::string: - { - if (not first.m_it.primitive_iterator.is_begin() or not last.m_it.primitive_iterator.is_end()) - { - JSON_THROW(invalid_iterator(204, "iterators out of range")); - } - - if (is_string()) - { - AllocatorType alloc; - alloc.destroy(m_value.string); - alloc.deallocate(m_value.string, 1); - m_value.string = nullptr; - } - - m_type = value_t::null; - assert_invariant(); - break; - } - - case value_t::object: - { - result.m_it.object_iterator = m_value.object->erase(first.m_it.object_iterator, - last.m_it.object_iterator); - break; - } - - case value_t::array: - { - result.m_it.array_iterator = m_value.array->erase(first.m_it.array_iterator, - last.m_it.array_iterator); - break; - } - - default: - { - JSON_THROW(type_error(307, "cannot use erase() with " + type_name())); - } - } - - return result; - } - - /*! - @brief remove element from a JSON object given a key - - Removes elements from a JSON object with the key value @a key. - - @param[in] key value of the elements to remove - - @return Number of elements removed. If @a ObjectType is the default - `std::map` type, the return value will always be `0` (@a key was not - found) or `1` (@a key was found). - - @post References and iterators to the erased elements are invalidated. - Other references and iterators are not affected. - - @throw type_error.307 when called on a type other than JSON object; - example: `"cannot use erase() with null"` - - @complexity `log(size()) + count(key)` - - @liveexample{The example shows the effect of `erase()`.,erase__key_type} - - @sa @ref erase(IteratorType) -- removes the element at a given position - @sa @ref erase(IteratorType, IteratorType) -- removes the elements in - the given range - @sa @ref erase(const size_type) -- removes the element from an array at - the given index - - @since version 1.0.0 - */ - size_type erase(const typename object_t::key_type& key) - { - // this erase only works for objects - if (is_object()) - { - return m_value.object->erase(key); - } - - JSON_THROW(type_error(307, "cannot use erase() with " + type_name())); - } - - /*! - @brief remove element from a JSON array given an index - - Removes element from a JSON array at the index @a idx. - - @param[in] idx index of the element to remove - - @throw type_error.307 when called on a type other than JSON object; - example: `"cannot use erase() with null"` - @throw out_of_range.401 when `idx >= size()`; example: `"array index 17 - is out of range"` - - @complexity Linear in distance between @a idx and the end of the container. - - @liveexample{The example shows the effect of `erase()`.,erase__size_type} - - @sa @ref erase(IteratorType) -- removes the element at a given position - @sa @ref erase(IteratorType, IteratorType) -- removes the elements in - the given range - @sa @ref erase(const typename object_t::key_type&) -- removes the element - from an object at the given key - - @since version 1.0.0 - */ - void erase(const size_type idx) - { - // this erase only works for arrays - if (is_array()) - { - if (idx >= size()) - { - JSON_THROW(out_of_range(401, "array index " + std::to_string(idx) + " is out of range")); - } - - m_value.array->erase(m_value.array->begin() + static_cast(idx)); - } - else - { - JSON_THROW(type_error(307, "cannot use erase() with " + type_name())); - } - } - - /// @} - - - //////////// - // lookup // - //////////// - - /// @name lookup - /// @{ - - /*! - @brief find an element in a JSON object - - Finds an element in a JSON object with key equivalent to @a key. If the - element is not found or the JSON value is not an object, end() is - returned. - - @note This method always returns @ref end() when executed on a JSON type - that is not an object. - - @param[in] key key value of the element to search for - - @return Iterator to an element with key equivalent to @a key. If no such - element is found or the JSON value is not an object, past-the-end (see - @ref end()) iterator is returned. - - @complexity Logarithmic in the size of the JSON object. - - @liveexample{The example shows how `find()` is used.,find__key_type} - - @since version 1.0.0 - */ - iterator find(typename object_t::key_type key) - { - auto result = end(); - - if (is_object()) - { - result.m_it.object_iterator = m_value.object->find(key); - } - - return result; - } - - /*! - @brief find an element in a JSON object - @copydoc find(typename object_t::key_type) - */ - const_iterator find(typename object_t::key_type key) const - { - auto result = cend(); - - if (is_object()) - { - result.m_it.object_iterator = m_value.object->find(key); - } - - return result; - } - - /*! - @brief returns the number of occurrences of a key in a JSON object - - Returns the number of elements with key @a key. If ObjectType is the - default `std::map` type, the return value will always be `0` (@a key was - not found) or `1` (@a key was found). - - @note This method always returns `0` when executed on a JSON type that is - not an object. - - @param[in] key key value of the element to count - - @return Number of elements with key @a key. If the JSON value is not an - object, the return value will be `0`. - - @complexity Logarithmic in the size of the JSON object. - - @liveexample{The example shows how `count()` is used.,count} - - @since version 1.0.0 - */ - size_type count(typename object_t::key_type key) const - { - // return 0 for all nonobject types - return is_object() ? m_value.object->count(key) : 0; - } - - /// @} - - - /////////////// - // iterators // - /////////////// - - /// @name iterators - /// @{ - - /*! - @brief returns an iterator to the first element - - Returns an iterator to the first element. - - @image html range-begin-end.svg "Illustration from cppreference.com" - - @return iterator to the first element - - @complexity Constant. - - @requirement This function helps `basic_json` satisfying the - [Container](http://en.cppreference.com/w/cpp/concept/Container) - requirements: - - The complexity is constant. - - @liveexample{The following code shows an example for `begin()`.,begin} - - @sa @ref cbegin() -- returns a const iterator to the beginning - @sa @ref end() -- returns an iterator to the end - @sa @ref cend() -- returns a const iterator to the end - - @since version 1.0.0 - */ - iterator begin() noexcept - { - iterator result(this); - result.set_begin(); - return result; - } - - /*! - @copydoc basic_json::cbegin() - */ - const_iterator begin() const noexcept - { - return cbegin(); - } - - /*! - @brief returns a const iterator to the first element - - Returns a const iterator to the first element. - - @image html range-begin-end.svg "Illustration from cppreference.com" - - @return const iterator to the first element - - @complexity Constant. - - @requirement This function helps `basic_json` satisfying the - [Container](http://en.cppreference.com/w/cpp/concept/Container) - requirements: - - The complexity is constant. - - Has the semantics of `const_cast(*this).begin()`. - - @liveexample{The following code shows an example for `cbegin()`.,cbegin} - - @sa @ref begin() -- returns an iterator to the beginning - @sa @ref end() -- returns an iterator to the end - @sa @ref cend() -- returns a const iterator to the end - - @since version 1.0.0 - */ - const_iterator cbegin() const noexcept - { - const_iterator result(this); - result.set_begin(); - return result; - } - - /*! - @brief returns an iterator to one past the last element - - Returns an iterator to one past the last element. - - @image html range-begin-end.svg "Illustration from cppreference.com" - - @return iterator one past the last element - - @complexity Constant. - - @requirement This function helps `basic_json` satisfying the - [Container](http://en.cppreference.com/w/cpp/concept/Container) - requirements: - - The complexity is constant. - - @liveexample{The following code shows an example for `end()`.,end} - - @sa @ref cend() -- returns a const iterator to the end - @sa @ref begin() -- returns an iterator to the beginning - @sa @ref cbegin() -- returns a const iterator to the beginning - - @since version 1.0.0 - */ - iterator end() noexcept - { - iterator result(this); - result.set_end(); - return result; - } - - /*! - @copydoc basic_json::cend() - */ - const_iterator end() const noexcept - { - return cend(); - } - - /*! - @brief returns a const iterator to one past the last element - - Returns a const iterator to one past the last element. - - @image html range-begin-end.svg "Illustration from cppreference.com" - - @return const iterator one past the last element - - @complexity Constant. - - @requirement This function helps `basic_json` satisfying the - [Container](http://en.cppreference.com/w/cpp/concept/Container) - requirements: - - The complexity is constant. - - Has the semantics of `const_cast(*this).end()`. - - @liveexample{The following code shows an example for `cend()`.,cend} - - @sa @ref end() -- returns an iterator to the end - @sa @ref begin() -- returns an iterator to the beginning - @sa @ref cbegin() -- returns a const iterator to the beginning - - @since version 1.0.0 - */ - const_iterator cend() const noexcept - { - const_iterator result(this); - result.set_end(); - return result; - } - - /*! - @brief returns an iterator to the reverse-beginning - - Returns an iterator to the reverse-beginning; that is, the last element. - - @image html range-rbegin-rend.svg "Illustration from cppreference.com" - - @complexity Constant. - - @requirement This function helps `basic_json` satisfying the - [ReversibleContainer](http://en.cppreference.com/w/cpp/concept/ReversibleContainer) - requirements: - - The complexity is constant. - - Has the semantics of `reverse_iterator(end())`. - - @liveexample{The following code shows an example for `rbegin()`.,rbegin} - - @sa @ref crbegin() -- returns a const reverse iterator to the beginning - @sa @ref rend() -- returns a reverse iterator to the end - @sa @ref crend() -- returns a const reverse iterator to the end - - @since version 1.0.0 - */ - reverse_iterator rbegin() noexcept - { - return reverse_iterator(end()); - } - - /*! - @copydoc basic_json::crbegin() - */ - const_reverse_iterator rbegin() const noexcept - { - return crbegin(); - } - - /*! - @brief returns an iterator to the reverse-end - - Returns an iterator to the reverse-end; that is, one before the first - element. - - @image html range-rbegin-rend.svg "Illustration from cppreference.com" - - @complexity Constant. - - @requirement This function helps `basic_json` satisfying the - [ReversibleContainer](http://en.cppreference.com/w/cpp/concept/ReversibleContainer) - requirements: - - The complexity is constant. - - Has the semantics of `reverse_iterator(begin())`. - - @liveexample{The following code shows an example for `rend()`.,rend} - - @sa @ref crend() -- returns a const reverse iterator to the end - @sa @ref rbegin() -- returns a reverse iterator to the beginning - @sa @ref crbegin() -- returns a const reverse iterator to the beginning - - @since version 1.0.0 - */ - reverse_iterator rend() noexcept - { - return reverse_iterator(begin()); - } - - /*! - @copydoc basic_json::crend() - */ - const_reverse_iterator rend() const noexcept - { - return crend(); - } - - /*! - @brief returns a const reverse iterator to the last element - - Returns a const iterator to the reverse-beginning; that is, the last - element. - - @image html range-rbegin-rend.svg "Illustration from cppreference.com" - - @complexity Constant. - - @requirement This function helps `basic_json` satisfying the - [ReversibleContainer](http://en.cppreference.com/w/cpp/concept/ReversibleContainer) - requirements: - - The complexity is constant. - - Has the semantics of `const_cast(*this).rbegin()`. - - @liveexample{The following code shows an example for `crbegin()`.,crbegin} - - @sa @ref rbegin() -- returns a reverse iterator to the beginning - @sa @ref rend() -- returns a reverse iterator to the end - @sa @ref crend() -- returns a const reverse iterator to the end - - @since version 1.0.0 - */ - const_reverse_iterator crbegin() const noexcept - { - return const_reverse_iterator(cend()); - } - - /*! - @brief returns a const reverse iterator to one before the first - - Returns a const reverse iterator to the reverse-end; that is, one before - the first element. - - @image html range-rbegin-rend.svg "Illustration from cppreference.com" - - @complexity Constant. - - @requirement This function helps `basic_json` satisfying the - [ReversibleContainer](http://en.cppreference.com/w/cpp/concept/ReversibleContainer) - requirements: - - The complexity is constant. - - Has the semantics of `const_cast(*this).rend()`. - - @liveexample{The following code shows an example for `crend()`.,crend} - - @sa @ref rend() -- returns a reverse iterator to the end - @sa @ref rbegin() -- returns a reverse iterator to the beginning - @sa @ref crbegin() -- returns a const reverse iterator to the beginning - - @since version 1.0.0 - */ - const_reverse_iterator crend() const noexcept - { - return const_reverse_iterator(cbegin()); - } - - private: - // forward declaration - template class iteration_proxy; - - public: - /*! - @brief wrapper to access iterator member functions in range-based for - - This function allows to access @ref iterator::key() and @ref - iterator::value() during range-based for loops. In these loops, a - reference to the JSON values is returned, so there is no access to the - underlying iterator. - - @note The name of this function is not yet final and may change in the - future. - */ - static iteration_proxy iterator_wrapper(reference cont) - { - return iteration_proxy(cont); - } - - /*! - @copydoc iterator_wrapper(reference) - */ - static iteration_proxy iterator_wrapper(const_reference cont) - { - return iteration_proxy(cont); - } - - /// @} - - - ////////////// - // capacity // - ////////////// - - /// @name capacity - /// @{ - - /*! - @brief checks whether the container is empty - - Checks if a JSON value has no elements. - - @return The return value depends on the different types and is - defined as follows: - Value type | return value - ----------- | ------------- - null | `true` - boolean | `false` - string | `false` - number | `false` - object | result of function `object_t::empty()` - array | result of function `array_t::empty()` - - @note This function does not return whether a string stored as JSON value - is empty - it returns whether the JSON container itself is empty which is - false in the case of a string. - - @complexity Constant, as long as @ref array_t and @ref object_t satisfy - the Container concept; that is, their `empty()` functions have constant - complexity. - - @requirement This function helps `basic_json` satisfying the - [Container](http://en.cppreference.com/w/cpp/concept/Container) - requirements: - - The complexity is constant. - - Has the semantics of `begin() == end()`. - - @liveexample{The following code uses `empty()` to check if a JSON - object contains any elements.,empty} - - @sa @ref size() -- returns the number of elements - - @since version 1.0.0 - */ - bool empty() const noexcept - { - switch (m_type) - { - case value_t::null: - { - // null values are empty - return true; - } - - case value_t::array: - { - // delegate call to array_t::empty() - return m_value.array->empty(); - } - - case value_t::object: - { - // delegate call to object_t::empty() - return m_value.object->empty(); - } - - default: - { - // all other types are nonempty - return false; - } - } - } - - /*! - @brief returns the number of elements - - Returns the number of elements in a JSON value. - - @return The return value depends on the different types and is - defined as follows: - Value type | return value - ----------- | ------------- - null | `0` - boolean | `1` - string | `1` - number | `1` - object | result of function object_t::size() - array | result of function array_t::size() - - @note This function does not return the length of a string stored as JSON - value - it returns the number of elements in the JSON value which is 1 in - the case of a string. - - @complexity Constant, as long as @ref array_t and @ref object_t satisfy - the Container concept; that is, their size() functions have constant - complexity. - - @requirement This function helps `basic_json` satisfying the - [Container](http://en.cppreference.com/w/cpp/concept/Container) - requirements: - - The complexity is constant. - - Has the semantics of `std::distance(begin(), end())`. - - @liveexample{The following code calls `size()` on the different value - types.,size} - - @sa @ref empty() -- checks whether the container is empty - @sa @ref max_size() -- returns the maximal number of elements - - @since version 1.0.0 - */ - size_type size() const noexcept - { - switch (m_type) - { - case value_t::null: - { - // null values are empty - return 0; - } - - case value_t::array: - { - // delegate call to array_t::size() - return m_value.array->size(); - } - - case value_t::object: - { - // delegate call to object_t::size() - return m_value.object->size(); - } - - default: - { - // all other types have size 1 - return 1; - } - } - } - - /*! - @brief returns the maximum possible number of elements - - Returns the maximum number of elements a JSON value is able to hold due to - system or library implementation limitations, i.e. `std::distance(begin(), - end())` for the JSON value. - - @return The return value depends on the different types and is - defined as follows: - Value type | return value - ----------- | ------------- - null | `0` (same as `size()`) - boolean | `1` (same as `size()`) - string | `1` (same as `size()`) - number | `1` (same as `size()`) - object | result of function `object_t::max_size()` - array | result of function `array_t::max_size()` - - @complexity Constant, as long as @ref array_t and @ref object_t satisfy - the Container concept; that is, their `max_size()` functions have constant - complexity. - - @requirement This function helps `basic_json` satisfying the - [Container](http://en.cppreference.com/w/cpp/concept/Container) - requirements: - - The complexity is constant. - - Has the semantics of returning `b.size()` where `b` is the largest - possible JSON value. - - @liveexample{The following code calls `max_size()` on the different value - types. Note the output is implementation specific.,max_size} - - @sa @ref size() -- returns the number of elements - - @since version 1.0.0 - */ - size_type max_size() const noexcept - { - switch (m_type) - { - case value_t::array: - { - // delegate call to array_t::max_size() - return m_value.array->max_size(); - } - - case value_t::object: - { - // delegate call to object_t::max_size() - return m_value.object->max_size(); - } - - default: - { - // all other types have max_size() == size() - return size(); - } - } - } - - /// @} - - - /////////////// - // modifiers // - /////////////// - - /// @name modifiers - /// @{ - - /*! - @brief clears the contents - - Clears the content of a JSON value and resets it to the default value as - if @ref basic_json(value_t) would have been called: - - Value type | initial value - ----------- | ------------- - null | `null` - boolean | `false` - string | `""` - number | `0` - object | `{}` - array | `[]` - - @complexity Linear in the size of the JSON value. - - @liveexample{The example below shows the effect of `clear()` to different - JSON types.,clear} - - @since version 1.0.0 - */ - void clear() noexcept - { - switch (m_type) - { - case value_t::number_integer: - { - m_value.number_integer = 0; - break; - } - - case value_t::number_unsigned: - { - m_value.number_unsigned = 0; - break; - } - - case value_t::number_float: - { - m_value.number_float = 0.0; - break; - } - - case value_t::boolean: - { - m_value.boolean = false; - break; - } - - case value_t::string: - { - m_value.string->clear(); - break; - } - - case value_t::array: - { - m_value.array->clear(); - break; - } - - case value_t::object: - { - m_value.object->clear(); - break; - } - - default: - { - break; - } - } - } - - /*! - @brief add an object to an array - - Appends the given element @a val to the end of the JSON value. If the - function is called on a JSON null value, an empty array is created before - appending @a val. - - @param[in] val the value to add to the JSON array - - @throw type_error.308 when called on a type other than JSON array or - null; example: `"cannot use push_back() with number"` - - @complexity Amortized constant. - - @liveexample{The example shows how `push_back()` and `+=` can be used to - add elements to a JSON array. Note how the `null` value was silently - converted to a JSON array.,push_back} - - @since version 1.0.0 - */ - void push_back(basic_json&& val) - { - // push_back only works for null objects or arrays - if (not(is_null() or is_array())) - { - JSON_THROW(type_error(308, "cannot use push_back() with " + type_name())); - } - - // transform null object into an array - if (is_null()) - { - m_type = value_t::array; - m_value = value_t::array; - assert_invariant(); - } - - // add element to array (move semantics) - m_value.array->push_back(std::move(val)); - // invalidate object - val.m_type = value_t::null; - } - - /*! - @brief add an object to an array - @copydoc push_back(basic_json&&) - */ - reference operator+=(basic_json&& val) - { - push_back(std::move(val)); - return *this; - } - - /*! - @brief add an object to an array - @copydoc push_back(basic_json&&) - */ - void push_back(const basic_json& val) - { - // push_back only works for null objects or arrays - if (not(is_null() or is_array())) - { - JSON_THROW(type_error(308, "cannot use push_back() with " + type_name())); - } - - // transform null object into an array - if (is_null()) - { - m_type = value_t::array; - m_value = value_t::array; - assert_invariant(); - } - - // add element to array - m_value.array->push_back(val); - } - - /*! - @brief add an object to an array - @copydoc push_back(basic_json&&) - */ - reference operator+=(const basic_json& val) - { - push_back(val); - return *this; - } - - /*! - @brief add an object to an object - - Inserts the given element @a val to the JSON object. If the function is - called on a JSON null value, an empty object is created before inserting - @a val. - - @param[in] val the value to add to the JSON object - - @throw type_error.308 when called on a type other than JSON object or - null; example: `"cannot use push_back() with number"` - - @complexity Logarithmic in the size of the container, O(log(`size()`)). - - @liveexample{The example shows how `push_back()` and `+=` can be used to - add elements to a JSON object. Note how the `null` value was silently - converted to a JSON object.,push_back__object_t__value} - - @since version 1.0.0 - */ - void push_back(const typename object_t::value_type& val) - { - // push_back only works for null objects or objects - if (not(is_null() or is_object())) - { - JSON_THROW(type_error(308, "cannot use push_back() with " + type_name())); - } - - // transform null object into an object - if (is_null()) - { - m_type = value_t::object; - m_value = value_t::object; - assert_invariant(); - } - - // add element to array - m_value.object->insert(val); - } - - /*! - @brief add an object to an object - @copydoc push_back(const typename object_t::value_type&) - */ - reference operator+=(const typename object_t::value_type& val) - { - push_back(val); - return *this; - } - - /*! - @brief add an object to an object - - This function allows to use `push_back` with an initializer list. In case - - 1. the current value is an object, - 2. the initializer list @a init contains only two elements, and - 3. the first element of @a init is a string, - - @a init is converted into an object element and added using - @ref push_back(const typename object_t::value_type&). Otherwise, @a init - is converted to a JSON value and added using @ref push_back(basic_json&&). - - @param init an initializer list - - @complexity Linear in the size of the initializer list @a init. - - @note This function is required to resolve an ambiguous overload error, - because pairs like `{"key", "value"}` can be both interpreted as - `object_t::value_type` or `std::initializer_list`, see - https://github.com/nlohmann/json/issues/235 for more information. - - @liveexample{The example shows how initializer lists are treated as - objects when possible.,push_back__initializer_list} - */ - void push_back(std::initializer_list init) - { - if (is_object() and init.size() == 2 and init.begin()->is_string()) - { - const string_t key = *init.begin(); - push_back(typename object_t::value_type(key, *(init.begin() + 1))); - } - else - { - push_back(basic_json(init)); - } - } - - /*! - @brief add an object to an object - @copydoc push_back(std::initializer_list) - */ - reference operator+=(std::initializer_list init) - { - push_back(init); - return *this; - } - - /*! - @brief add an object to an array - - Creates a JSON value from the passed parameters @a args to the end of the - JSON value. If the function is called on a JSON null value, an empty array - is created before appending the value created from @a args. - - @param[in] args arguments to forward to a constructor of @ref basic_json - @tparam Args compatible types to create a @ref basic_json object - - @throw type_error.311 when called on a type other than JSON array or - null; example: `"cannot use emplace_back() with number"` - - @complexity Amortized constant. - - @liveexample{The example shows how `push_back()` can be used to add - elements to a JSON array. Note how the `null` value was silently converted - to a JSON array.,emplace_back} - - @since version 2.0.8 - */ - template - void emplace_back(Args&& ... args) - { - // emplace_back only works for null objects or arrays - if (not(is_null() or is_array())) - { - JSON_THROW(type_error(311, "cannot use emplace_back() with " + type_name())); - } - - // transform null object into an array - if (is_null()) - { - m_type = value_t::array; - m_value = value_t::array; - assert_invariant(); - } - - // add element to array (perfect forwarding) - m_value.array->emplace_back(std::forward(args)...); - } - - /*! - @brief add an object to an object if key does not exist - - Inserts a new element into a JSON object constructed in-place with the - given @a args if there is no element with the key in the container. If the - function is called on a JSON null value, an empty object is created before - appending the value created from @a args. - - @param[in] args arguments to forward to a constructor of @ref basic_json - @tparam Args compatible types to create a @ref basic_json object - - @return a pair consisting of an iterator to the inserted element, or the - already-existing element if no insertion happened, and a bool - denoting whether the insertion took place. - - @throw type_error.311 when called on a type other than JSON object or - null; example: `"cannot use emplace() with number"` - - @complexity Logarithmic in the size of the container, O(log(`size()`)). - - @liveexample{The example shows how `emplace()` can be used to add elements - to a JSON object. Note how the `null` value was silently converted to a - JSON object. Further note how no value is added if there was already one - value stored with the same key.,emplace} - - @since version 2.0.8 - */ - template - std::pair emplace(Args&& ... args) - { - // emplace only works for null objects or arrays - if (not(is_null() or is_object())) - { - JSON_THROW(type_error(311, "cannot use emplace() with " + type_name())); - } - - // transform null object into an object - if (is_null()) - { - m_type = value_t::object; - m_value = value_t::object; - assert_invariant(); - } - - // add element to array (perfect forwarding) - auto res = m_value.object->emplace(std::forward(args)...); - // create result iterator and set iterator to the result of emplace - auto it = begin(); - it.m_it.object_iterator = res.first; - - // return pair of iterator and boolean - return {it, res.second}; - } - - /*! - @brief inserts element - - Inserts element @a val before iterator @a pos. - - @param[in] pos iterator before which the content will be inserted; may be - the end() iterator - @param[in] val element to insert - @return iterator pointing to the inserted @a val. - - @throw type_error.309 if called on JSON values other than arrays; - example: `"cannot use insert() with string"` - @throw invalid_iterator.202 if @a pos is not an iterator of *this; - example: `"iterator does not fit current value"` - - @complexity Constant plus linear in the distance between @a pos and end of - the container. - - @liveexample{The example shows how `insert()` is used.,insert} - - @since version 1.0.0 - */ - iterator insert(const_iterator pos, const basic_json& val) - { - // insert only works for arrays - if (is_array()) - { - // check if iterator pos fits to this JSON value - if (pos.m_object != this) - { - JSON_THROW(invalid_iterator(202, "iterator does not fit current value")); - } - - // insert to array and return iterator - iterator result(this); - result.m_it.array_iterator = m_value.array->insert(pos.m_it.array_iterator, val); - return result; - } - - JSON_THROW(type_error(309, "cannot use insert() with " + type_name())); - } - - /*! - @brief inserts element - @copydoc insert(const_iterator, const basic_json&) - */ - iterator insert(const_iterator pos, basic_json&& val) - { - return insert(pos, val); - } - - /*! - @brief inserts elements - - Inserts @a cnt copies of @a val before iterator @a pos. - - @param[in] pos iterator before which the content will be inserted; may be - the end() iterator - @param[in] cnt number of copies of @a val to insert - @param[in] val element to insert - @return iterator pointing to the first element inserted, or @a pos if - `cnt==0` - - @throw type_error.309 if called on JSON values other than arrays; example: - `"cannot use insert() with string"` - @throw invalid_iterator.202 if @a pos is not an iterator of *this; - example: `"iterator does not fit current value"` - - @complexity Linear in @a cnt plus linear in the distance between @a pos - and end of the container. - - @liveexample{The example shows how `insert()` is used.,insert__count} - - @since version 1.0.0 - */ - iterator insert(const_iterator pos, size_type cnt, const basic_json& val) - { - // insert only works for arrays - if (is_array()) - { - // check if iterator pos fits to this JSON value - if (pos.m_object != this) - { - JSON_THROW(invalid_iterator(202, "iterator does not fit current value")); - } - - // insert to array and return iterator - iterator result(this); - result.m_it.array_iterator = m_value.array->insert(pos.m_it.array_iterator, cnt, val); - return result; - } - - JSON_THROW(type_error(309, "cannot use insert() with " + type_name())); - } - - /*! - @brief inserts elements - - Inserts elements from range `[first, last)` before iterator @a pos. - - @param[in] pos iterator before which the content will be inserted; may be - the end() iterator - @param[in] first begin of the range of elements to insert - @param[in] last end of the range of elements to insert - - @throw type_error.309 if called on JSON values other than arrays; example: - `"cannot use insert() with string"` - @throw invalid_iterator.202 if @a pos is not an iterator of *this; - example: `"iterator does not fit current value"` - @throw invalid_iterator.210 if @a first and @a last do not belong to the - same JSON value; example: `"iterators do not fit"` - @throw invalid_iterator.211 if @a first or @a last are iterators into - container for which insert is called; example: `"passed iterators may not - belong to container"` - - @return iterator pointing to the first element inserted, or @a pos if - `first==last` - - @complexity Linear in `std::distance(first, last)` plus linear in the - distance between @a pos and end of the container. - - @liveexample{The example shows how `insert()` is used.,insert__range} - - @since version 1.0.0 - */ - iterator insert(const_iterator pos, const_iterator first, const_iterator last) - { - // insert only works for arrays - if (not is_array()) - { - JSON_THROW(type_error(309, "cannot use insert() with " + type_name())); - } - - // check if iterator pos fits to this JSON value - if (pos.m_object != this) - { - JSON_THROW(invalid_iterator(202, "iterator does not fit current value")); - } - - // check if range iterators belong to the same JSON object - if (first.m_object != last.m_object) - { - JSON_THROW(invalid_iterator(210, "iterators do not fit")); - } - - if (first.m_object == this or last.m_object == this) - { - JSON_THROW(invalid_iterator(211, "passed iterators may not belong to container")); - } - - // insert to array and return iterator - iterator result(this); - result.m_it.array_iterator = m_value.array->insert( - pos.m_it.array_iterator, - first.m_it.array_iterator, - last.m_it.array_iterator); - return result; - } - - /*! - @brief inserts elements - - Inserts elements from initializer list @a ilist before iterator @a pos. - - @param[in] pos iterator before which the content will be inserted; may be - the end() iterator - @param[in] ilist initializer list to insert the values from - - @throw type_error.309 if called on JSON values other than arrays; example: - `"cannot use insert() with string"` - @throw invalid_iterator.202 if @a pos is not an iterator of *this; - example: `"iterator does not fit current value"` - - @return iterator pointing to the first element inserted, or @a pos if - `ilist` is empty - - @complexity Linear in `ilist.size()` plus linear in the distance between - @a pos and end of the container. - - @liveexample{The example shows how `insert()` is used.,insert__ilist} - - @since version 1.0.0 - */ - iterator insert(const_iterator pos, std::initializer_list ilist) - { - // insert only works for arrays - if (not is_array()) - { - JSON_THROW(type_error(309, "cannot use insert() with " + type_name())); - } - - // check if iterator pos fits to this JSON value - if (pos.m_object != this) - { - JSON_THROW(invalid_iterator(202, "iterator does not fit current value")); - } - - // insert to array and return iterator - iterator result(this); - result.m_it.array_iterator = m_value.array->insert(pos.m_it.array_iterator, ilist); - return result; - } - - /*! - @brief exchanges the values - - Exchanges the contents of the JSON value with those of @a other. Does not - invoke any move, copy, or swap operations on individual elements. All - iterators and references remain valid. The past-the-end iterator is - invalidated. - - @param[in,out] other JSON value to exchange the contents with - - @complexity Constant. - - @liveexample{The example below shows how JSON values can be swapped with - `swap()`.,swap__reference} - - @since version 1.0.0 - */ - void swap(reference other) noexcept ( - std::is_nothrow_move_constructible::value and - std::is_nothrow_move_assignable::value and - std::is_nothrow_move_constructible::value and - std::is_nothrow_move_assignable::value - ) - { - std::swap(m_type, other.m_type); - std::swap(m_value, other.m_value); - assert_invariant(); - } - - /*! - @brief exchanges the values - - Exchanges the contents of a JSON array with those of @a other. Does not - invoke any move, copy, or swap operations on individual elements. All - iterators and references remain valid. The past-the-end iterator is - invalidated. - - @param[in,out] other array to exchange the contents with - - @throw type_error.310 when JSON value is not an array; example: `"cannot - use swap() with string"` - - @complexity Constant. - - @liveexample{The example below shows how arrays can be swapped with - `swap()`.,swap__array_t} - - @since version 1.0.0 - */ - void swap(array_t& other) - { - // swap only works for arrays - if (is_array()) - { - std::swap(*(m_value.array), other); - } - else - { - JSON_THROW(type_error(310, "cannot use swap() with " + type_name())); - } - } - - /*! - @brief exchanges the values - - Exchanges the contents of a JSON object with those of @a other. Does not - invoke any move, copy, or swap operations on individual elements. All - iterators and references remain valid. The past-the-end iterator is - invalidated. - - @param[in,out] other object to exchange the contents with - - @throw type_error.310 when JSON value is not an object; example: - `"cannot use swap() with string"` - - @complexity Constant. - - @liveexample{The example below shows how objects can be swapped with - `swap()`.,swap__object_t} - - @since version 1.0.0 - */ - void swap(object_t& other) - { - // swap only works for objects - if (is_object()) - { - std::swap(*(m_value.object), other); - } - else - { - JSON_THROW(type_error(310, "cannot use swap() with " + type_name())); - } - } - - /*! - @brief exchanges the values - - Exchanges the contents of a JSON string with those of @a other. Does not - invoke any move, copy, or swap operations on individual elements. All - iterators and references remain valid. The past-the-end iterator is - invalidated. - - @param[in,out] other string to exchange the contents with - - @throw type_error.310 when JSON value is not a string; example: `"cannot - use swap() with boolean"` - - @complexity Constant. - - @liveexample{The example below shows how strings can be swapped with - `swap()`.,swap__string_t} - - @since version 1.0.0 - */ - void swap(string_t& other) - { - // swap only works for strings - if (is_string()) - { - std::swap(*(m_value.string), other); - } - else - { - JSON_THROW(type_error(310, "cannot use swap() with " + type_name())); - } - } - - /// @} - - public: - ////////////////////////////////////////// - // lexicographical comparison operators // - ////////////////////////////////////////// - - /// @name lexicographical comparison operators - /// @{ - - /*! - @brief comparison: equal - - Compares two JSON values for equality according to the following rules: - - Two JSON values are equal if (1) they are from the same type and (2) - their stored values are the same. - - Integer and floating-point numbers are automatically converted before - comparison. Floating-point numbers are compared indirectly: two - floating-point numbers `f1` and `f2` are considered equal if neither - `f1 > f2` nor `f2 > f1` holds. - - Two JSON null values are equal. - - @param[in] lhs first JSON value to consider - @param[in] rhs second JSON value to consider - @return whether the values @a lhs and @a rhs are equal - - @complexity Linear. - - @liveexample{The example demonstrates comparing several JSON - types.,operator__equal} - - @since version 1.0.0 - */ - friend bool operator==(const_reference lhs, const_reference rhs) noexcept - { - const auto lhs_type = lhs.type(); - const auto rhs_type = rhs.type(); - - if (lhs_type == rhs_type) - { - switch (lhs_type) - { - case value_t::array: - { - return *lhs.m_value.array == *rhs.m_value.array; - } - case value_t::object: - { - return *lhs.m_value.object == *rhs.m_value.object; - } - case value_t::null: - { - return true; - } - case value_t::string: - { - return *lhs.m_value.string == *rhs.m_value.string; - } - case value_t::boolean: - { - return lhs.m_value.boolean == rhs.m_value.boolean; - } - case value_t::number_integer: - { - return lhs.m_value.number_integer == rhs.m_value.number_integer; - } - case value_t::number_unsigned: - { - return lhs.m_value.number_unsigned == rhs.m_value.number_unsigned; - } - case value_t::number_float: - { - return lhs.m_value.number_float == rhs.m_value.number_float; - } - default: - { - return false; - } - } - } - else if (lhs_type == value_t::number_integer and rhs_type == value_t::number_float) - { - return static_cast(lhs.m_value.number_integer) == rhs.m_value.number_float; - } - else if (lhs_type == value_t::number_float and rhs_type == value_t::number_integer) - { - return lhs.m_value.number_float == static_cast(rhs.m_value.number_integer); - } - else if (lhs_type == value_t::number_unsigned and rhs_type == value_t::number_float) - { - return static_cast(lhs.m_value.number_unsigned) == rhs.m_value.number_float; - } - else if (lhs_type == value_t::number_float and rhs_type == value_t::number_unsigned) - { - return lhs.m_value.number_float == static_cast(rhs.m_value.number_unsigned); - } - else if (lhs_type == value_t::number_unsigned and rhs_type == value_t::number_integer) - { - return static_cast(lhs.m_value.number_unsigned) == rhs.m_value.number_integer; - } - else if (lhs_type == value_t::number_integer and rhs_type == value_t::number_unsigned) - { - return lhs.m_value.number_integer == static_cast(rhs.m_value.number_unsigned); - } - - return false; - } - - /*! - @brief comparison: equal - @copydoc operator==(const_reference, const_reference) - */ - template::value, int>::type = 0> - friend bool operator==(const_reference lhs, const ScalarType rhs) noexcept - { - return (lhs == basic_json(rhs)); - } - - /*! - @brief comparison: equal - @copydoc operator==(const_reference, const_reference) - */ - template::value, int>::type = 0> - friend bool operator==(const ScalarType lhs, const_reference rhs) noexcept - { - return (basic_json(lhs) == rhs); - } - - /*! - @brief comparison: not equal - - Compares two JSON values for inequality by calculating `not (lhs == rhs)`. - - @param[in] lhs first JSON value to consider - @param[in] rhs second JSON value to consider - @return whether the values @a lhs and @a rhs are not equal - - @complexity Linear. - - @liveexample{The example demonstrates comparing several JSON - types.,operator__notequal} - - @since version 1.0.0 - */ - friend bool operator!=(const_reference lhs, const_reference rhs) noexcept - { - return not (lhs == rhs); - } - - /*! - @brief comparison: not equal - @copydoc operator!=(const_reference, const_reference) - */ - template::value, int>::type = 0> - friend bool operator!=(const_reference lhs, const ScalarType rhs) noexcept - { - return (lhs != basic_json(rhs)); - } - - /*! - @brief comparison: not equal - @copydoc operator!=(const_reference, const_reference) - */ - template::value, int>::type = 0> - friend bool operator!=(const ScalarType lhs, const_reference rhs) noexcept - { - return (basic_json(lhs) != rhs); - } - - /*! - @brief comparison: less than - - Compares whether one JSON value @a lhs is less than another JSON value @a - rhs according to the following rules: - - If @a lhs and @a rhs have the same type, the values are compared using - the default `<` operator. - - Integer and floating-point numbers are automatically converted before - comparison - - In case @a lhs and @a rhs have different types, the values are ignored - and the order of the types is considered, see - @ref operator<(const value_t, const value_t). - - @param[in] lhs first JSON value to consider - @param[in] rhs second JSON value to consider - @return whether @a lhs is less than @a rhs - - @complexity Linear. - - @liveexample{The example demonstrates comparing several JSON - types.,operator__less} - - @since version 1.0.0 - */ - friend bool operator<(const_reference lhs, const_reference rhs) noexcept - { - const auto lhs_type = lhs.type(); - const auto rhs_type = rhs.type(); - - if (lhs_type == rhs_type) - { - switch (lhs_type) - { - case value_t::array: - { - return *lhs.m_value.array < *rhs.m_value.array; - } - case value_t::object: - { - return *lhs.m_value.object < *rhs.m_value.object; - } - case value_t::null: - { - return false; - } - case value_t::string: - { - return *lhs.m_value.string < *rhs.m_value.string; - } - case value_t::boolean: - { - return lhs.m_value.boolean < rhs.m_value.boolean; - } - case value_t::number_integer: - { - return lhs.m_value.number_integer < rhs.m_value.number_integer; - } - case value_t::number_unsigned: - { - return lhs.m_value.number_unsigned < rhs.m_value.number_unsigned; - } - case value_t::number_float: - { - return lhs.m_value.number_float < rhs.m_value.number_float; - } - default: - { - return false; - } - } - } - else if (lhs_type == value_t::number_integer and rhs_type == value_t::number_float) - { - return static_cast(lhs.m_value.number_integer) < rhs.m_value.number_float; - } - else if (lhs_type == value_t::number_float and rhs_type == value_t::number_integer) - { - return lhs.m_value.number_float < static_cast(rhs.m_value.number_integer); - } - else if (lhs_type == value_t::number_unsigned and rhs_type == value_t::number_float) - { - return static_cast(lhs.m_value.number_unsigned) < rhs.m_value.number_float; - } - else if (lhs_type == value_t::number_float and rhs_type == value_t::number_unsigned) - { - return lhs.m_value.number_float < static_cast(rhs.m_value.number_unsigned); - } - else if (lhs_type == value_t::number_integer and rhs_type == value_t::number_unsigned) - { - return lhs.m_value.number_integer < static_cast(rhs.m_value.number_unsigned); - } - else if (lhs_type == value_t::number_unsigned and rhs_type == value_t::number_integer) - { - return static_cast(lhs.m_value.number_unsigned) < rhs.m_value.number_integer; - } - - // We only reach this line if we cannot compare values. In that case, - // we compare types. Note we have to call the operator explicitly, - // because MSVC has problems otherwise. - return operator<(lhs_type, rhs_type); - } - - /*! - @brief comparison: less than or equal - - Compares whether one JSON value @a lhs is less than or equal to another - JSON value by calculating `not (rhs < lhs)`. - - @param[in] lhs first JSON value to consider - @param[in] rhs second JSON value to consider - @return whether @a lhs is less than or equal to @a rhs - - @complexity Linear. - - @liveexample{The example demonstrates comparing several JSON - types.,operator__greater} - - @since version 1.0.0 - */ - friend bool operator<=(const_reference lhs, const_reference rhs) noexcept - { - return not (rhs < lhs); - } - - /*! - @brief comparison: greater than - - Compares whether one JSON value @a lhs is greater than another - JSON value by calculating `not (lhs <= rhs)`. - - @param[in] lhs first JSON value to consider - @param[in] rhs second JSON value to consider - @return whether @a lhs is greater than to @a rhs - - @complexity Linear. - - @liveexample{The example demonstrates comparing several JSON - types.,operator__lessequal} - - @since version 1.0.0 - */ - friend bool operator>(const_reference lhs, const_reference rhs) noexcept - { - return not (lhs <= rhs); - } - - /*! - @brief comparison: greater than or equal - - Compares whether one JSON value @a lhs is greater than or equal to another - JSON value by calculating `not (lhs < rhs)`. - - @param[in] lhs first JSON value to consider - @param[in] rhs second JSON value to consider - @return whether @a lhs is greater than or equal to @a rhs - - @complexity Linear. - - @liveexample{The example demonstrates comparing several JSON - types.,operator__greaterequal} - - @since version 1.0.0 - */ - friend bool operator>=(const_reference lhs, const_reference rhs) noexcept - { - return not (lhs < rhs); - } - - /// @} - - - /////////////////// - // serialization // - /////////////////// - - /// @name serialization - /// @{ - - private: - /*! - @brief wrapper around the serialization functions - */ - class serializer - { - private: - serializer(const serializer&) = delete; - serializer& operator=(const serializer&) = delete; - - public: - /*! - @param[in] s output stream to serialize to - */ - serializer(std::ostream& s) - : o(s), loc(std::localeconv()), - thousands_sep(!loc->thousands_sep ? '\0' : loc->thousands_sep[0]), - decimal_point(!loc->decimal_point ? '\0' : loc->decimal_point[0]) - {} - - /*! - @brief internal implementation of the serialization function - - This function is called by the public member function dump and - organizes the serialization internally. The indentation level is - propagated as additional parameter. In case of arrays and objects, the - function is called recursively. - - - strings and object keys are escaped using `escape_string()` - - integer numbers are converted implicitly via `operator<<` - - floating-point numbers are converted to a string using `"%g"` format - - @param[in] val value to serialize - @param[in] pretty_print whether the output shall be pretty-printed - @param[in] indent_step the indent level - @param[in] current_indent the current indent level (only used internally) - */ - void dump(const basic_json& val, - const bool pretty_print, - const unsigned int indent_step, - const unsigned int current_indent = 0) - { - switch (val.m_type) - { - case value_t::object: - { - if (val.m_value.object->empty()) - { - o.write("{}", 2); - return; - } - - if (pretty_print) - { - o.write("{\n", 2); - - // variable to hold indentation for recursive calls - const auto new_indent = current_indent + indent_step; - if (indent_string.size() < new_indent) - { - indent_string.resize(new_indent, ' '); - } - - // first n-1 elements - auto i = val.m_value.object->cbegin(); - for (size_t cnt = 0; cnt < val.m_value.object->size() - 1; ++cnt, ++i) - { - o.write(indent_string.c_str(), static_cast(new_indent)); - o.put('\"'); - dump_escaped(i->first); - o.write("\": ", 3); - dump(i->second, true, indent_step, new_indent); - o.write(",\n", 2); - } - - // last element - assert(i != val.m_value.object->cend()); - o.write(indent_string.c_str(), static_cast(new_indent)); - o.put('\"'); - dump_escaped(i->first); - o.write("\": ", 3); - dump(i->second, true, indent_step, new_indent); - - o.put('\n'); - o.write(indent_string.c_str(), static_cast(current_indent)); - o.put('}'); - } - else - { - o.put('{'); - - // first n-1 elements - auto i = val.m_value.object->cbegin(); - for (size_t cnt = 0; cnt < val.m_value.object->size() - 1; ++cnt, ++i) - { - o.put('\"'); - dump_escaped(i->first); - o.write("\":", 2); - dump(i->second, false, indent_step, current_indent); - o.put(','); - } - - // last element - assert(i != val.m_value.object->cend()); - o.put('\"'); - dump_escaped(i->first); - o.write("\":", 2); - dump(i->second, false, indent_step, current_indent); - - o.put('}'); - } - - return; - } - - case value_t::array: - { - if (val.m_value.array->empty()) - { - o.write("[]", 2); - return; - } - - if (pretty_print) - { - o.write("[\n", 2); - - // variable to hold indentation for recursive calls - const auto new_indent = current_indent + indent_step; - if (indent_string.size() < new_indent) - { - indent_string.resize(new_indent, ' '); - } - - // first n-1 elements - for (auto i = val.m_value.array->cbegin(); i != val.m_value.array->cend() - 1; ++i) - { - o.write(indent_string.c_str(), static_cast(new_indent)); - dump(*i, true, indent_step, new_indent); - o.write(",\n", 2); - } - - // last element - assert(not val.m_value.array->empty()); - o.write(indent_string.c_str(), static_cast(new_indent)); - dump(val.m_value.array->back(), true, indent_step, new_indent); - - o.put('\n'); - o.write(indent_string.c_str(), static_cast(current_indent)); - o.put(']'); - } - else - { - o.put('['); - - // first n-1 elements - for (auto i = val.m_value.array->cbegin(); i != val.m_value.array->cend() - 1; ++i) - { - dump(*i, false, indent_step, current_indent); - o.put(','); - } - - // last element - assert(not val.m_value.array->empty()); - dump(val.m_value.array->back(), false, indent_step, current_indent); - - o.put(']'); - } - - return; - } - - case value_t::string: - { - o.put('\"'); - dump_escaped(*val.m_value.string); - o.put('\"'); - return; - } - - case value_t::boolean: - { - if (val.m_value.boolean) - { - o.write("true", 4); - } - else - { - o.write("false", 5); - } - return; - } - - case value_t::number_integer: - { - dump_integer(val.m_value.number_integer); - return; - } - - case value_t::number_unsigned: - { - dump_integer(val.m_value.number_unsigned); - return; - } - - case value_t::number_float: - { - dump_float(val.m_value.number_float); - return; - } - - case value_t::discarded: - { - o.write("", 11); - return; - } - - case value_t::null: - { - o.write("null", 4); - return; - } - } - } - - private: - /*! - @brief calculates the extra space to escape a JSON string - - @param[in] s the string to escape - @return the number of characters required to escape string @a s - - @complexity Linear in the length of string @a s. - */ - static std::size_t extra_space(const string_t& s) noexcept - { - return std::accumulate(s.begin(), s.end(), size_t{}, - [](size_t res, typename string_t::value_type c) - { - switch (c) - { - case '"': - case '\\': - case '\b': - case '\f': - case '\n': - case '\r': - case '\t': - { - // from c (1 byte) to \x (2 bytes) - return res + 1; - } - - case 0x00: - case 0x01: - case 0x02: - case 0x03: - case 0x04: - case 0x05: - case 0x06: - case 0x07: - case 0x0b: - case 0x0e: - case 0x0f: - case 0x10: - case 0x11: - case 0x12: - case 0x13: - case 0x14: - case 0x15: - case 0x16: - case 0x17: - case 0x18: - case 0x19: - case 0x1a: - case 0x1b: - case 0x1c: - case 0x1d: - case 0x1e: - case 0x1f: - { - // from c (1 byte) to \uxxxx (6 bytes) - return res + 5; - } - - default: - { - return res; - } - } - }); - } - - /*! - @brief dump escaped string - - Escape a string by replacing certain special characters by a sequence - of an escape character (backslash) and another character and other - control characters by a sequence of "\u" followed by a four-digit hex - representation. The escaped string is written to output stream @a o. - - @param[in] s the string to escape - - @complexity Linear in the length of string @a s. - */ - void dump_escaped(const string_t& s) const - { - const auto space = extra_space(s); - if (space == 0) - { - o.write(s.c_str(), static_cast(s.size())); - return; - } - - // create a result string of necessary size - string_t result(s.size() + space, '\\'); - std::size_t pos = 0; - - for (const auto& c : s) - { - switch (c) - { - // quotation mark (0x22) - case '"': - { - result[pos + 1] = '"'; - pos += 2; - break; - } - - // reverse solidus (0x5c) - case '\\': - { - // nothing to change - pos += 2; - break; - } - - // backspace (0x08) - case '\b': - { - result[pos + 1] = 'b'; - pos += 2; - break; - } - - // formfeed (0x0c) - case '\f': - { - result[pos + 1] = 'f'; - pos += 2; - break; - } - - // newline (0x0a) - case '\n': - { - result[pos + 1] = 'n'; - pos += 2; - break; - } - - // carriage return (0x0d) - case '\r': - { - result[pos + 1] = 'r'; - pos += 2; - break; - } - - // horizontal tab (0x09) - case '\t': - { - result[pos + 1] = 't'; - pos += 2; - break; - } - - case 0x00: - case 0x01: - case 0x02: - case 0x03: - case 0x04: - case 0x05: - case 0x06: - case 0x07: - case 0x0b: - case 0x0e: - case 0x0f: - case 0x10: - case 0x11: - case 0x12: - case 0x13: - case 0x14: - case 0x15: - case 0x16: - case 0x17: - case 0x18: - case 0x19: - case 0x1a: - case 0x1b: - case 0x1c: - case 0x1d: - case 0x1e: - case 0x1f: - { - // convert a number 0..15 to its hex representation - // (0..f) - static const char hexify[16] = - { - '0', '1', '2', '3', '4', '5', '6', '7', - '8', '9', 'a', 'b', 'c', 'd', 'e', 'f' - }; - - // print character c as \uxxxx - for (const char m : - { 'u', '0', '0', hexify[c >> 4], hexify[c & 0x0f] - }) - { - result[++pos] = m; - } - - ++pos; - break; - } - - default: - { - // all other characters are added as-is - result[pos++] = c; - break; - } - } - } - - assert(pos == s.size() + space); - o.write(result.c_str(), static_cast(result.size())); - } - - /*! - @brief dump an integer - - Dump a given integer to output stream @a o. Works internally with - @a number_buffer. - - @param[in] x integer number (signed or unsigned) to dump - @tparam NumberType either @a number_integer_t or @a number_unsigned_t - */ - template::value or - std::is_same::value, int> = 0> - void dump_integer(NumberType x) - { - // special case for "0" - if (x == 0) - { - o.put('0'); - return; - } - - const bool is_negative = x < 0; - size_t i = 0; - - // spare 1 byte for '\0' - while (x != 0 and i < number_buffer.size() - 1) - { - const auto digit = std::labs(static_cast(x % 10)); - number_buffer[i++] = static_cast('0' + digit); - x /= 10; - } - - // make sure the number has been processed completely - assert(x == 0); - - if (is_negative) - { - // make sure there is capacity for the '-' - assert(i < number_buffer.size() - 2); - number_buffer[i++] = '-'; - } - - std::reverse(number_buffer.begin(), number_buffer.begin() + i); - o.write(number_buffer.data(), static_cast(i)); - } - - /*! - @brief dump a floating-point number - - Dump a given floating-point number to output stream @a o. Works - internally with @a number_buffer. - - @param[in] x floating-point number to dump - */ - void dump_float(number_float_t x) - { - // NaN / inf - if (not std::isfinite(x) or std::isnan(x)) - { - o.write("null", 4); - return; - } - - // special case for 0.0 and -0.0 - if (x == 0) - { - if (std::signbit(x)) - { - o.write("-0.0", 4); - } - else - { - o.write("0.0", 3); - } - return; - } - - // get number of digits for a text -> float -> text round-trip - static constexpr auto d = std::numeric_limits::digits10; - - // the actual conversion - std::ptrdiff_t len = snprintf(number_buffer.data(), number_buffer.size(), - "%.*g", d, x); - - // negative value indicates an error - assert(len > 0); - // check if buffer was large enough - assert(static_cast(len) < number_buffer.size()); - - // erase thousands separator - if (thousands_sep != '\0') - { - const auto end = std::remove(number_buffer.begin(), - number_buffer.begin() + len, - thousands_sep); - std::fill(end, number_buffer.end(), '\0'); - assert((end - number_buffer.begin()) <= len); - len = (end - number_buffer.begin()); - } - - // convert decimal point to '.' - if (decimal_point != '\0' and decimal_point != '.') - { - for (auto& c : number_buffer) - { - if (c == decimal_point) - { - c = '.'; - break; - } - } - } - - o.write(number_buffer.data(), static_cast(len)); - - // determine if need to append ".0" - const bool value_is_int_like = std::none_of(number_buffer.begin(), - number_buffer.begin() + len + 1, - [](char c) - { - return c == '.' or c == 'e'; - }); - - if (value_is_int_like) - { - o.write(".0", 2); - } - } - - private: - /// the output of the serializer - std::ostream& o; - - /// a (hopefully) large enough character buffer - std::array number_buffer{{}}; - - /// the locale - const std::lconv* loc = nullptr; - /// the locale's thousand separator character - const char thousands_sep = '\0'; - /// the locale's decimal point character - const char decimal_point = '\0'; - - /// the indentation string - string_t indent_string = string_t(512, ' '); - }; - - public: - /*! - @brief serialize to stream - - Serialize the given JSON value @a j to the output stream @a o. The JSON - value will be serialized using the @ref dump member function. The - indentation of the output can be controlled with the member variable - `width` of the output stream @a o. For instance, using the manipulator - `std::setw(4)` on @a o sets the indentation level to `4` and the - serialization result is the same as calling `dump(4)`. - - @param[in,out] o stream to serialize to - @param[in] j JSON value to serialize - - @return the stream @a o - - @complexity Linear. - - @liveexample{The example below shows the serialization with different - parameters to `width` to adjust the indentation level.,operator_serialize} - - @since version 1.0.0 - */ - friend std::ostream& operator<<(std::ostream& o, const basic_json& j) - { - // read width member and use it as indentation parameter if nonzero - const bool pretty_print = (o.width() > 0); - const auto indentation = (pretty_print ? o.width() : 0); - - // reset width to 0 for subsequent calls to this stream - o.width(0); - - // do the actual serialization - serializer s(o); - s.dump(j, pretty_print, static_cast(indentation)); - return o; - } - - /*! - @brief serialize to stream - @copydoc operator<<(std::ostream&, const basic_json&) - */ - friend std::ostream& operator>>(const basic_json& j, std::ostream& o) - { - return o << j; - } - - /// @} - - - ///////////////////// - // deserialization // - ///////////////////// - - /// @name deserialization - /// @{ - - /*! - @brief deserialize from an array - - This function reads from an array of 1-byte values. - - @pre Each element of the container has a size of 1 byte. Violating this - precondition yields undefined behavior. **This precondition is enforced - with a static assertion.** - - @param[in] array array to read from - @param[in] cb a parser callback function of type @ref parser_callback_t - which is used to control the deserialization by filtering unwanted values - (optional) - - @return result of the deserialization - - @throw parse_error.101 if a parse error occurs; example: `""unexpected end - of input; expected string literal""` - @throw parse_error.102 if to_unicode fails or surrogate error - @throw parse_error.103 if to_unicode fails - - @complexity Linear in the length of the input. The parser is a predictive - LL(1) parser. The complexity can be higher if the parser callback function - @a cb has a super-linear complexity. - - @note A UTF-8 byte order mark is silently ignored. - - @liveexample{The example below demonstrates the `parse()` function reading - from an array.,parse__array__parser_callback_t} - - @since version 2.0.3 - */ - template - static basic_json parse(T (&array)[N], - const parser_callback_t cb = nullptr) - { - // delegate the call to the iterator-range parse overload - return parse(std::begin(array), std::end(array), cb); - } - - /*! - @brief deserialize from string literal - - @tparam CharT character/literal type with size of 1 byte - @param[in] s string literal to read a serialized JSON value from - @param[in] cb a parser callback function of type @ref parser_callback_t - which is used to control the deserialization by filtering unwanted values - (optional) - - @return result of the deserialization - - @throw parse_error.101 in case of an unexpected token - @throw parse_error.102 if to_unicode fails or surrogate error - @throw parse_error.103 if to_unicode fails - - @complexity Linear in the length of the input. The parser is a predictive - LL(1) parser. The complexity can be higher if the parser callback function - @a cb has a super-linear complexity. - - @note A UTF-8 byte order mark is silently ignored. - @note String containers like `std::string` or @ref string_t can be parsed - with @ref parse(const ContiguousContainer&, const parser_callback_t) - - @liveexample{The example below demonstrates the `parse()` function with - and without callback function.,parse__string__parser_callback_t} - - @sa @ref parse(std::istream&, const parser_callback_t) for a version that - reads from an input stream - - @since version 1.0.0 (originally for @ref string_t) - */ - template::value and - std::is_integral::type>::value and - sizeof(typename std::remove_pointer::type) == 1, int>::type = 0> - static basic_json parse(const CharT s, - const parser_callback_t cb = nullptr) - { - return parser(reinterpret_cast(s), cb).parse(); - } - - /*! - @brief deserialize from stream - - @param[in,out] i stream to read a serialized JSON value from - @param[in] cb a parser callback function of type @ref parser_callback_t - which is used to control the deserialization by filtering unwanted values - (optional) - - @return result of the deserialization - - @throw parse_error.101 in case of an unexpected token - @throw parse_error.102 if to_unicode fails or surrogate error - @throw parse_error.103 if to_unicode fails - @throw parse_error.111 if input stream is in a bad state - - @complexity Linear in the length of the input. The parser is a predictive - LL(1) parser. The complexity can be higher if the parser callback function - @a cb has a super-linear complexity. - - @note A UTF-8 byte order mark is silently ignored. - - @liveexample{The example below demonstrates the `parse()` function with - and without callback function.,parse__istream__parser_callback_t} - - @sa @ref parse(const CharT, const parser_callback_t) for a version - that reads from a string - - @since version 1.0.0 - */ - static basic_json parse(std::istream& i, - const parser_callback_t cb = nullptr) - { - return parser(i, cb).parse(); - } - - /*! - @copydoc parse(std::istream&, const parser_callback_t) - */ - static basic_json parse(std::istream&& i, - const parser_callback_t cb = nullptr) - { - return parser(i, cb).parse(); - } - - /*! - @brief deserialize from an iterator range with contiguous storage - - This function reads from an iterator range of a container with contiguous - storage of 1-byte values. Compatible container types include - `std::vector`, `std::string`, `std::array`, `std::valarray`, and - `std::initializer_list`. Furthermore, C-style arrays can be used with - `std::begin()`/`std::end()`. User-defined containers can be used as long - as they implement random-access iterators and a contiguous storage. - - @pre The iterator range is contiguous. Violating this precondition yields - undefined behavior. **This precondition is enforced with an assertion.** - @pre Each element in the range has a size of 1 byte. Violating this - precondition yields undefined behavior. **This precondition is enforced - with a static assertion.** - - @warning There is no way to enforce all preconditions at compile-time. If - the function is called with noncompliant iterators and with - assertions switched off, the behavior is undefined and will most - likely yield segmentation violation. - - @tparam IteratorType iterator of container with contiguous storage - @param[in] first begin of the range to parse (included) - @param[in] last end of the range to parse (excluded) - @param[in] cb a parser callback function of type @ref parser_callback_t - which is used to control the deserialization by filtering unwanted values - (optional) - - @return result of the deserialization - - @throw parse_error.101 in case of an unexpected token - @throw parse_error.102 if to_unicode fails or surrogate error - @throw parse_error.103 if to_unicode fails - - @complexity Linear in the length of the input. The parser is a predictive - LL(1) parser. The complexity can be higher if the parser callback function - @a cb has a super-linear complexity. - - @note A UTF-8 byte order mark is silently ignored. - - @liveexample{The example below demonstrates the `parse()` function reading - from an iterator range.,parse__iteratortype__parser_callback_t} - - @since version 2.0.3 - */ - template::iterator_category>::value, int>::type = 0> - static basic_json parse(IteratorType first, IteratorType last, - const parser_callback_t cb = nullptr) - { - // assertion to check that the iterator range is indeed contiguous, - // see http://stackoverflow.com/a/35008842/266378 for more discussion - assert(std::accumulate(first, last, std::pair(true, 0), - [&first](std::pair res, decltype(*first) val) - { - res.first &= (val == *(std::next(std::addressof(*first), res.second++))); - return res; - }).first); - - // assertion to check that each element is 1 byte long - static_assert(sizeof(typename std::iterator_traits::value_type) == 1, - "each element in the iterator range must have the size of 1 byte"); - - // if iterator range is empty, create a parser with an empty string - // to generate "unexpected EOF" error message - if (std::distance(first, last) <= 0) - { - return parser("").parse(); - } - - return parser(first, last, cb).parse(); - } - - /*! - @brief deserialize from a container with contiguous storage - - This function reads from a container with contiguous storage of 1-byte - values. Compatible container types include `std::vector`, `std::string`, - `std::array`, and `std::initializer_list`. User-defined containers can be - used as long as they implement random-access iterators and a contiguous - storage. - - @pre The container storage is contiguous. Violating this precondition - yields undefined behavior. **This precondition is enforced with an - assertion.** - @pre Each element of the container has a size of 1 byte. Violating this - precondition yields undefined behavior. **This precondition is enforced - with a static assertion.** - - @warning There is no way to enforce all preconditions at compile-time. If - the function is called with a noncompliant container and with - assertions switched off, the behavior is undefined and will most - likely yield segmentation violation. - - @tparam ContiguousContainer container type with contiguous storage - @param[in] c container to read from - @param[in] cb a parser callback function of type @ref parser_callback_t - which is used to control the deserialization by filtering unwanted values - (optional) - - @return result of the deserialization - - @throw parse_error.101 in case of an unexpected token - @throw parse_error.102 if to_unicode fails or surrogate error - @throw parse_error.103 if to_unicode fails - - @complexity Linear in the length of the input. The parser is a predictive - LL(1) parser. The complexity can be higher if the parser callback function - @a cb has a super-linear complexity. - - @note A UTF-8 byte order mark is silently ignored. - - @liveexample{The example below demonstrates the `parse()` function reading - from a contiguous container.,parse__contiguouscontainer__parser_callback_t} - - @since version 2.0.3 - */ - template::value and - std::is_base_of< - std::random_access_iterator_tag, - typename std::iterator_traits()))>::iterator_category>::value - , int>::type = 0> - static basic_json parse(const ContiguousContainer& c, - const parser_callback_t cb = nullptr) - { - // delegate the call to the iterator-range parse overload - return parse(std::begin(c), std::end(c), cb); - } - - /*! - @brief deserialize from stream - - Deserializes an input stream to a JSON value. - - @param[in,out] i input stream to read a serialized JSON value from - @param[in,out] j JSON value to write the deserialized input to - - @throw parse_error.101 in case of an unexpected token - @throw parse_error.102 if to_unicode fails or surrogate error - @throw parse_error.103 if to_unicode fails - @throw parse_error.111 if input stream is in a bad state - - @complexity Linear in the length of the input. The parser is a predictive - LL(1) parser. - - @note A UTF-8 byte order mark is silently ignored. - - @liveexample{The example below shows how a JSON value is constructed by - reading a serialization from a stream.,operator_deserialize} - - @sa parse(std::istream&, const parser_callback_t) for a variant with a - parser callback function to filter values while parsing - - @since version 1.0.0 - */ - friend std::istream& operator<<(basic_json& j, std::istream& i) - { - j = parser(i).parse(); - return i; - } - - /*! - @brief deserialize from stream - @copydoc operator<<(basic_json&, std::istream&) - */ - friend std::istream& operator>>(std::istream& i, basic_json& j) - { - j = parser(i).parse(); - return i; - } - - /// @} - - ////////////////////////////////////////// - // binary serialization/deserialization // - ////////////////////////////////////////// - - /// @name binary serialization/deserialization support - /// @{ - - private: - /*! - @note Some code in the switch cases has been copied, because otherwise - copilers would complain about implicit fallthrough and there is no - portable attribute to mute such warnings. - */ - template - static void add_to_vector(std::vector& vec, size_t bytes, const T number) - { - assert(bytes == 1 or bytes == 2 or bytes == 4 or bytes == 8); - - switch (bytes) - { - case 8: - { - vec.push_back(static_cast((static_cast(number) >> 070) & 0xff)); - vec.push_back(static_cast((static_cast(number) >> 060) & 0xff)); - vec.push_back(static_cast((static_cast(number) >> 050) & 0xff)); - vec.push_back(static_cast((static_cast(number) >> 040) & 0xff)); - vec.push_back(static_cast((number >> 030) & 0xff)); - vec.push_back(static_cast((number >> 020) & 0xff)); - vec.push_back(static_cast((number >> 010) & 0xff)); - vec.push_back(static_cast(number & 0xff)); - break; - } - - case 4: - { - vec.push_back(static_cast((number >> 030) & 0xff)); - vec.push_back(static_cast((number >> 020) & 0xff)); - vec.push_back(static_cast((number >> 010) & 0xff)); - vec.push_back(static_cast(number & 0xff)); - break; - } - - case 2: - { - vec.push_back(static_cast((number >> 010) & 0xff)); - vec.push_back(static_cast(number & 0xff)); - break; - } - - case 1: - { - vec.push_back(static_cast(number & 0xff)); - break; - } - } - } - - /*! - @brief take sufficient bytes from a vector to fill an integer variable - - In the context of binary serialization formats, we need to read several - bytes from a byte vector and combine them to multi-byte integral data - types. - - @param[in] vec byte vector to read from - @param[in] current_index the position in the vector after which to read - - @return the next sizeof(T) bytes from @a vec, in reverse order as T - - @tparam T the integral return type - - @throw parse_error.110 if there are less than sizeof(T)+1 bytes in the - vector @a vec to read - - In the for loop, the bytes from the vector are copied in reverse order into - the return value. In the figures below, let sizeof(T)=4 and `i` be the loop - variable. - - Precondition: - - vec: | | | a | b | c | d | T: | | | | | - ^ ^ ^ ^ - current_index i ptr sizeof(T) - - Postcondition: - - vec: | | | a | b | c | d | T: | d | c | b | a | - ^ ^ ^ - | i ptr - current_index - - @sa Code adapted from . - */ - template - static T get_from_vector(const std::vector& vec, const size_t current_index) - { - // check if we can read sizeof(T) bytes starting the next index - check_length(vec.size(), sizeof(T), current_index + 1); - - T result; - auto* ptr = reinterpret_cast(&result); - for (size_t i = 0; i < sizeof(T); ++i) - { - *ptr++ = vec[current_index + sizeof(T) - i]; - } - return result; - } - - /*! - @brief create a MessagePack serialization of a given JSON value - - This is a straightforward implementation of the MessagePack specification. - - @param[in] j JSON value to serialize - @param[in,out] v byte vector to write the serialization to - - @sa https://github.com/msgpack/msgpack/blob/master/spec.md - */ - static void to_msgpack_internal(const basic_json& j, std::vector& v) - { - switch (j.type()) - { - case value_t::null: - { - // nil - v.push_back(0xc0); - break; - } - - case value_t::boolean: - { - // true and false - v.push_back(j.m_value.boolean ? 0xc3 : 0xc2); - break; - } - - case value_t::number_integer: - { - if (j.m_value.number_integer >= 0) - { - // MessagePack does not differentiate between positive - // signed integers and unsigned integers. Therefore, we - // used the code from the value_t::number_unsigned case - // here. - if (j.m_value.number_unsigned < 128) - { - // positive fixnum - add_to_vector(v, 1, j.m_value.number_unsigned); - } - else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) - { - // uint 8 - v.push_back(0xcc); - add_to_vector(v, 1, j.m_value.number_unsigned); - } - else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) - { - // uint 16 - v.push_back(0xcd); - add_to_vector(v, 2, j.m_value.number_unsigned); - } - else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) - { - // uint 32 - v.push_back(0xce); - add_to_vector(v, 4, j.m_value.number_unsigned); - } - else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) - { - // uint 64 - v.push_back(0xcf); - add_to_vector(v, 8, j.m_value.number_unsigned); - } - } - else - { - if (j.m_value.number_integer >= -32) - { - // negative fixnum - add_to_vector(v, 1, j.m_value.number_integer); - } - else if (j.m_value.number_integer >= (std::numeric_limits::min)() and j.m_value.number_integer <= (std::numeric_limits::max)()) - { - // int 8 - v.push_back(0xd0); - add_to_vector(v, 1, j.m_value.number_integer); - } - else if (j.m_value.number_integer >= (std::numeric_limits::min)() and j.m_value.number_integer <= (std::numeric_limits::max)()) - { - // int 16 - v.push_back(0xd1); - add_to_vector(v, 2, j.m_value.number_integer); - } - else if (j.m_value.number_integer >= (std::numeric_limits::min)() and j.m_value.number_integer <= (std::numeric_limits::max)()) - { - // int 32 - v.push_back(0xd2); - add_to_vector(v, 4, j.m_value.number_integer); - } - else if (j.m_value.number_integer >= (std::numeric_limits::min)() and j.m_value.number_integer <= (std::numeric_limits::max)()) - { - // int 64 - v.push_back(0xd3); - add_to_vector(v, 8, j.m_value.number_integer); - } - } - break; - } - - case value_t::number_unsigned: - { - if (j.m_value.number_unsigned < 128) - { - // positive fixnum - add_to_vector(v, 1, j.m_value.number_unsigned); - } - else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) - { - // uint 8 - v.push_back(0xcc); - add_to_vector(v, 1, j.m_value.number_unsigned); - } - else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) - { - // uint 16 - v.push_back(0xcd); - add_to_vector(v, 2, j.m_value.number_unsigned); - } - else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) - { - // uint 32 - v.push_back(0xce); - add_to_vector(v, 4, j.m_value.number_unsigned); - } - else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) - { - // uint 64 - v.push_back(0xcf); - add_to_vector(v, 8, j.m_value.number_unsigned); - } - break; - } - - case value_t::number_float: - { - // float 64 - v.push_back(0xcb); - const auto* helper = reinterpret_cast(&(j.m_value.number_float)); - for (size_t i = 0; i < 8; ++i) - { - v.push_back(helper[7 - i]); - } - break; - } - - case value_t::string: - { - const auto N = j.m_value.string->size(); - if (N <= 31) - { - // fixstr - v.push_back(static_cast(0xa0 | N)); - } - else if (N <= 255) - { - // str 8 - v.push_back(0xd9); - add_to_vector(v, 1, N); - } - else if (N <= 65535) - { - // str 16 - v.push_back(0xda); - add_to_vector(v, 2, N); - } - else if (N <= 4294967295) - { - // str 32 - v.push_back(0xdb); - add_to_vector(v, 4, N); - } - - // append string - std::copy(j.m_value.string->begin(), j.m_value.string->end(), - std::back_inserter(v)); - break; - } - - case value_t::array: - { - const auto N = j.m_value.array->size(); - if (N <= 15) - { - // fixarray - v.push_back(static_cast(0x90 | N)); - } - else if (N <= 0xffff) - { - // array 16 - v.push_back(0xdc); - add_to_vector(v, 2, N); - } - else if (N <= 0xffffffff) - { - // array 32 - v.push_back(0xdd); - add_to_vector(v, 4, N); - } - - // append each element - for (const auto& el : *j.m_value.array) - { - to_msgpack_internal(el, v); - } - break; - } - - case value_t::object: - { - const auto N = j.m_value.object->size(); - if (N <= 15) - { - // fixmap - v.push_back(static_cast(0x80 | (N & 0xf))); - } - else if (N <= 65535) - { - // map 16 - v.push_back(0xde); - add_to_vector(v, 2, N); - } - else if (N <= 4294967295) - { - // map 32 - v.push_back(0xdf); - add_to_vector(v, 4, N); - } - - // append each element - for (const auto& el : *j.m_value.object) - { - to_msgpack_internal(el.first, v); - to_msgpack_internal(el.second, v); - } - break; - } - - default: - { - break; - } - } - } - - /*! - @brief create a CBOR serialization of a given JSON value - - This is a straightforward implementation of the CBOR specification. - - @param[in] j JSON value to serialize - @param[in,out] v byte vector to write the serialization to - - @sa https://tools.ietf.org/html/rfc7049 - */ - static void to_cbor_internal(const basic_json& j, std::vector& v) - { - switch (j.type()) - { - case value_t::null: - { - v.push_back(0xf6); - break; - } - - case value_t::boolean: - { - v.push_back(j.m_value.boolean ? 0xf5 : 0xf4); - break; - } - - case value_t::number_integer: - { - if (j.m_value.number_integer >= 0) - { - // CBOR does not differentiate between positive signed - // integers and unsigned integers. Therefore, we used the - // code from the value_t::number_unsigned case here. - if (j.m_value.number_integer <= 0x17) - { - add_to_vector(v, 1, j.m_value.number_integer); - } - else if (j.m_value.number_integer <= (std::numeric_limits::max)()) - { - v.push_back(0x18); - // one-byte uint8_t - add_to_vector(v, 1, j.m_value.number_integer); - } - else if (j.m_value.number_integer <= (std::numeric_limits::max)()) - { - v.push_back(0x19); - // two-byte uint16_t - add_to_vector(v, 2, j.m_value.number_integer); - } - else if (j.m_value.number_integer <= (std::numeric_limits::max)()) - { - v.push_back(0x1a); - // four-byte uint32_t - add_to_vector(v, 4, j.m_value.number_integer); - } - else - { - v.push_back(0x1b); - // eight-byte uint64_t - add_to_vector(v, 8, j.m_value.number_integer); - } - } - else - { - // The conversions below encode the sign in the first - // byte, and the value is converted to a positive number. - const auto positive_number = -1 - j.m_value.number_integer; - if (j.m_value.number_integer >= -24) - { - v.push_back(static_cast(0x20 + positive_number)); - } - else if (positive_number <= (std::numeric_limits::max)()) - { - // int 8 - v.push_back(0x38); - add_to_vector(v, 1, positive_number); - } - else if (positive_number <= (std::numeric_limits::max)()) - { - // int 16 - v.push_back(0x39); - add_to_vector(v, 2, positive_number); - } - else if (positive_number <= (std::numeric_limits::max)()) - { - // int 32 - v.push_back(0x3a); - add_to_vector(v, 4, positive_number); - } - else - { - // int 64 - v.push_back(0x3b); - add_to_vector(v, 8, positive_number); - } - } - break; - } - - case value_t::number_unsigned: - { - if (j.m_value.number_unsigned <= 0x17) - { - v.push_back(static_cast(j.m_value.number_unsigned)); - } - else if (j.m_value.number_unsigned <= 0xff) - { - v.push_back(0x18); - // one-byte uint8_t - add_to_vector(v, 1, j.m_value.number_unsigned); - } - else if (j.m_value.number_unsigned <= 0xffff) - { - v.push_back(0x19); - // two-byte uint16_t - add_to_vector(v, 2, j.m_value.number_unsigned); - } - else if (j.m_value.number_unsigned <= 0xffffffff) - { - v.push_back(0x1a); - // four-byte uint32_t - add_to_vector(v, 4, j.m_value.number_unsigned); - } - else if (j.m_value.number_unsigned <= 0xffffffffffffffff) - { - v.push_back(0x1b); - // eight-byte uint64_t - add_to_vector(v, 8, j.m_value.number_unsigned); - } - break; - } - - case value_t::number_float: - { - // Double-Precision Float - v.push_back(0xfb); - const auto* helper = reinterpret_cast(&(j.m_value.number_float)); - for (size_t i = 0; i < 8; ++i) - { - v.push_back(helper[7 - i]); - } - break; - } - - case value_t::string: - { - const auto N = j.m_value.string->size(); - if (N <= 0x17) - { - v.push_back(static_cast(0x60 + N)); // 1 byte for string + size - } - else if (N <= 0xff) - { - v.push_back(0x78); // one-byte uint8_t for N - add_to_vector(v, 1, N); - } - else if (N <= 0xffff) - { - v.push_back(0x79); // two-byte uint16_t for N - add_to_vector(v, 2, N); - } - else if (N <= 0xffffffff) - { - v.push_back(0x7a); // four-byte uint32_t for N - add_to_vector(v, 4, N); - } - // LCOV_EXCL_START - else if (N <= 0xffffffffffffffff) - { - v.push_back(0x7b); // eight-byte uint64_t for N - add_to_vector(v, 8, N); - } - // LCOV_EXCL_STOP - - // append string - std::copy(j.m_value.string->begin(), j.m_value.string->end(), - std::back_inserter(v)); - break; - } - - case value_t::array: - { - const auto N = j.m_value.array->size(); - if (N <= 0x17) - { - v.push_back(static_cast(0x80 + N)); // 1 byte for array + size - } - else if (N <= 0xff) - { - v.push_back(0x98); // one-byte uint8_t for N - add_to_vector(v, 1, N); - } - else if (N <= 0xffff) - { - v.push_back(0x99); // two-byte uint16_t for N - add_to_vector(v, 2, N); - } - else if (N <= 0xffffffff) - { - v.push_back(0x9a); // four-byte uint32_t for N - add_to_vector(v, 4, N); - } - // LCOV_EXCL_START - else if (N <= 0xffffffffffffffff) - { - v.push_back(0x9b); // eight-byte uint64_t for N - add_to_vector(v, 8, N); - } - // LCOV_EXCL_STOP - - // append each element - for (const auto& el : *j.m_value.array) - { - to_cbor_internal(el, v); - } - break; - } - - case value_t::object: - { - const auto N = j.m_value.object->size(); - if (N <= 0x17) - { - v.push_back(static_cast(0xa0 + N)); // 1 byte for object + size - } - else if (N <= 0xff) - { - v.push_back(0xb8); - add_to_vector(v, 1, N); // one-byte uint8_t for N - } - else if (N <= 0xffff) - { - v.push_back(0xb9); - add_to_vector(v, 2, N); // two-byte uint16_t for N - } - else if (N <= 0xffffffff) - { - v.push_back(0xba); - add_to_vector(v, 4, N); // four-byte uint32_t for N - } - // LCOV_EXCL_START - else if (N <= 0xffffffffffffffff) - { - v.push_back(0xbb); - add_to_vector(v, 8, N); // eight-byte uint64_t for N - } - // LCOV_EXCL_STOP - - // append each element - for (const auto& el : *j.m_value.object) - { - to_cbor_internal(el.first, v); - to_cbor_internal(el.second, v); - } - break; - } - - default: - { - break; - } - } - } - - - /* - @brief checks if given lengths do not exceed the size of a given vector - - To secure the access to the byte vector during CBOR/MessagePack - deserialization, bytes are copied from the vector into buffers. This - function checks if the number of bytes to copy (@a len) does not exceed - the size @s size of the vector. Additionally, an @a offset is given from - where to start reading the bytes. - - This function checks whether reading the bytes is safe; that is, offset is - a valid index in the vector, offset+len - - @param[in] size size of the byte vector - @param[in] len number of bytes to read - @param[in] offset offset where to start reading - - vec: x x x x x X X X X X - ^ ^ ^ - 0 offset len - - @throws out_of_range if `len > v.size()` - */ - static void check_length(const size_t size, const size_t len, const size_t offset) - { - // simple case: requested length is greater than the vector's length - if (len > size or offset > size) - { - JSON_THROW(parse_error(110, offset + 1, "cannot read " + std::to_string(len) + " bytes from vector")); - } - - // second case: adding offset would result in overflow - if ((size > ((std::numeric_limits::max)() - offset))) - { - JSON_THROW(parse_error(110, offset + 1, "cannot read " + std::to_string(len) + " bytes from vector")); - } - - // last case: reading past the end of the vector - if (len + offset > size) - { - JSON_THROW(parse_error(110, offset + 1, "cannot read " + std::to_string(len) + " bytes from vector")); - } - } - - /*! - @brief check if the next byte belongs to a string - - While parsing a map, the keys must be strings. This function checks if the - current byte is one of the start bytes for a string in MessagePack: - - - 0xa0 - 0xbf: fixstr - - 0xd9: str 8 - - 0xda: str 16 - - 0xdb: str 32 - - @param[in] v MessagePack serialization - @param[in] idx byte index in @a v to check for a string - - @throw parse_error.113 if `v[idx]` does not belong to a string - */ - static void msgpack_expect_string(const std::vector& v, size_t idx) - { - check_length(v.size(), 1, idx); - - const auto byte = v[idx]; - if ((byte >= 0xa0 and byte <= 0xbf) or (byte >= 0xd9 and byte <= 0xdb)) - { - return; - } - - std::stringstream ss; - ss << std::hex << static_cast(v[idx]); - JSON_THROW(parse_error(113, idx + 1, "expected a MessagePack string; last byte: 0x" + ss.str())); - } - - /*! - @brief check if the next byte belongs to a string - - While parsing a map, the keys must be strings. This function checks if the - current byte is one of the start bytes for a string in CBOR: - - - 0x60 - 0x77: fixed length - - 0x78 - 0x7b: variable length - - 0x7f: indefinity length - - @param[in] v CBOR serialization - @param[in] idx byte index in @a v to check for a string - - @throw parse_error.113 if `v[idx]` does not belong to a string - */ - static void cbor_expect_string(const std::vector& v, size_t idx) - { - check_length(v.size(), 1, idx); - - const auto byte = v[idx]; - if ((byte >= 0x60 and byte <= 0x7b) or byte == 0x7f) - { - return; - } - - std::stringstream ss; - ss << std::hex << static_cast(v[idx]); - JSON_THROW(parse_error(113, idx + 1, "expected a CBOR string; last byte: 0x" + ss.str())); - } - - /*! - @brief create a JSON value from a given MessagePack vector - - @param[in] v MessagePack serialization - @param[in] idx byte index to start reading from @a v - - @return deserialized JSON value - - @throw parse_error.110 if the given vector ends prematurely - @throw parse_error.112 if unsupported features from MessagePack were - used in the given vector @a v or if the input is not valid MessagePack - @throw parse_error.113 if a string was expected as map key, but not found - - @sa https://github.com/msgpack/msgpack/blob/master/spec.md - */ - static basic_json from_msgpack_internal(const std::vector& v, size_t& idx) - { - // store and increment index - const size_t current_idx = idx++; - - // make sure reading 1 byte is safe - check_length(v.size(), 1, current_idx); - - if (v[current_idx] <= 0xbf) - { - if (v[current_idx] <= 0x7f) // positive fixint - { - return v[current_idx]; - } - if (v[current_idx] <= 0x8f) // fixmap - { - basic_json result = value_t::object; - const size_t len = v[current_idx] & 0x0f; - for (size_t i = 0; i < len; ++i) - { - msgpack_expect_string(v, idx); - std::string key = from_msgpack_internal(v, idx); - result[key] = from_msgpack_internal(v, idx); - } - return result; - } - else if (v[current_idx] <= 0x9f) // fixarray - { - basic_json result = value_t::array; - const size_t len = v[current_idx] & 0x0f; - for (size_t i = 0; i < len; ++i) - { - result.push_back(from_msgpack_internal(v, idx)); - } - return result; - } - else // fixstr - { - const size_t len = v[current_idx] & 0x1f; - const size_t offset = current_idx + 1; - idx += len; // skip content bytes - check_length(v.size(), len, offset); - return std::string(reinterpret_cast(v.data()) + offset, len); - } - } - else if (v[current_idx] >= 0xe0) // negative fixint - { - return static_cast(v[current_idx]); - } - else - { - switch (v[current_idx]) - { - case 0xc0: // nil - { - return value_t::null; - } - - case 0xc2: // false - { - return false; - } - - case 0xc3: // true - { - return true; - } - - case 0xca: // float 32 - { - // copy bytes in reverse order into the double variable - float res; - check_length(v.size(), sizeof(float), current_idx + 1); - for (size_t byte = 0; byte < sizeof(float); ++byte) - { - reinterpret_cast(&res)[sizeof(float) - byte - 1] = v[current_idx + 1 + byte]; - } - idx += sizeof(float); // skip content bytes - return res; - } - - case 0xcb: // float 64 - { - // copy bytes in reverse order into the double variable - double res; - check_length(v.size(), sizeof(double), current_idx + 1); - for (size_t byte = 0; byte < sizeof(double); ++byte) - { - reinterpret_cast(&res)[sizeof(double) - byte - 1] = v[current_idx + 1 + byte]; - } - idx += sizeof(double); // skip content bytes - return res; - } - - case 0xcc: // uint 8 - { - idx += 1; // skip content byte - return get_from_vector(v, current_idx); - } - - case 0xcd: // uint 16 - { - idx += 2; // skip 2 content bytes - return get_from_vector(v, current_idx); - } - - case 0xce: // uint 32 - { - idx += 4; // skip 4 content bytes - return get_from_vector(v, current_idx); - } - - case 0xcf: // uint 64 - { - idx += 8; // skip 8 content bytes - return get_from_vector(v, current_idx); - } - - case 0xd0: // int 8 - { - idx += 1; // skip content byte - return get_from_vector(v, current_idx); - } - - case 0xd1: // int 16 - { - idx += 2; // skip 2 content bytes - return get_from_vector(v, current_idx); - } - - case 0xd2: // int 32 - { - idx += 4; // skip 4 content bytes - return get_from_vector(v, current_idx); - } - - case 0xd3: // int 64 - { - idx += 8; // skip 8 content bytes - return get_from_vector(v, current_idx); - } - - case 0xd9: // str 8 - { - const auto len = static_cast(get_from_vector(v, current_idx)); - const size_t offset = current_idx + 2; - idx += len + 1; // skip size byte + content bytes - check_length(v.size(), len, offset); - return std::string(reinterpret_cast(v.data()) + offset, len); - } - - case 0xda: // str 16 - { - const auto len = static_cast(get_from_vector(v, current_idx)); - const size_t offset = current_idx + 3; - idx += len + 2; // skip 2 size bytes + content bytes - check_length(v.size(), len, offset); - return std::string(reinterpret_cast(v.data()) + offset, len); - } - - case 0xdb: // str 32 - { - const auto len = static_cast(get_from_vector(v, current_idx)); - const size_t offset = current_idx + 5; - idx += len + 4; // skip 4 size bytes + content bytes - check_length(v.size(), len, offset); - return std::string(reinterpret_cast(v.data()) + offset, len); - } - - case 0xdc: // array 16 - { - basic_json result = value_t::array; - const auto len = static_cast(get_from_vector(v, current_idx)); - idx += 2; // skip 2 size bytes - for (size_t i = 0; i < len; ++i) - { - result.push_back(from_msgpack_internal(v, idx)); - } - return result; - } - - case 0xdd: // array 32 - { - basic_json result = value_t::array; - const auto len = static_cast(get_from_vector(v, current_idx)); - idx += 4; // skip 4 size bytes - for (size_t i = 0; i < len; ++i) - { - result.push_back(from_msgpack_internal(v, idx)); - } - return result; - } - - case 0xde: // map 16 - { - basic_json result = value_t::object; - const auto len = static_cast(get_from_vector(v, current_idx)); - idx += 2; // skip 2 size bytes - for (size_t i = 0; i < len; ++i) - { - msgpack_expect_string(v, idx); - std::string key = from_msgpack_internal(v, idx); - result[key] = from_msgpack_internal(v, idx); - } - return result; - } - - case 0xdf: // map 32 - { - basic_json result = value_t::object; - const auto len = static_cast(get_from_vector(v, current_idx)); - idx += 4; // skip 4 size bytes - for (size_t i = 0; i < len; ++i) - { - msgpack_expect_string(v, idx); - std::string key = from_msgpack_internal(v, idx); - result[key] = from_msgpack_internal(v, idx); - } - return result; - } - - default: - { - std::stringstream ss; - ss << std::hex << static_cast(v[current_idx]); - JSON_THROW(parse_error(112, current_idx + 1, "error reading MessagePack; last byte: 0x" + ss.str())); - } - } - } - } - - /*! - @brief create a JSON value from a given CBOR vector - - @param[in] v CBOR serialization - @param[in] idx byte index to start reading from @a v - - @return deserialized JSON value - - @throw parse_error.110 if the given vector ends prematurely - @throw parse_error.112 if unsupported features from CBOR were - used in the given vector @a v or if the input is not valid CBOR - @throw parse_error.113 if a string was expected as map key, but not found - - @sa https://tools.ietf.org/html/rfc7049 - */ - static basic_json from_cbor_internal(const std::vector& v, size_t& idx) - { - // store and increment index - const size_t current_idx = idx++; - - // make sure reading 1 byte is safe - check_length(v.size(), 1, current_idx); - - switch (v[current_idx]) - { - // Integer 0x00..0x17 (0..23) - case 0x00: - case 0x01: - case 0x02: - case 0x03: - case 0x04: - case 0x05: - case 0x06: - case 0x07: - case 0x08: - case 0x09: - case 0x0a: - case 0x0b: - case 0x0c: - case 0x0d: - case 0x0e: - case 0x0f: - case 0x10: - case 0x11: - case 0x12: - case 0x13: - case 0x14: - case 0x15: - case 0x16: - case 0x17: - { - return v[current_idx]; - } - - case 0x18: // Unsigned integer (one-byte uint8_t follows) - { - idx += 1; // skip content byte - return get_from_vector(v, current_idx); - } - - case 0x19: // Unsigned integer (two-byte uint16_t follows) - { - idx += 2; // skip 2 content bytes - return get_from_vector(v, current_idx); - } - - case 0x1a: // Unsigned integer (four-byte uint32_t follows) - { - idx += 4; // skip 4 content bytes - return get_from_vector(v, current_idx); - } - - case 0x1b: // Unsigned integer (eight-byte uint64_t follows) - { - idx += 8; // skip 8 content bytes - return get_from_vector(v, current_idx); - } - - // Negative integer -1-0x00..-1-0x17 (-1..-24) - case 0x20: - case 0x21: - case 0x22: - case 0x23: - case 0x24: - case 0x25: - case 0x26: - case 0x27: - case 0x28: - case 0x29: - case 0x2a: - case 0x2b: - case 0x2c: - case 0x2d: - case 0x2e: - case 0x2f: - case 0x30: - case 0x31: - case 0x32: - case 0x33: - case 0x34: - case 0x35: - case 0x36: - case 0x37: - { - return static_cast(0x20 - 1 - v[current_idx]); - } - - case 0x38: // Negative integer (one-byte uint8_t follows) - { - idx += 1; // skip content byte - // must be uint8_t ! - return static_cast(-1) - get_from_vector(v, current_idx); - } - - case 0x39: // Negative integer -1-n (two-byte uint16_t follows) - { - idx += 2; // skip 2 content bytes - return static_cast(-1) - get_from_vector(v, current_idx); - } - - case 0x3a: // Negative integer -1-n (four-byte uint32_t follows) - { - idx += 4; // skip 4 content bytes - return static_cast(-1) - get_from_vector(v, current_idx); - } - - case 0x3b: // Negative integer -1-n (eight-byte uint64_t follows) - { - idx += 8; // skip 8 content bytes - return static_cast(-1) - static_cast(get_from_vector(v, current_idx)); - } - - // UTF-8 string (0x00..0x17 bytes follow) - case 0x60: - case 0x61: - case 0x62: - case 0x63: - case 0x64: - case 0x65: - case 0x66: - case 0x67: - case 0x68: - case 0x69: - case 0x6a: - case 0x6b: - case 0x6c: - case 0x6d: - case 0x6e: - case 0x6f: - case 0x70: - case 0x71: - case 0x72: - case 0x73: - case 0x74: - case 0x75: - case 0x76: - case 0x77: - { - const auto len = static_cast(v[current_idx] - 0x60); - const size_t offset = current_idx + 1; - idx += len; // skip content bytes - check_length(v.size(), len, offset); - return std::string(reinterpret_cast(v.data()) + offset, len); - } - - case 0x78: // UTF-8 string (one-byte uint8_t for n follows) - { - const auto len = static_cast(get_from_vector(v, current_idx)); - const size_t offset = current_idx + 2; - idx += len + 1; // skip size byte + content bytes - check_length(v.size(), len, offset); - return std::string(reinterpret_cast(v.data()) + offset, len); - } - - case 0x79: // UTF-8 string (two-byte uint16_t for n follow) - { - const auto len = static_cast(get_from_vector(v, current_idx)); - const size_t offset = current_idx + 3; - idx += len + 2; // skip 2 size bytes + content bytes - check_length(v.size(), len, offset); - return std::string(reinterpret_cast(v.data()) + offset, len); - } - - case 0x7a: // UTF-8 string (four-byte uint32_t for n follow) - { - const auto len = static_cast(get_from_vector(v, current_idx)); - const size_t offset = current_idx + 5; - idx += len + 4; // skip 4 size bytes + content bytes - check_length(v.size(), len, offset); - return std::string(reinterpret_cast(v.data()) + offset, len); - } - - case 0x7b: // UTF-8 string (eight-byte uint64_t for n follow) - { - const auto len = static_cast(get_from_vector(v, current_idx)); - const size_t offset = current_idx + 9; - idx += len + 8; // skip 8 size bytes + content bytes - check_length(v.size(), len, offset); - return std::string(reinterpret_cast(v.data()) + offset, len); - } - - case 0x7f: // UTF-8 string (indefinite length) - { - std::string result; - while (check_length(v.size(), 1, idx), v[idx] != 0xff) - { - string_t s = from_cbor_internal(v, idx); - result += s; - } - // skip break byte (0xFF) - idx += 1; - return result; - } - - // array (0x00..0x17 data items follow) - case 0x80: - case 0x81: - case 0x82: - case 0x83: - case 0x84: - case 0x85: - case 0x86: - case 0x87: - case 0x88: - case 0x89: - case 0x8a: - case 0x8b: - case 0x8c: - case 0x8d: - case 0x8e: - case 0x8f: - case 0x90: - case 0x91: - case 0x92: - case 0x93: - case 0x94: - case 0x95: - case 0x96: - case 0x97: - { - basic_json result = value_t::array; - const auto len = static_cast(v[current_idx] - 0x80); - for (size_t i = 0; i < len; ++i) - { - result.push_back(from_cbor_internal(v, idx)); - } - return result; - } - - case 0x98: // array (one-byte uint8_t for n follows) - { - basic_json result = value_t::array; - const auto len = static_cast(get_from_vector(v, current_idx)); - idx += 1; // skip 1 size byte - for (size_t i = 0; i < len; ++i) - { - result.push_back(from_cbor_internal(v, idx)); - } - return result; - } - - case 0x99: // array (two-byte uint16_t for n follow) - { - basic_json result = value_t::array; - const auto len = static_cast(get_from_vector(v, current_idx)); - idx += 2; // skip 4 size bytes - for (size_t i = 0; i < len; ++i) - { - result.push_back(from_cbor_internal(v, idx)); - } - return result; - } - - case 0x9a: // array (four-byte uint32_t for n follow) - { - basic_json result = value_t::array; - const auto len = static_cast(get_from_vector(v, current_idx)); - idx += 4; // skip 4 size bytes - for (size_t i = 0; i < len; ++i) - { - result.push_back(from_cbor_internal(v, idx)); - } - return result; - } - - case 0x9b: // array (eight-byte uint64_t for n follow) - { - basic_json result = value_t::array; - const auto len = static_cast(get_from_vector(v, current_idx)); - idx += 8; // skip 8 size bytes - for (size_t i = 0; i < len; ++i) - { - result.push_back(from_cbor_internal(v, idx)); - } - return result; - } - - case 0x9f: // array (indefinite length) - { - basic_json result = value_t::array; - while (check_length(v.size(), 1, idx), v[idx] != 0xff) - { - result.push_back(from_cbor_internal(v, idx)); - } - // skip break byte (0xFF) - idx += 1; - return result; - } - - // map (0x00..0x17 pairs of data items follow) - case 0xa0: - case 0xa1: - case 0xa2: - case 0xa3: - case 0xa4: - case 0xa5: - case 0xa6: - case 0xa7: - case 0xa8: - case 0xa9: - case 0xaa: - case 0xab: - case 0xac: - case 0xad: - case 0xae: - case 0xaf: - case 0xb0: - case 0xb1: - case 0xb2: - case 0xb3: - case 0xb4: - case 0xb5: - case 0xb6: - case 0xb7: - { - basic_json result = value_t::object; - const auto len = static_cast(v[current_idx] - 0xa0); - for (size_t i = 0; i < len; ++i) - { - cbor_expect_string(v, idx); - std::string key = from_cbor_internal(v, idx); - result[key] = from_cbor_internal(v, idx); - } - return result; - } - - case 0xb8: // map (one-byte uint8_t for n follows) - { - basic_json result = value_t::object; - const auto len = static_cast(get_from_vector(v, current_idx)); - idx += 1; // skip 1 size byte - for (size_t i = 0; i < len; ++i) - { - cbor_expect_string(v, idx); - std::string key = from_cbor_internal(v, idx); - result[key] = from_cbor_internal(v, idx); - } - return result; - } - - case 0xb9: // map (two-byte uint16_t for n follow) - { - basic_json result = value_t::object; - const auto len = static_cast(get_from_vector(v, current_idx)); - idx += 2; // skip 2 size bytes - for (size_t i = 0; i < len; ++i) - { - cbor_expect_string(v, idx); - std::string key = from_cbor_internal(v, idx); - result[key] = from_cbor_internal(v, idx); - } - return result; - } - - case 0xba: // map (four-byte uint32_t for n follow) - { - basic_json result = value_t::object; - const auto len = static_cast(get_from_vector(v, current_idx)); - idx += 4; // skip 4 size bytes - for (size_t i = 0; i < len; ++i) - { - cbor_expect_string(v, idx); - std::string key = from_cbor_internal(v, idx); - result[key] = from_cbor_internal(v, idx); - } - return result; - } - - case 0xbb: // map (eight-byte uint64_t for n follow) - { - basic_json result = value_t::object; - const auto len = static_cast(get_from_vector(v, current_idx)); - idx += 8; // skip 8 size bytes - for (size_t i = 0; i < len; ++i) - { - cbor_expect_string(v, idx); - std::string key = from_cbor_internal(v, idx); - result[key] = from_cbor_internal(v, idx); - } - return result; - } - - case 0xbf: // map (indefinite length) - { - basic_json result = value_t::object; - while (check_length(v.size(), 1, idx), v[idx] != 0xff) - { - cbor_expect_string(v, idx); - std::string key = from_cbor_internal(v, idx); - result[key] = from_cbor_internal(v, idx); - } - // skip break byte (0xFF) - idx += 1; - return result; - } - - case 0xf4: // false - { - return false; - } - - case 0xf5: // true - { - return true; - } - - case 0xf6: // null - { - return value_t::null; - } - - case 0xf9: // Half-Precision Float (two-byte IEEE 754) - { - idx += 2; // skip two content bytes - - // code from RFC 7049, Appendix D, Figure 3: - // As half-precision floating-point numbers were only added to - // IEEE 754 in 2008, today's programming platforms often still - // only have limited support for them. It is very easy to - // include at least decoding support for them even without such - // support. An example of a small decoder for half-precision - // floating-point numbers in the C language is shown in Fig. 3. - check_length(v.size(), 2, current_idx + 1); - const int half = (v[current_idx + 1] << 8) + v[current_idx + 2]; - const int exp = (half >> 10) & 0x1f; - const int mant = half & 0x3ff; - double val; - if (exp == 0) - { - val = std::ldexp(mant, -24); - } - else if (exp != 31) - { - val = std::ldexp(mant + 1024, exp - 25); - } - else - { - val = mant == 0 - ? std::numeric_limits::infinity() - : std::numeric_limits::quiet_NaN(); - } - return (half & 0x8000) != 0 ? -val : val; - } - - case 0xfa: // Single-Precision Float (four-byte IEEE 754) - { - // copy bytes in reverse order into the float variable - float res; - check_length(v.size(), sizeof(float), current_idx + 1); - for (size_t byte = 0; byte < sizeof(float); ++byte) - { - reinterpret_cast(&res)[sizeof(float) - byte - 1] = v[current_idx + 1 + byte]; - } - idx += sizeof(float); // skip content bytes - return res; - } - - case 0xfb: // Double-Precision Float (eight-byte IEEE 754) - { - // copy bytes in reverse order into the double variable - double res; - check_length(v.size(), sizeof(double), current_idx + 1); - for (size_t byte = 0; byte < sizeof(double); ++byte) - { - reinterpret_cast(&res)[sizeof(double) - byte - 1] = v[current_idx + 1 + byte]; - } - idx += sizeof(double); // skip content bytes - return res; - } - - default: // anything else (0xFF is handled inside the other types) - { - std::stringstream ss; - ss << std::hex << static_cast(v[current_idx]); - JSON_THROW(parse_error(112, current_idx + 1, "error reading CBOR; last byte: 0x" + ss.str())); - } - } - } - - public: - /*! - @brief create a MessagePack serialization of a given JSON value - - Serializes a given JSON value @a j to a byte vector using the MessagePack - serialization format. MessagePack is a binary serialization format which - aims to be more compact than JSON itself, yet more efficient to parse. - - The library uses the following mapping from JSON values types to - MessagePack types according to the MessagePack specification: - - JSON value type | value/range | MessagePack type | first byte - --------------- | --------------------------------- | ---------------- | ---------- - null | `null` | nil | 0xc0 - boolean | `true` | true | 0xc3 - boolean | `false` | false | 0xc2 - number_integer | -9223372036854775808..-2147483649 | int64 | 0xd3 - number_integer | -2147483648..-32769 | int32 | 0xd2 - number_integer | -32768..-129 | int16 | 0xd1 - number_integer | -128..-33 | int8 | 0xd0 - number_integer | -32..-1 | negative fixint | 0xe0..0xff - number_integer | 0..127 | positive fixint | 0x00..0x7f - number_integer | 128..255 | uint 8 | 0xcc - number_integer | 256..65535 | uint 16 | 0xcd - number_integer | 65536..4294967295 | uint 32 | 0xce - number_integer | 4294967296..18446744073709551615 | uint 64 | 0xcf - number_unsigned | 0..127 | positive fixint | 0x00..0x7f - number_unsigned | 128..255 | uint 8 | 0xcc - number_unsigned | 256..65535 | uint 16 | 0xcd - number_unsigned | 65536..4294967295 | uint 32 | 0xce - number_unsigned | 4294967296..18446744073709551615 | uint 64 | 0xcf - number_float | *any value* | float 64 | 0xcb - string | *length*: 0..31 | fixstr | 0xa0..0xbf - string | *length*: 32..255 | str 8 | 0xd9 - string | *length*: 256..65535 | str 16 | 0xda - string | *length*: 65536..4294967295 | str 32 | 0xdb - array | *size*: 0..15 | fixarray | 0x90..0x9f - array | *size*: 16..65535 | array 16 | 0xdc - array | *size*: 65536..4294967295 | array 32 | 0xdd - object | *size*: 0..15 | fix map | 0x80..0x8f - object | *size*: 16..65535 | map 16 | 0xde - object | *size*: 65536..4294967295 | map 32 | 0xdf - - @note The mapping is **complete** in the sense that any JSON value type - can be converted to a MessagePack value. - - @note The following values can **not** be converted to a MessagePack value: - - strings with more than 4294967295 bytes - - arrays with more than 4294967295 elements - - objects with more than 4294967295 elements - - @note The following MessagePack types are not used in the conversion: - - bin 8 - bin 32 (0xc4..0xc6) - - ext 8 - ext 32 (0xc7..0xc9) - - float 32 (0xca) - - fixext 1 - fixext 16 (0xd4..0xd8) - - @note Any MessagePack output created @ref to_msgpack can be successfully - parsed by @ref from_msgpack. - - @param[in] j JSON value to serialize - @return MessagePack serialization as byte vector - - @complexity Linear in the size of the JSON value @a j. - - @liveexample{The example shows the serialization of a JSON value to a byte - vector in MessagePack format.,to_msgpack} - - @sa http://msgpack.org - @sa @ref from_msgpack(const std::vector&, const size_t) for the - analogous deserialization - @sa @ref to_cbor(const basic_json& for the related CBOR format - - @since version 2.0.9 - */ - static std::vector to_msgpack(const basic_json& j) - { - std::vector result; - to_msgpack_internal(j, result); - return result; - } - - /*! - @brief create a JSON value from a byte vector in MessagePack format - - Deserializes a given byte vector @a v to a JSON value using the MessagePack - serialization format. - - The library maps MessagePack types to JSON value types as follows: - - MessagePack type | JSON value type | first byte - ---------------- | --------------- | ---------- - positive fixint | number_unsigned | 0x00..0x7f - fixmap | object | 0x80..0x8f - fixarray | array | 0x90..0x9f - fixstr | string | 0xa0..0xbf - nil | `null` | 0xc0 - false | `false` | 0xc2 - true | `true` | 0xc3 - float 32 | number_float | 0xca - float 64 | number_float | 0xcb - uint 8 | number_unsigned | 0xcc - uint 16 | number_unsigned | 0xcd - uint 32 | number_unsigned | 0xce - uint 64 | number_unsigned | 0xcf - int 8 | number_integer | 0xd0 - int 16 | number_integer | 0xd1 - int 32 | number_integer | 0xd2 - int 64 | number_integer | 0xd3 - str 8 | string | 0xd9 - str 16 | string | 0xda - str 32 | string | 0xdb - array 16 | array | 0xdc - array 32 | array | 0xdd - map 16 | object | 0xde - map 32 | object | 0xdf - negative fixint | number_integer | 0xe0-0xff - - @warning The mapping is **incomplete** in the sense that not all - MessagePack types can be converted to a JSON value. The following - MessagePack types are not supported and will yield parse errors: - - bin 8 - bin 32 (0xc4..0xc6) - - ext 8 - ext 32 (0xc7..0xc9) - - fixext 1 - fixext 16 (0xd4..0xd8) - - @note Any MessagePack output created @ref to_msgpack can be successfully - parsed by @ref from_msgpack. - - @param[in] v a byte vector in MessagePack format - @param[in] start_index the index to start reading from @a v (0 by default) - @return deserialized JSON value - - @throw parse_error.110 if the given vector ends prematurely - @throw parse_error.112 if unsupported features from MessagePack were - used in the given vector @a v or if the input is not valid MessagePack - @throw parse_error.113 if a string was expected as map key, but not found - - @complexity Linear in the size of the byte vector @a v. - - @liveexample{The example shows the deserialization of a byte vector in - MessagePack format to a JSON value.,from_msgpack} - - @sa http://msgpack.org - @sa @ref to_msgpack(const basic_json&) for the analogous serialization - @sa @ref from_cbor(const std::vector&, const size_t) for the - related CBOR format - - @since version 2.0.9, parameter @a start_index since 2.1.1 - */ - static basic_json from_msgpack(const std::vector& v, - const size_t start_index = 0) - { - size_t i = start_index; - return from_msgpack_internal(v, i); - } - - /*! - @brief create a MessagePack serialization of a given JSON value - - Serializes a given JSON value @a j to a byte vector using the CBOR (Concise - Binary Object Representation) serialization format. CBOR is a binary - serialization format which aims to be more compact than JSON itself, yet - more efficient to parse. - - The library uses the following mapping from JSON values types to - CBOR types according to the CBOR specification (RFC 7049): - - JSON value type | value/range | CBOR type | first byte - --------------- | ------------------------------------------ | ---------------------------------- | --------------- - null | `null` | Null | 0xf6 - boolean | `true` | True | 0xf5 - boolean | `false` | False | 0xf4 - number_integer | -9223372036854775808..-2147483649 | Negative integer (8 bytes follow) | 0x3b - number_integer | -2147483648..-32769 | Negative integer (4 bytes follow) | 0x3a - number_integer | -32768..-129 | Negative integer (2 bytes follow) | 0x39 - number_integer | -128..-25 | Negative integer (1 byte follow) | 0x38 - number_integer | -24..-1 | Negative integer | 0x20..0x37 - number_integer | 0..23 | Integer | 0x00..0x17 - number_integer | 24..255 | Unsigned integer (1 byte follow) | 0x18 - number_integer | 256..65535 | Unsigned integer (2 bytes follow) | 0x19 - number_integer | 65536..4294967295 | Unsigned integer (4 bytes follow) | 0x1a - number_integer | 4294967296..18446744073709551615 | Unsigned integer (8 bytes follow) | 0x1b - number_unsigned | 0..23 | Integer | 0x00..0x17 - number_unsigned | 24..255 | Unsigned integer (1 byte follow) | 0x18 - number_unsigned | 256..65535 | Unsigned integer (2 bytes follow) | 0x19 - number_unsigned | 65536..4294967295 | Unsigned integer (4 bytes follow) | 0x1a - number_unsigned | 4294967296..18446744073709551615 | Unsigned integer (8 bytes follow) | 0x1b - number_float | *any value* | Double-Precision Float | 0xfb - string | *length*: 0..23 | UTF-8 string | 0x60..0x77 - string | *length*: 23..255 | UTF-8 string (1 byte follow) | 0x78 - string | *length*: 256..65535 | UTF-8 string (2 bytes follow) | 0x79 - string | *length*: 65536..4294967295 | UTF-8 string (4 bytes follow) | 0x7a - string | *length*: 4294967296..18446744073709551615 | UTF-8 string (8 bytes follow) | 0x7b - array | *size*: 0..23 | array | 0x80..0x97 - array | *size*: 23..255 | array (1 byte follow) | 0x98 - array | *size*: 256..65535 | array (2 bytes follow) | 0x99 - array | *size*: 65536..4294967295 | array (4 bytes follow) | 0x9a - array | *size*: 4294967296..18446744073709551615 | array (8 bytes follow) | 0x9b - object | *size*: 0..23 | map | 0xa0..0xb7 - object | *size*: 23..255 | map (1 byte follow) | 0xb8 - object | *size*: 256..65535 | map (2 bytes follow) | 0xb9 - object | *size*: 65536..4294967295 | map (4 bytes follow) | 0xba - object | *size*: 4294967296..18446744073709551615 | map (8 bytes follow) | 0xbb - - @note The mapping is **complete** in the sense that any JSON value type - can be converted to a CBOR value. - - @note The following CBOR types are not used in the conversion: - - byte strings (0x40..0x5f) - - UTF-8 strings terminated by "break" (0x7f) - - arrays terminated by "break" (0x9f) - - maps terminated by "break" (0xbf) - - date/time (0xc0..0xc1) - - bignum (0xc2..0xc3) - - decimal fraction (0xc4) - - bigfloat (0xc5) - - tagged items (0xc6..0xd4, 0xd8..0xdb) - - expected conversions (0xd5..0xd7) - - simple values (0xe0..0xf3, 0xf8) - - undefined (0xf7) - - half and single-precision floats (0xf9-0xfa) - - break (0xff) - - @param[in] j JSON value to serialize - @return MessagePack serialization as byte vector - - @complexity Linear in the size of the JSON value @a j. - - @liveexample{The example shows the serialization of a JSON value to a byte - vector in CBOR format.,to_cbor} - - @sa http://cbor.io - @sa @ref from_cbor(const std::vector&, const size_t) for the - analogous deserialization - @sa @ref to_msgpack(const basic_json& for the related MessagePack format - - @since version 2.0.9 - */ - static std::vector to_cbor(const basic_json& j) - { - std::vector result; - to_cbor_internal(j, result); - return result; - } - - /*! - @brief create a JSON value from a byte vector in CBOR format - - Deserializes a given byte vector @a v to a JSON value using the CBOR - (Concise Binary Object Representation) serialization format. - - The library maps CBOR types to JSON value types as follows: - - CBOR type | JSON value type | first byte - ---------------------- | --------------- | ---------- - Integer | number_unsigned | 0x00..0x17 - Unsigned integer | number_unsigned | 0x18 - Unsigned integer | number_unsigned | 0x19 - Unsigned integer | number_unsigned | 0x1a - Unsigned integer | number_unsigned | 0x1b - Negative integer | number_integer | 0x20..0x37 - Negative integer | number_integer | 0x38 - Negative integer | number_integer | 0x39 - Negative integer | number_integer | 0x3a - Negative integer | number_integer | 0x3b - Negative integer | number_integer | 0x40..0x57 - UTF-8 string | string | 0x60..0x77 - UTF-8 string | string | 0x78 - UTF-8 string | string | 0x79 - UTF-8 string | string | 0x7a - UTF-8 string | string | 0x7b - UTF-8 string | string | 0x7f - array | array | 0x80..0x97 - array | array | 0x98 - array | array | 0x99 - array | array | 0x9a - array | array | 0x9b - array | array | 0x9f - map | object | 0xa0..0xb7 - map | object | 0xb8 - map | object | 0xb9 - map | object | 0xba - map | object | 0xbb - map | object | 0xbf - False | `false` | 0xf4 - True | `true` | 0xf5 - Nill | `null` | 0xf6 - Half-Precision Float | number_float | 0xf9 - Single-Precision Float | number_float | 0xfa - Double-Precision Float | number_float | 0xfb - - @warning The mapping is **incomplete** in the sense that not all CBOR - types can be converted to a JSON value. The following CBOR types - are not supported and will yield parse errors (parse_error.112): - - byte strings (0x40..0x5f) - - date/time (0xc0..0xc1) - - bignum (0xc2..0xc3) - - decimal fraction (0xc4) - - bigfloat (0xc5) - - tagged items (0xc6..0xd4, 0xd8..0xdb) - - expected conversions (0xd5..0xd7) - - simple values (0xe0..0xf3, 0xf8) - - undefined (0xf7) - - @warning CBOR allows map keys of any type, whereas JSON only allows - strings as keys in object values. Therefore, CBOR maps with keys - other than UTF-8 strings are rejected (parse_error.113). - - @note Any CBOR output created @ref to_cbor can be successfully parsed by - @ref from_cbor. - - @param[in] v a byte vector in CBOR format - @param[in] start_index the index to start reading from @a v (0 by default) - @return deserialized JSON value - - @throw parse_error.110 if the given vector ends prematurely - @throw parse_error.112 if unsupported features from CBOR were - used in the given vector @a v or if the input is not valid CBOR - @throw parse_error.113 if a string was expected as map key, but not found - - @complexity Linear in the size of the byte vector @a v. - - @liveexample{The example shows the deserialization of a byte vector in CBOR - format to a JSON value.,from_cbor} - - @sa http://cbor.io - @sa @ref to_cbor(const basic_json&) for the analogous serialization - @sa @ref from_msgpack(const std::vector&, const size_t) for the - related MessagePack format - - @since version 2.0.9, parameter @a start_index since 2.1.1 - */ - static basic_json from_cbor(const std::vector& v, - const size_t start_index = 0) - { - size_t i = start_index; - return from_cbor_internal(v, i); - } - - /// @} - - /////////////////////////// - // convenience functions // - /////////////////////////// - - /*! - @brief return the type as string - - Returns the type name as string to be used in error messages - usually to - indicate that a function was called on a wrong JSON type. - - @return basically a string representation of a the @a m_type member - - @complexity Constant. - - @liveexample{The following code exemplifies `type_name()` for all JSON - types.,type_name} - - @since version 1.0.0, public since 2.1.0 - */ - std::string type_name() const - { - { - switch (m_type) - { - case value_t::null: - return "null"; - case value_t::object: - return "object"; - case value_t::array: - return "array"; - case value_t::string: - return "string"; - case value_t::boolean: - return "boolean"; - case value_t::discarded: - return "discarded"; - default: - return "number"; - } - } - } - - - private: - ////////////////////// - // member variables // - ////////////////////// - - /// the type of the current element - value_t m_type = value_t::null; - - /// the value of the current element - json_value m_value = {}; - - - private: - /////////////// - // iterators // - /////////////// - - /*! - @brief an iterator for primitive JSON types - - This class models an iterator for primitive JSON types (boolean, number, - string). It's only purpose is to allow the iterator/const_iterator classes - to "iterate" over primitive values. Internally, the iterator is modeled by - a `difference_type` variable. Value begin_value (`0`) models the begin, - end_value (`1`) models past the end. - */ - class primitive_iterator_t - { - public: - - difference_type get_value() const noexcept - { - return m_it; - } - /// set iterator to a defined beginning - void set_begin() noexcept - { - m_it = begin_value; - } - - /// set iterator to a defined past the end - void set_end() noexcept - { - m_it = end_value; - } - - /// return whether the iterator can be dereferenced - constexpr bool is_begin() const noexcept - { - return (m_it == begin_value); - } - - /// return whether the iterator is at end - constexpr bool is_end() const noexcept - { - return (m_it == end_value); - } - - friend constexpr bool operator==(primitive_iterator_t lhs, primitive_iterator_t rhs) noexcept - { - return lhs.m_it == rhs.m_it; - } - - friend constexpr bool operator!=(primitive_iterator_t lhs, primitive_iterator_t rhs) noexcept - { - return !(lhs == rhs); - } - - friend constexpr bool operator<(primitive_iterator_t lhs, primitive_iterator_t rhs) noexcept - { - return lhs.m_it < rhs.m_it; - } - - friend constexpr bool operator<=(primitive_iterator_t lhs, primitive_iterator_t rhs) noexcept - { - return lhs.m_it <= rhs.m_it; - } - - friend constexpr bool operator>(primitive_iterator_t lhs, primitive_iterator_t rhs) noexcept - { - return lhs.m_it > rhs.m_it; - } - - friend constexpr bool operator>=(primitive_iterator_t lhs, primitive_iterator_t rhs) noexcept - { - return lhs.m_it >= rhs.m_it; - } - - primitive_iterator_t operator+(difference_type i) - { - auto result = *this; - result += i; - return result; - } - - friend constexpr difference_type operator-(primitive_iterator_t lhs, primitive_iterator_t rhs) noexcept - { - return lhs.m_it - rhs.m_it; - } - - friend std::ostream& operator<<(std::ostream& os, primitive_iterator_t it) - { - return os << it.m_it; - } - - primitive_iterator_t& operator++() - { - ++m_it; - return *this; - } - - primitive_iterator_t operator++(int) - { - auto result = *this; - m_it++; - return result; - } - - primitive_iterator_t& operator--() - { - --m_it; - return *this; - } - - primitive_iterator_t operator--(int) - { - auto result = *this; - m_it--; - return result; - } - - primitive_iterator_t& operator+=(difference_type n) - { - m_it += n; - return *this; - } - - primitive_iterator_t& operator-=(difference_type n) - { - m_it -= n; - return *this; - } - - private: - static constexpr difference_type begin_value = 0; - static constexpr difference_type end_value = begin_value + 1; - - /// iterator as signed integer type - difference_type m_it = std::numeric_limits::denorm_min(); - }; - - /*! - @brief an iterator value - - @note This structure could easily be a union, but MSVC currently does not - allow unions members with complex constructors, see - https://github.com/nlohmann/json/pull/105. - */ - struct internal_iterator - { - /// iterator for JSON objects - typename object_t::iterator object_iterator; - /// iterator for JSON arrays - typename array_t::iterator array_iterator; - /// generic iterator for all other types - primitive_iterator_t primitive_iterator; - - /// create an uninitialized internal_iterator - internal_iterator() noexcept - : object_iterator(), array_iterator(), primitive_iterator() - {} - }; - - /// proxy class for the iterator_wrapper functions - template - class iteration_proxy - { - private: - /// helper class for iteration - class iteration_proxy_internal - { - private: - /// the iterator - IteratorType anchor; - /// an index for arrays (used to create key names) - size_t array_index = 0; - - public: - explicit iteration_proxy_internal(IteratorType it) noexcept - : anchor(it) - {} - - /// dereference operator (needed for range-based for) - iteration_proxy_internal& operator*() - { - return *this; - } - - /// increment operator (needed for range-based for) - iteration_proxy_internal& operator++() - { - ++anchor; - ++array_index; - - return *this; - } - - /// inequality operator (needed for range-based for) - bool operator!= (const iteration_proxy_internal& o) const - { - return anchor != o.anchor; - } - - /// return key of the iterator - typename basic_json::string_t key() const - { - assert(anchor.m_object != nullptr); - - switch (anchor.m_object->type()) - { - // use integer array index as key - case value_t::array: - { - return std::to_string(array_index); - } - - // use key from the object - case value_t::object: - { - return anchor.key(); - } - - // use an empty key for all primitive types - default: - { - return ""; - } - } - } - - /// return value of the iterator - typename IteratorType::reference value() const - { - return anchor.value(); - } - }; - - /// the container to iterate - typename IteratorType::reference container; - - public: - /// construct iteration proxy from a container - explicit iteration_proxy(typename IteratorType::reference cont) - : container(cont) - {} - - /// return iterator begin (needed for range-based for) - iteration_proxy_internal begin() noexcept - { - return iteration_proxy_internal(container.begin()); - } - - /// return iterator end (needed for range-based for) - iteration_proxy_internal end() noexcept - { - return iteration_proxy_internal(container.end()); - } - }; - - public: - /*! - @brief a template for a random access iterator for the @ref basic_json class - - This class implements a both iterators (iterator and const_iterator) for the - @ref basic_json class. - - @note An iterator is called *initialized* when a pointer to a JSON value - has been set (e.g., by a constructor or a copy assignment). If the - iterator is default-constructed, it is *uninitialized* and most - methods are undefined. **The library uses assertions to detect calls - on uninitialized iterators.** - - @requirement The class satisfies the following concept requirements: - - [RandomAccessIterator](http://en.cppreference.com/w/cpp/concept/RandomAccessIterator): - The iterator that can be moved to point (forward and backward) to any - element in constant time. - - @since version 1.0.0, simplified in version 2.0.9 - */ - template - class iter_impl : public std::iterator - { - /// allow basic_json to access private members - friend class basic_json; - - // make sure U is basic_json or const basic_json - static_assert(std::is_same::value - or std::is_same::value, - "iter_impl only accepts (const) basic_json"); - - public: - /// the type of the values when the iterator is dereferenced - using value_type = typename basic_json::value_type; - /// a type to represent differences between iterators - using difference_type = typename basic_json::difference_type; - /// defines a pointer to the type iterated over (value_type) - using pointer = typename std::conditional::value, - typename basic_json::const_pointer, - typename basic_json::pointer>::type; - /// defines a reference to the type iterated over (value_type) - using reference = typename std::conditional::value, - typename basic_json::const_reference, - typename basic_json::reference>::type; - /// the category of the iterator - using iterator_category = std::bidirectional_iterator_tag; - - /// default constructor - iter_impl() = default; - - /*! - @brief constructor for a given JSON instance - @param[in] object pointer to a JSON object for this iterator - @pre object != nullptr - @post The iterator is initialized; i.e. `m_object != nullptr`. - */ - explicit iter_impl(pointer object) noexcept - : m_object(object) - { - assert(m_object != nullptr); - - switch (m_object->m_type) - { - case basic_json::value_t::object: - { - m_it.object_iterator = typename object_t::iterator(); - break; - } - - case basic_json::value_t::array: - { - m_it.array_iterator = typename array_t::iterator(); - break; - } - - default: - { - m_it.primitive_iterator = primitive_iterator_t(); - break; - } - } - } - - /* - Use operator `const_iterator` instead of `const_iterator(const iterator& - other) noexcept` to avoid two class definitions for @ref iterator and - @ref const_iterator. - - This function is only called if this class is an @ref iterator. If this - class is a @ref const_iterator this function is not called. - */ - operator const_iterator() const - { - const_iterator ret; - - if (m_object) - { - ret.m_object = m_object; - ret.m_it = m_it; - } - - return ret; - } - - /*! - @brief copy constructor - @param[in] other iterator to copy from - @note It is not checked whether @a other is initialized. - */ - iter_impl(const iter_impl& other) noexcept - : m_object(other.m_object), m_it(other.m_it) - {} - - /*! - @brief copy assignment - @param[in,out] other iterator to copy from - @note It is not checked whether @a other is initialized. - */ - iter_impl& operator=(iter_impl other) noexcept( - std::is_nothrow_move_constructible::value and - std::is_nothrow_move_assignable::value and - std::is_nothrow_move_constructible::value and - std::is_nothrow_move_assignable::value - ) - { - std::swap(m_object, other.m_object); - std::swap(m_it, other.m_it); - return *this; - } - - private: - /*! - @brief set the iterator to the first value - @pre The iterator is initialized; i.e. `m_object != nullptr`. - */ - void set_begin() noexcept - { - assert(m_object != nullptr); - - switch (m_object->m_type) - { - case basic_json::value_t::object: - { - m_it.object_iterator = m_object->m_value.object->begin(); - break; - } - - case basic_json::value_t::array: - { - m_it.array_iterator = m_object->m_value.array->begin(); - break; - } - - case basic_json::value_t::null: - { - // set to end so begin()==end() is true: null is empty - m_it.primitive_iterator.set_end(); - break; - } - - default: - { - m_it.primitive_iterator.set_begin(); - break; - } - } - } - - /*! - @brief set the iterator past the last value - @pre The iterator is initialized; i.e. `m_object != nullptr`. - */ - void set_end() noexcept - { - assert(m_object != nullptr); - - switch (m_object->m_type) - { - case basic_json::value_t::object: - { - m_it.object_iterator = m_object->m_value.object->end(); - break; - } - - case basic_json::value_t::array: - { - m_it.array_iterator = m_object->m_value.array->end(); - break; - } - - default: - { - m_it.primitive_iterator.set_end(); - break; - } - } - } - - public: - /*! - @brief return a reference to the value pointed to by the iterator - @pre The iterator is initialized; i.e. `m_object != nullptr`. - */ - reference operator*() const - { - assert(m_object != nullptr); - - switch (m_object->m_type) - { - case basic_json::value_t::object: - { - assert(m_it.object_iterator != m_object->m_value.object->end()); - return m_it.object_iterator->second; - } - - case basic_json::value_t::array: - { - assert(m_it.array_iterator != m_object->m_value.array->end()); - return *m_it.array_iterator; - } - - case basic_json::value_t::null: - { - JSON_THROW(invalid_iterator(214, "cannot get value")); - } - - default: - { - if (m_it.primitive_iterator.is_begin()) - { - return *m_object; - } - - JSON_THROW(invalid_iterator(214, "cannot get value")); - } - } - } - - /*! - @brief dereference the iterator - @pre The iterator is initialized; i.e. `m_object != nullptr`. - */ - pointer operator->() const - { - assert(m_object != nullptr); - - switch (m_object->m_type) - { - case basic_json::value_t::object: - { - assert(m_it.object_iterator != m_object->m_value.object->end()); - return &(m_it.object_iterator->second); - } - - case basic_json::value_t::array: - { - assert(m_it.array_iterator != m_object->m_value.array->end()); - return &*m_it.array_iterator; - } - - default: - { - if (m_it.primitive_iterator.is_begin()) - { - return m_object; - } - - JSON_THROW(invalid_iterator(214, "cannot get value")); - } - } - } - - /*! - @brief post-increment (it++) - @pre The iterator is initialized; i.e. `m_object != nullptr`. - */ - iter_impl operator++(int) - { - auto result = *this; - ++(*this); - return result; - } - - /*! - @brief pre-increment (++it) - @pre The iterator is initialized; i.e. `m_object != nullptr`. - */ - iter_impl& operator++() - { - assert(m_object != nullptr); - - switch (m_object->m_type) - { - case basic_json::value_t::object: - { - std::advance(m_it.object_iterator, 1); - break; - } - - case basic_json::value_t::array: - { - std::advance(m_it.array_iterator, 1); - break; - } - - default: - { - ++m_it.primitive_iterator; - break; - } - } - - return *this; - } - - /*! - @brief post-decrement (it--) - @pre The iterator is initialized; i.e. `m_object != nullptr`. - */ - iter_impl operator--(int) - { - auto result = *this; - --(*this); - return result; - } - - /*! - @brief pre-decrement (--it) - @pre The iterator is initialized; i.e. `m_object != nullptr`. - */ - iter_impl& operator--() - { - assert(m_object != nullptr); - - switch (m_object->m_type) - { - case basic_json::value_t::object: - { - std::advance(m_it.object_iterator, -1); - break; - } - - case basic_json::value_t::array: - { - std::advance(m_it.array_iterator, -1); - break; - } - - default: - { - --m_it.primitive_iterator; - break; - } - } - - return *this; - } - - /*! - @brief comparison: equal - @pre The iterator is initialized; i.e. `m_object != nullptr`. - */ - bool operator==(const iter_impl& other) const - { - // if objects are not the same, the comparison is undefined - if (m_object != other.m_object) - { - JSON_THROW(invalid_iterator(212, "cannot compare iterators of different containers")); - } - - assert(m_object != nullptr); - - switch (m_object->m_type) - { - case basic_json::value_t::object: - { - return (m_it.object_iterator == other.m_it.object_iterator); - } - - case basic_json::value_t::array: - { - return (m_it.array_iterator == other.m_it.array_iterator); - } - - default: - { - return (m_it.primitive_iterator == other.m_it.primitive_iterator); - } - } - } - - /*! - @brief comparison: not equal - @pre The iterator is initialized; i.e. `m_object != nullptr`. - */ - bool operator!=(const iter_impl& other) const - { - return not operator==(other); - } - - /*! - @brief comparison: smaller - @pre The iterator is initialized; i.e. `m_object != nullptr`. - */ - bool operator<(const iter_impl& other) const - { - // if objects are not the same, the comparison is undefined - if (m_object != other.m_object) - { - JSON_THROW(invalid_iterator(212, "cannot compare iterators of different containers")); - } - - assert(m_object != nullptr); - - switch (m_object->m_type) - { - case basic_json::value_t::object: - { - JSON_THROW(invalid_iterator(213, "cannot compare order of object iterators")); - } - - case basic_json::value_t::array: - { - return (m_it.array_iterator < other.m_it.array_iterator); - } - - default: - { - return (m_it.primitive_iterator < other.m_it.primitive_iterator); - } - } - } - - /*! - @brief comparison: less than or equal - @pre The iterator is initialized; i.e. `m_object != nullptr`. - */ - bool operator<=(const iter_impl& other) const - { - return not other.operator < (*this); - } - - /*! - @brief comparison: greater than - @pre The iterator is initialized; i.e. `m_object != nullptr`. - */ - bool operator>(const iter_impl& other) const - { - return not operator<=(other); - } - - /*! - @brief comparison: greater than or equal - @pre The iterator is initialized; i.e. `m_object != nullptr`. - */ - bool operator>=(const iter_impl& other) const - { - return not operator<(other); - } - - /*! - @brief add to iterator - @pre The iterator is initialized; i.e. `m_object != nullptr`. - */ - iter_impl& operator+=(difference_type i) - { - assert(m_object != nullptr); - - switch (m_object->m_type) - { - case basic_json::value_t::object: - { - JSON_THROW(invalid_iterator(209, "cannot use offsets with object iterators")); - } - - case basic_json::value_t::array: - { - std::advance(m_it.array_iterator, i); - break; - } - - default: - { - m_it.primitive_iterator += i; - break; - } - } - - return *this; - } - - /*! - @brief subtract from iterator - @pre The iterator is initialized; i.e. `m_object != nullptr`. - */ - iter_impl& operator-=(difference_type i) - { - return operator+=(-i); - } - - /*! - @brief add to iterator - @pre The iterator is initialized; i.e. `m_object != nullptr`. - */ - iter_impl operator+(difference_type i) - { - auto result = *this; - result += i; - return result; - } - - /*! - @brief subtract from iterator - @pre The iterator is initialized; i.e. `m_object != nullptr`. - */ - iter_impl operator-(difference_type i) - { - auto result = *this; - result -= i; - return result; - } - - /*! - @brief return difference - @pre The iterator is initialized; i.e. `m_object != nullptr`. - */ - difference_type operator-(const iter_impl& other) const - { - assert(m_object != nullptr); - - switch (m_object->m_type) - { - case basic_json::value_t::object: - { - JSON_THROW(invalid_iterator(209, "cannot use offsets with object iterators")); - } - - case basic_json::value_t::array: - { - return m_it.array_iterator - other.m_it.array_iterator; - } - - default: - { - return m_it.primitive_iterator - other.m_it.primitive_iterator; - } - } - } - - /*! - @brief access to successor - @pre The iterator is initialized; i.e. `m_object != nullptr`. - */ - reference operator[](difference_type n) const - { - assert(m_object != nullptr); - - switch (m_object->m_type) - { - case basic_json::value_t::object: - { - JSON_THROW(invalid_iterator(208, "cannot use operator[] for object iterators")); - } - - case basic_json::value_t::array: - { - return *std::next(m_it.array_iterator, n); - } - - case basic_json::value_t::null: - { - JSON_THROW(invalid_iterator(214, "cannot get value")); - } - - default: - { - if (m_it.primitive_iterator.get_value() == -n) - { - return *m_object; - } - - JSON_THROW(invalid_iterator(214, "cannot get value")); - } - } - } - - /*! - @brief return the key of an object iterator - @pre The iterator is initialized; i.e. `m_object != nullptr`. - */ - typename object_t::key_type key() const - { - assert(m_object != nullptr); - - if (m_object->is_object()) - { - return m_it.object_iterator->first; - } - - JSON_THROW(invalid_iterator(207, "cannot use key() for non-object iterators")); - } - - /*! - @brief return the value of an iterator - @pre The iterator is initialized; i.e. `m_object != nullptr`. - */ - reference value() const - { - return operator*(); - } - - private: - /// associated JSON instance - pointer m_object = nullptr; - /// the actual iterator of the associated instance - internal_iterator m_it = internal_iterator(); - }; - - /*! - @brief a template for a reverse iterator class - - @tparam Base the base iterator type to reverse. Valid types are @ref - iterator (to create @ref reverse_iterator) and @ref const_iterator (to - create @ref const_reverse_iterator). - - @requirement The class satisfies the following concept requirements: - - [RandomAccessIterator](http://en.cppreference.com/w/cpp/concept/RandomAccessIterator): - The iterator that can be moved to point (forward and backward) to any - element in constant time. - - [OutputIterator](http://en.cppreference.com/w/cpp/concept/OutputIterator): - It is possible to write to the pointed-to element (only if @a Base is - @ref iterator). - - @since version 1.0.0 - */ - template - class json_reverse_iterator : public std::reverse_iterator - { - public: - /// shortcut to the reverse iterator adaptor - using base_iterator = std::reverse_iterator; - /// the reference type for the pointed-to element - using reference = typename Base::reference; - - /// create reverse iterator from iterator - json_reverse_iterator(const typename base_iterator::iterator_type& it) noexcept - : base_iterator(it) - {} - - /// create reverse iterator from base class - json_reverse_iterator(const base_iterator& it) noexcept - : base_iterator(it) - {} - - /// post-increment (it++) - json_reverse_iterator operator++(int) - { - return base_iterator::operator++(1); - } - - /// pre-increment (++it) - json_reverse_iterator& operator++() - { - base_iterator::operator++(); - return *this; - } - - /// post-decrement (it--) - json_reverse_iterator operator--(int) - { - return base_iterator::operator--(1); - } - - /// pre-decrement (--it) - json_reverse_iterator& operator--() - { - base_iterator::operator--(); - return *this; - } - - /// add to iterator - json_reverse_iterator& operator+=(difference_type i) - { - base_iterator::operator+=(i); - return *this; - } - - /// add to iterator - json_reverse_iterator operator+(difference_type i) const - { - auto result = *this; - result += i; - return result; - } - - /// subtract from iterator - json_reverse_iterator operator-(difference_type i) const - { - auto result = *this; - result -= i; - return result; - } - - /// return difference - difference_type operator-(const json_reverse_iterator& other) const - { - return this->base() - other.base(); - } - - /// access to successor - reference operator[](difference_type n) const - { - return *(this->operator+(n)); - } - - /// return the key of an object iterator - typename object_t::key_type key() const - { - auto it = --this->base(); - return it.key(); - } - - /// return the value of an iterator - reference value() const - { - auto it = --this->base(); - return it.operator * (); - } - }; - - - private: - ////////////////////// - // lexer and parser // - ////////////////////// - - /*! - @brief lexical analysis - - This class organizes the lexical analysis during JSON deserialization. The - core of it is a scanner generated by [re2c](http://re2c.org) that - processes a buffer and recognizes tokens according to RFC 7159. - */ - class lexer - { - public: - /// token types for the parser - enum class token_type - { - uninitialized, ///< indicating the scanner is uninitialized - literal_true, ///< the `true` literal - literal_false, ///< the `false` literal - literal_null, ///< the `null` literal - value_string, ///< a string -- use get_string() for actual value - value_unsigned, ///< an unsigned integer -- use get_number() for actual value - value_integer, ///< a signed integer -- use get_number() for actual value - value_float, ///< an floating point number -- use get_number() for actual value - begin_array, ///< the character for array begin `[` - begin_object, ///< the character for object begin `{` - end_array, ///< the character for array end `]` - end_object, ///< the character for object end `}` - name_separator, ///< the name separator `:` - value_separator, ///< the value separator `,` - parse_error, ///< indicating a parse error - end_of_input ///< indicating the end of the input buffer - }; - - /// the char type to use in the lexer - using lexer_char_t = unsigned char; - - /// a lexer from a buffer with given length - lexer(const lexer_char_t* buff, const size_t len) noexcept - : m_content(buff) - { - assert(m_content != nullptr); - m_start = m_cursor = m_content; - m_limit = m_content + len; - } - - /*! - @brief a lexer from an input stream - @throw parse_error.111 if input stream is in a bad state - */ - explicit lexer(std::istream& s) - : m_stream(&s), m_line_buffer() - { - // immediately abort if stream is erroneous - if (s.fail()) - { - JSON_THROW(parse_error(111, 0, "bad input stream")); - } - - // fill buffer - fill_line_buffer(); - - // skip UTF-8 byte-order mark - if (m_line_buffer.size() >= 3 and m_line_buffer.substr(0, 3) == "\xEF\xBB\xBF") - { - m_line_buffer[0] = ' '; - m_line_buffer[1] = ' '; - m_line_buffer[2] = ' '; - } - } - - // switch off unwanted functions (due to pointer members) - lexer() = delete; - lexer(const lexer&) = delete; - lexer operator=(const lexer&) = delete; - - /*! - @brief create a string from one or two Unicode code points - - There are two cases: (1) @a codepoint1 is in the Basic Multilingual - Plane (U+0000 through U+FFFF) and @a codepoint2 is 0, or (2) - @a codepoint1 and @a codepoint2 are a UTF-16 surrogate pair to - represent a code point above U+FFFF. - - @param[in] codepoint1 the code point (can be high surrogate) - @param[in] codepoint2 the code point (can be low surrogate or 0) - - @return string representation of the code point; the length of the - result string is between 1 and 4 characters. - - @throw parse_error.102 if the low surrogate is invalid; example: - `""missing or wrong low surrogate""` - @throw parse_error.103 if code point is > 0x10ffff; example: `"code - points above 0x10FFFF are invalid"` - - @complexity Constant. - - @see - */ - string_t to_unicode(const std::size_t codepoint1, - const std::size_t codepoint2 = 0) const - { - // calculate the code point from the given code points - std::size_t codepoint = codepoint1; - - // check if codepoint1 is a high surrogate - if (codepoint1 >= 0xD800 and codepoint1 <= 0xDBFF) - { - // check if codepoint2 is a low surrogate - if (codepoint2 >= 0xDC00 and codepoint2 <= 0xDFFF) - { - codepoint = - // high surrogate occupies the most significant 22 bits - (codepoint1 << 10) - // low surrogate occupies the least significant 15 bits - + codepoint2 - // there is still the 0xD800, 0xDC00 and 0x10000 noise - // in the result so we have to subtract with: - // (0xD800 << 10) + DC00 - 0x10000 = 0x35FDC00 - - 0x35FDC00; - } - else - { - JSON_THROW(parse_error(102, get_position(), "missing or wrong low surrogate")); - } - } - - string_t result; - - if (codepoint < 0x80) - { - // 1-byte characters: 0xxxxxxx (ASCII) - result.append(1, static_cast(codepoint)); - } - else if (codepoint <= 0x7ff) - { - // 2-byte characters: 110xxxxx 10xxxxxx - result.append(1, static_cast(0xC0 | ((codepoint >> 6) & 0x1F))); - result.append(1, static_cast(0x80 | (codepoint & 0x3F))); - } - else if (codepoint <= 0xffff) - { - // 3-byte characters: 1110xxxx 10xxxxxx 10xxxxxx - result.append(1, static_cast(0xE0 | ((codepoint >> 12) & 0x0F))); - result.append(1, static_cast(0x80 | ((codepoint >> 6) & 0x3F))); - result.append(1, static_cast(0x80 | (codepoint & 0x3F))); - } - else if (codepoint <= 0x10ffff) - { - // 4-byte characters: 11110xxx 10xxxxxx 10xxxxxx 10xxxxxx - result.append(1, static_cast(0xF0 | ((codepoint >> 18) & 0x07))); - result.append(1, static_cast(0x80 | ((codepoint >> 12) & 0x3F))); - result.append(1, static_cast(0x80 | ((codepoint >> 6) & 0x3F))); - result.append(1, static_cast(0x80 | (codepoint & 0x3F))); - } - else - { - JSON_THROW(parse_error(103, get_position(), "code points above 0x10FFFF are invalid")); - } - - return result; - } - - /// return name of values of type token_type (only used for errors) - static std::string token_type_name(const token_type t) - { - switch (t) - { - case token_type::uninitialized: - return ""; - case token_type::literal_true: - return "true literal"; - case token_type::literal_false: - return "false literal"; - case token_type::literal_null: - return "null literal"; - case token_type::value_string: - return "string literal"; - case lexer::token_type::value_unsigned: - case lexer::token_type::value_integer: - case lexer::token_type::value_float: - return "number literal"; - case token_type::begin_array: - return "'['"; - case token_type::begin_object: - return "'{'"; - case token_type::end_array: - return "']'"; - case token_type::end_object: - return "'}'"; - case token_type::name_separator: - return "':'"; - case token_type::value_separator: - return "','"; - case token_type::parse_error: - return ""; - case token_type::end_of_input: - return "end of input"; - default: - { - // catch non-enum values - return "unknown token"; // LCOV_EXCL_LINE - } - } - } - - /*! - This function implements a scanner for JSON. It is specified using - regular expressions that try to follow RFC 7159 as close as possible. - These regular expressions are then translated into a minimized - deterministic finite automaton (DFA) by the tool - [re2c](http://re2c.org). As a result, the translated code for this - function consists of a large block of code with `goto` jumps. - - @return the class of the next token read from the buffer - - @complexity Linear in the length of the input.\n - - Proposition: The loop below will always terminate for finite input.\n - - Proof (by contradiction): Assume a finite input. To loop forever, the - loop must never hit code with a `break` statement. The only code - snippets without a `break` statement is the continue statement for - whitespace. To loop forever, the input must be an infinite sequence - whitespace. This contradicts the assumption of finite input, q.e.d. - */ - token_type scan() - { - while (true) - { - // pointer for backtracking information - m_marker = nullptr; - - // remember the begin of the token - m_start = m_cursor; - assert(m_start != nullptr); - - /*!re2c - re2c:define:YYCTYPE = lexer_char_t; - re2c:define:YYCURSOR = m_cursor; - re2c:define:YYLIMIT = m_limit; - re2c:define:YYMARKER = m_marker; - re2c:define:YYFILL = "fill_line_buffer(@@); // LCOV_EXCL_LINE"; - re2c:define:YYFILL:naked = 1; - re2c:yyfill:enable = 1; - re2c:indent:string = " "; - re2c:indent:top = 1; - re2c:labelprefix = "basic_json_parser_"; - - // ignore whitespace - ws = [ \t\n\r]+; - ws { position += static_cast((m_cursor - m_start)); continue; } - - // structural characters - "[" { last_token_type = token_type::begin_array; break; } - "]" { last_token_type = token_type::end_array; break; } - "{" { last_token_type = token_type::begin_object; break; } - "}" { last_token_type = token_type::end_object; break; } - "," { last_token_type = token_type::value_separator; break; } - ":" { last_token_type = token_type::name_separator; break; } - - // literal names - "null" { last_token_type = token_type::literal_null; break; } - "true" { last_token_type = token_type::literal_true; break; } - "false" { last_token_type = token_type::literal_false; break; } - - // number - decimal_point = "."; - digit = [0-9]; - digit_1_9 = [1-9]; - e = "e" | "E"; - minus = "-"; - plus = "+"; - zero = "0"; - exp = e (minus | plus)? digit+; - frac = decimal_point digit+; - int = (zero | digit_1_9 digit*); - invalid_int = minus? "0" digit+; - invalid_int { last_token_type = token_type::parse_error; break; } - number_unsigned = int; - number_unsigned { last_token_type = token_type::value_unsigned; break; } - number_integer = minus int; - number_integer { last_token_type = token_type::value_integer; break; } - number_float = minus? int frac? exp?; - number_float { last_token_type = token_type::value_float; break; } - - // string - quotation_mark = "\""; - escape = "\\"; - unescaped = [^"\\\x00-\x1f]; - single_escaped = "\"" | "\\" | "/" | "b" | "f" | "n" | "r" | "t"; - unicode_escaped = "u" [0-9a-fA-F]{4}; - escaped = escape (single_escaped | unicode_escaped); - char = unescaped | escaped; - string = quotation_mark char* quotation_mark; - string { last_token_type = token_type::value_string; break; } - - // end of file - "\x00" { last_token_type = token_type::end_of_input; break; } - - // anything else is an error - * { last_token_type = token_type::parse_error; break; } - */ - } - - position += static_cast((m_cursor - m_start)); - return last_token_type; - } - - /*! - @brief append data from the stream to the line buffer - - This function is called by the scan() function when the end of the - buffer (`m_limit`) is reached and the `m_cursor` pointer cannot be - incremented without leaving the limits of the line buffer. Note re2c - decides when to call this function. - - If the lexer reads from contiguous storage, there is no trailing null - byte. Therefore, this function must make sure to add these padding - null bytes. - - If the lexer reads from an input stream, this function reads the next - line of the input. - - @pre - p p p p p p u u u u u x . . . . . . - ^ ^ ^ ^ - m_content m_start | m_limit - m_cursor - - @post - u u u u u x x x x x x x . . . . . . - ^ ^ ^ - | m_cursor m_limit - m_start - m_content - */ - void fill_line_buffer(size_t n = 0) - { - // if line buffer is used, m_content points to its data - assert(m_line_buffer.empty() - or m_content == reinterpret_cast(m_line_buffer.data())); - - // if line buffer is used, m_limit is set past the end of its data - assert(m_line_buffer.empty() - or m_limit == m_content + m_line_buffer.size()); - - // pointer relationships - assert(m_content <= m_start); - assert(m_start <= m_cursor); - assert(m_cursor <= m_limit); - assert(m_marker == nullptr or m_marker <= m_limit); - - // number of processed characters (p) - const auto num_processed_chars = static_cast(m_start - m_content); - // offset for m_marker wrt. to m_start - const auto offset_marker = (m_marker == nullptr) ? 0 : m_marker - m_start; - // number of unprocessed characters (u) - const auto offset_cursor = m_cursor - m_start; - - // no stream is used or end of file is reached - if (m_stream == nullptr or m_stream->eof()) - { - // m_start may or may not be pointing into m_line_buffer at - // this point. We trust the standard library to do the right - // thing. See http://stackoverflow.com/q/28142011/266378 - m_line_buffer.assign(m_start, m_limit); - - // append n characters to make sure that there is sufficient - // space between m_cursor and m_limit - m_line_buffer.append(1, '\x00'); - if (n > 0) - { - m_line_buffer.append(n - 1, '\x01'); - } - } - else - { - // delete processed characters from line buffer - m_line_buffer.erase(0, num_processed_chars); - // read next line from input stream - m_line_buffer_tmp.clear(); - - // check if stream is still good - if (m_stream->fail()) - { - JSON_THROW(parse_error(111, 0, "bad input stream")); - } - - std::getline(*m_stream, m_line_buffer_tmp, '\n'); - - // add line with newline symbol to the line buffer - m_line_buffer += m_line_buffer_tmp; - m_line_buffer.push_back('\n'); - } - - // set pointers - m_content = reinterpret_cast(m_line_buffer.data()); - assert(m_content != nullptr); - m_start = m_content; - m_marker = m_start + offset_marker; - m_cursor = m_start + offset_cursor; - m_limit = m_start + m_line_buffer.size(); - } - - /// return string representation of last read token - string_t get_token_string() const - { - assert(m_start != nullptr); - return string_t(reinterpret_cast(m_start), - static_cast(m_cursor - m_start)); - } - - /*! - @brief return string value for string tokens - - The function iterates the characters between the opening and closing - quotes of the string value. The complete string is the range - [m_start,m_cursor). Consequently, we iterate from m_start+1 to - m_cursor-1. - - We differentiate two cases: - - 1. Escaped characters. In this case, a new character is constructed - according to the nature of the escape. Some escapes create new - characters (e.g., `"\\n"` is replaced by `"\n"`), some are copied - as is (e.g., `"\\\\"`). Furthermore, Unicode escapes of the shape - `"\\uxxxx"` need special care. In this case, to_unicode takes care - of the construction of the values. - 2. Unescaped characters are copied as is. - - @pre `m_cursor - m_start >= 2`, meaning the length of the last token - is at least 2 bytes which is trivially true for any string (which - consists of at least two quotes). - - " c1 c2 c3 ... " - ^ ^ - m_start m_cursor - - @complexity Linear in the length of the string.\n - - Lemma: The loop body will always terminate.\n - - Proof (by contradiction): Assume the loop body does not terminate. As - the loop body does not contain another loop, one of the called - functions must never return. The called functions are `std::strtoul` - and to_unicode. Neither function can loop forever, so the loop body - will never loop forever which contradicts the assumption that the loop - body does not terminate, q.e.d.\n - - Lemma: The loop condition for the for loop is eventually false.\n - - Proof (by contradiction): Assume the loop does not terminate. Due to - the above lemma, this can only be due to a tautological loop - condition; that is, the loop condition i < m_cursor - 1 must always be - true. Let x be the change of i for any loop iteration. Then - m_start + 1 + x < m_cursor - 1 must hold to loop indefinitely. This - can be rephrased to m_cursor - m_start - 2 > x. With the - precondition, we x <= 0, meaning that the loop condition holds - indefinitely if i is always decreased. However, observe that the value - of i is strictly increasing with each iteration, as it is incremented - by 1 in the iteration expression and never decremented inside the loop - body. Hence, the loop condition will eventually be false which - contradicts the assumption that the loop condition is a tautology, - q.e.d. - - @return string value of current token without opening and closing - quotes - @throw parse_error.102 if to_unicode fails or surrogate error - @throw parse_error.103 if to_unicode fails - */ - string_t get_string() const - { - assert(m_cursor - m_start >= 2); - - string_t result; - result.reserve(static_cast(m_cursor - m_start - 2)); - - // iterate the result between the quotes - for (const lexer_char_t* i = m_start + 1; i < m_cursor - 1; ++i) - { - // find next escape character - auto e = std::find(i, m_cursor - 1, '\\'); - if (e != i) - { - // see https://github.com/nlohmann/json/issues/365#issuecomment-262874705 - for (auto k = i; k < e; k++) - { - result.push_back(static_cast(*k)); - } - i = e - 1; // -1 because of ++i - } - else - { - // processing escaped character - // read next character - ++i; - - switch (*i) - { - // the default escapes - case 't': - { - result += "\t"; - break; - } - case 'b': - { - result += "\b"; - break; - } - case 'f': - { - result += "\f"; - break; - } - case 'n': - { - result += "\n"; - break; - } - case 'r': - { - result += "\r"; - break; - } - case '\\': - { - result += "\\"; - break; - } - case '/': - { - result += "/"; - break; - } - case '"': - { - result += "\""; - break; - } - - // unicode - case 'u': - { - // get code xxxx from uxxxx - auto codepoint = std::strtoul(std::string(reinterpret_cast(i + 1), - 4).c_str(), nullptr, 16); - - // check if codepoint is a high surrogate - if (codepoint >= 0xD800 and codepoint <= 0xDBFF) - { - // make sure there is a subsequent unicode - if ((i + 6 >= m_limit) or * (i + 5) != '\\' or * (i + 6) != 'u') - { - JSON_THROW(parse_error(102, get_position(), "missing low surrogate")); - } - - // get code yyyy from uxxxx\uyyyy - auto codepoint2 = std::strtoul(std::string(reinterpret_cast - (i + 7), 4).c_str(), nullptr, 16); - result += to_unicode(codepoint, codepoint2); - // skip the next 10 characters (xxxx\uyyyy) - i += 10; - } - else if (codepoint >= 0xDC00 and codepoint <= 0xDFFF) - { - // we found a lone low surrogate - JSON_THROW(parse_error(102, get_position(), "missing high surrogate")); - } - else - { - // add unicode character(s) - result += to_unicode(codepoint); - // skip the next four characters (xxxx) - i += 4; - } - break; - } - } - } - } - - return result; - } - - - /*! - @brief parse string into a built-in arithmetic type as if the current - locale is POSIX. - - @note in floating-point case strtod may parse past the token's end - - this is not an error - - @note any leading blanks are not handled - */ - struct strtonum - { - public: - strtonum(const char* start, const char* end) - : m_start(start), m_end(end) - {} - - /*! - @return true iff parsed successfully as number of type T - - @param[in,out] val shall contain parsed value, or undefined value - if could not parse - */ - template::value>::type> - bool to(T& val) const - { - return parse(val, std::is_integral()); - } - - private: - const char* const m_start = nullptr; - const char* const m_end = nullptr; - - // floating-point conversion - - // overloaded wrappers for strtod/strtof/strtold - // that will be called from parse - static void strtof(float& f, const char* str, char** endptr) - { - f = std::strtof(str, endptr); - } - - static void strtof(double& f, const char* str, char** endptr) - { - f = std::strtod(str, endptr); - } - - static void strtof(long double& f, const char* str, char** endptr) - { - f = std::strtold(str, endptr); - } - - template - bool parse(T& value, /*is_integral=*/std::false_type) const - { - // replace decimal separator with locale-specific version, - // when necessary; data will point to either the original - // string, or buf, or tempstr containing the fixed string. - std::string tempstr; - std::array buf; - const size_t len = static_cast(m_end - m_start); - - // lexer will reject empty numbers - assert(len > 0); - - // since dealing with strtod family of functions, we're - // getting the decimal point char from the C locale facilities - // instead of C++'s numpunct facet of the current std::locale - const auto loc = localeconv(); - assert(loc != nullptr); - const char decimal_point_char = (loc->decimal_point == nullptr) ? '.' : loc->decimal_point[0]; - - const char* data = m_start; - - if (decimal_point_char != '.') - { - const size_t ds_pos = static_cast(std::find(m_start, m_end, '.') - m_start); - - if (ds_pos != len) - { - // copy the data into the local buffer or tempstr, if - // buffer is too small; replace decimal separator, and - // update data to point to the modified bytes - if ((len + 1) < buf.size()) - { - std::copy(m_start, m_end, buf.begin()); - buf[len] = 0; - buf[ds_pos] = decimal_point_char; - data = buf.data(); - } - else - { - tempstr.assign(m_start, m_end); - tempstr[ds_pos] = decimal_point_char; - data = tempstr.c_str(); - } - } - } - - char* endptr = nullptr; - value = 0; - // this calls appropriate overload depending on T - strtof(value, data, &endptr); - - // parsing was successful iff strtof parsed exactly the number - // of characters determined by the lexer (len) - const bool ok = (endptr == (data + len)); - - if (ok and (value == static_cast(0.0)) and (*data == '-')) - { - // some implementations forget to negate the zero - value = -0.0; - } - - return ok; - } - - // integral conversion - - signed long long parse_integral(char** endptr, /*is_signed*/std::true_type) const - { - return std::strtoll(m_start, endptr, 10); - } - - unsigned long long parse_integral(char** endptr, /*is_signed*/std::false_type) const - { - return std::strtoull(m_start, endptr, 10); - } - - template - bool parse(T& value, /*is_integral=*/std::true_type) const - { - char* endptr = nullptr; - errno = 0; // these are thread-local - const auto x = parse_integral(&endptr, std::is_signed()); - - // called right overload? - static_assert(std::is_signed() == std::is_signed(), ""); - - value = static_cast(x); - - return (x == static_cast(value)) // x fits into destination T - and (x < 0) == (value < 0) // preserved sign - //and ((x != 0) or is_integral()) // strto[u]ll did nto fail - and (errno == 0) // strto[u]ll did not overflow - and (m_start < m_end) // token was not empty - and (endptr == m_end); // parsed entire token exactly - } - }; - - /*! - @brief return number value for number tokens - - This function translates the last token into the most appropriate - number type (either integer, unsigned integer or floating point), - which is passed back to the caller via the result parameter. - - integral numbers that don't fit into the the range of the respective - type are parsed as number_float_t - - floating-point values do not satisfy std::isfinite predicate - are converted to value_t::null - - throws if the entire string [m_start .. m_cursor) cannot be - interpreted as a number - - @param[out] result @ref basic_json object to receive the number. - @param[in] token the type of the number token - */ - bool get_number(basic_json& result, const token_type token) const - { - assert(m_start != nullptr); - assert(m_start < m_cursor); - assert((token == token_type::value_unsigned) or - (token == token_type::value_integer) or - (token == token_type::value_float)); - - strtonum num_converter(reinterpret_cast(m_start), - reinterpret_cast(m_cursor)); - - switch (token) - { - case lexer::token_type::value_unsigned: - { - number_unsigned_t val; - if (num_converter.to(val)) - { - // parsing successful - result.m_type = value_t::number_unsigned; - result.m_value = val; - return true; - } - break; - } - - case lexer::token_type::value_integer: - { - number_integer_t val; - if (num_converter.to(val)) - { - // parsing successful - result.m_type = value_t::number_integer; - result.m_value = val; - return true; - } - break; - } - - default: - { - break; - } - } - - // parse float (either explicitly or because a previous conversion - // failed) - number_float_t val; - if (num_converter.to(val)) - { - // parsing successful - result.m_type = value_t::number_float; - result.m_value = val; - - // throw in case of infinity or NAN - if (not std::isfinite(result.m_value.number_float)) - { - JSON_THROW(out_of_range(406, "number overflow parsing '" + get_token_string() + "'")); - } - - return true; - } - - // couldn't parse number in any format - return false; - } - - constexpr size_t get_position() const - { - return position; - } - - private: - /// optional input stream - std::istream* m_stream = nullptr; - /// line buffer buffer for m_stream - string_t m_line_buffer {}; - /// used for filling m_line_buffer - string_t m_line_buffer_tmp {}; - /// the buffer pointer - const lexer_char_t* m_content = nullptr; - /// pointer to the beginning of the current symbol - const lexer_char_t* m_start = nullptr; - /// pointer for backtracking information - const lexer_char_t* m_marker = nullptr; - /// pointer to the current symbol - const lexer_char_t* m_cursor = nullptr; - /// pointer to the end of the buffer - const lexer_char_t* m_limit = nullptr; - /// the last token type - token_type last_token_type = token_type::end_of_input; - /// current position in the input (read bytes) - size_t position = 0; - }; - - /*! - @brief syntax analysis - - This class implements a recursive decent parser. - */ - class parser - { - public: - /// a parser reading from a string literal - parser(const char* buff, const parser_callback_t cb = nullptr) - : callback(cb), - m_lexer(reinterpret_cast(buff), std::strlen(buff)) - {} - - /*! - @brief a parser reading from an input stream - @throw parse_error.111 if input stream is in a bad state - */ - parser(std::istream& is, const parser_callback_t cb = nullptr) - : callback(cb), m_lexer(is) - {} - - /// a parser reading from an iterator range with contiguous storage - template::iterator_category, std::random_access_iterator_tag>::value - , int>::type - = 0> - parser(IteratorType first, IteratorType last, const parser_callback_t cb = nullptr) - : callback(cb), - m_lexer(reinterpret_cast(&(*first)), - static_cast(std::distance(first, last))) - {} - - /*! - @brief public parser interface - @throw parse_error.101 in case of an unexpected token - @throw parse_error.102 if to_unicode fails or surrogate error - @throw parse_error.103 if to_unicode fails - */ - basic_json parse() - { - // read first token - get_token(); - - basic_json result = parse_internal(true); - result.assert_invariant(); - - expect(lexer::token_type::end_of_input); - - // return parser result and replace it with null in case the - // top-level value was discarded by the callback function - return result.is_discarded() ? basic_json() : std::move(result); - } - - private: - /*! - @brief the actual parser - @throw parse_error.101 in case of an unexpected token - @throw parse_error.102 if to_unicode fails or surrogate error - @throw parse_error.103 if to_unicode fails - */ - basic_json parse_internal(bool keep) - { - auto result = basic_json(value_t::discarded); - - switch (last_token) - { - case lexer::token_type::begin_object: - { - if (keep and (not callback - or ((keep = callback(depth++, parse_event_t::object_start, result)) != 0))) - { - // explicitly set result to object to cope with {} - result.m_type = value_t::object; - result.m_value = value_t::object; - } - - // read next token - get_token(); - - // closing } -> we are done - if (last_token == lexer::token_type::end_object) - { - get_token(); - if (keep and callback and not callback(--depth, parse_event_t::object_end, result)) - { - result = basic_json(value_t::discarded); - } - return result; - } - - // no comma is expected here - unexpect(lexer::token_type::value_separator); - - // otherwise: parse key-value pairs - do - { - // ugly, but could be fixed with loop reorganization - if (last_token == lexer::token_type::value_separator) - { - get_token(); - } - - // store key - expect(lexer::token_type::value_string); - const auto key = m_lexer.get_string(); - - bool keep_tag = false; - if (keep) - { - if (callback) - { - basic_json k(key); - keep_tag = callback(depth, parse_event_t::key, k); - } - else - { - keep_tag = true; - } - } - - // parse separator (:) - get_token(); - expect(lexer::token_type::name_separator); - - // parse and add value - get_token(); - auto value = parse_internal(keep); - if (keep and keep_tag and not value.is_discarded()) - { - result[key] = std::move(value); - } - } - while (last_token == lexer::token_type::value_separator); - - // closing } - expect(lexer::token_type::end_object); - get_token(); - if (keep and callback and not callback(--depth, parse_event_t::object_end, result)) - { - result = basic_json(value_t::discarded); - } - - return result; - } - - case lexer::token_type::begin_array: - { - if (keep and (not callback - or ((keep = callback(depth++, parse_event_t::array_start, result)) != 0))) - { - // explicitly set result to object to cope with [] - result.m_type = value_t::array; - result.m_value = value_t::array; - } - - // read next token - get_token(); - - // closing ] -> we are done - if (last_token == lexer::token_type::end_array) - { - get_token(); - if (callback and not callback(--depth, parse_event_t::array_end, result)) - { - result = basic_json(value_t::discarded); - } - return result; - } - - // no comma is expected here - unexpect(lexer::token_type::value_separator); - - // otherwise: parse values - do - { - // ugly, but could be fixed with loop reorganization - if (last_token == lexer::token_type::value_separator) - { - get_token(); - } - - // parse value - auto value = parse_internal(keep); - if (keep and not value.is_discarded()) - { - result.push_back(std::move(value)); - } - } - while (last_token == lexer::token_type::value_separator); - - // closing ] - expect(lexer::token_type::end_array); - get_token(); - if (keep and callback and not callback(--depth, parse_event_t::array_end, result)) - { - result = basic_json(value_t::discarded); - } - - return result; - } - - case lexer::token_type::literal_null: - { - get_token(); - result.m_type = value_t::null; - break; - } - - case lexer::token_type::value_string: - { - const auto s = m_lexer.get_string(); - get_token(); - result = basic_json(s); - break; - } - - case lexer::token_type::literal_true: - { - get_token(); - result.m_type = value_t::boolean; - result.m_value = true; - break; - } - - case lexer::token_type::literal_false: - { - get_token(); - result.m_type = value_t::boolean; - result.m_value = false; - break; - } - - case lexer::token_type::value_unsigned: - case lexer::token_type::value_integer: - case lexer::token_type::value_float: - { - m_lexer.get_number(result, last_token); - get_token(); - break; - } - - default: - { - // the last token was unexpected - unexpect(last_token); - } - } - - if (keep and callback and not callback(depth, parse_event_t::value, result)) - { - result = basic_json(value_t::discarded); - } - return result; - } - - /// get next token from lexer - typename lexer::token_type get_token() - { - last_token = m_lexer.scan(); - return last_token; - } - - /*! - @throw parse_error.101 if expected token did not occur - */ - void expect(typename lexer::token_type t) const - { - if (t != last_token) - { - std::string error_msg = "parse error - unexpected "; - error_msg += (last_token == lexer::token_type::parse_error ? ("'" + m_lexer.get_token_string() + - "'") : - lexer::token_type_name(last_token)); - error_msg += "; expected " + lexer::token_type_name(t); - JSON_THROW(parse_error(101, m_lexer.get_position(), error_msg)); - } - } - - /*! - @throw parse_error.101 if unexpected token occurred - */ - void unexpect(typename lexer::token_type t) const - { - if (t == last_token) - { - std::string error_msg = "parse error - unexpected "; - error_msg += (last_token == lexer::token_type::parse_error ? ("'" + m_lexer.get_token_string() + - "'") : - lexer::token_type_name(last_token)); - JSON_THROW(parse_error(101, m_lexer.get_position(), error_msg)); - } - } - - private: - /// current level of recursion - int depth = 0; - /// callback function - const parser_callback_t callback = nullptr; - /// the type of the last read token - typename lexer::token_type last_token = lexer::token_type::uninitialized; - /// the lexer - lexer m_lexer; - }; - - public: - /*! - @brief JSON Pointer - - A JSON pointer defines a string syntax for identifying a specific value - within a JSON document. It can be used with functions `at` and - `operator[]`. Furthermore, JSON pointers are the base for JSON patches. - - @sa [RFC 6901](https://tools.ietf.org/html/rfc6901) - - @since version 2.0.0 - */ - class json_pointer - { - /// allow basic_json to access private members - friend class basic_json; - - public: - /*! - @brief create JSON pointer - - Create a JSON pointer according to the syntax described in - [Section 3 of RFC6901](https://tools.ietf.org/html/rfc6901#section-3). - - @param[in] s string representing the JSON pointer; if omitted, the - empty string is assumed which references the whole JSON - value - - @throw parse_error.107 if the given JSON pointer @a s is nonempty and - does not begin with a slash (`/`); see example below - - @throw parse_error.108 if a tilde (`~`) in the given JSON pointer @a s - is not followed by `0` (representing `~`) or `1` (representing `/`); - see example below - - @liveexample{The example shows the construction several valid JSON - pointers as well as the exceptional behavior.,json_pointer} - - @since version 2.0.0 - */ - explicit json_pointer(const std::string& s = "") - : reference_tokens(split(s)) - {} - - /*! - @brief return a string representation of the JSON pointer - - @invariant For each JSON pointer `ptr`, it holds: - @code {.cpp} - ptr == json_pointer(ptr.to_string()); - @endcode - - @return a string representation of the JSON pointer - - @liveexample{The example shows the result of `to_string`., - json_pointer__to_string} - - @since version 2.0.0 - */ - std::string to_string() const noexcept - { - return std::accumulate(reference_tokens.begin(), - reference_tokens.end(), std::string{}, - [](const std::string & a, const std::string & b) - { - return a + "/" + escape(b); - }); - } - - /// @copydoc to_string() - operator std::string() const - { - return to_string(); - } - - private: - /*! - @brief remove and return last reference pointer - @throw out_of_range.405 if JSON pointer has no parent - */ - std::string pop_back() - { - if (is_root()) - { - JSON_THROW(out_of_range(405, "JSON pointer has no parent")); - } - - auto last = reference_tokens.back(); - reference_tokens.pop_back(); - return last; - } - - /// return whether pointer points to the root document - bool is_root() const - { - return reference_tokens.empty(); - } - - json_pointer top() const - { - if (is_root()) - { - JSON_THROW(out_of_range(405, "JSON pointer has no parent")); - } - - json_pointer result = *this; - result.reference_tokens = {reference_tokens[0]}; - return result; - } - - /*! - @brief create and return a reference to the pointed to value - - @complexity Linear in the number of reference tokens. - - @throw parse_error.109 if array index is not a number - @throw type_error.313 if value cannot be unflattened - */ - reference get_and_create(reference j) const - { - pointer result = &j; - - // in case no reference tokens exist, return a reference to the - // JSON value j which will be overwritten by a primitive value - for (const auto& reference_token : reference_tokens) - { - switch (result->m_type) - { - case value_t::null: - { - if (reference_token == "0") - { - // start a new array if reference token is 0 - result = &result->operator[](0); - } - else - { - // start a new object otherwise - result = &result->operator[](reference_token); - } - break; - } - - case value_t::object: - { - // create an entry in the object - result = &result->operator[](reference_token); - break; - } - - case value_t::array: - { - // create an entry in the array - JSON_TRY - { - result = &result->operator[](static_cast(std::stoi(reference_token))); - } - JSON_CATCH(std::invalid_argument&) - { - JSON_THROW(parse_error(109, 0, "array index '" + reference_token + "' is not a number")); - } - break; - } - - /* - The following code is only reached if there exists a - reference token _and_ the current value is primitive. In - this case, we have an error situation, because primitive - values may only occur as single value; that is, with an - empty list of reference tokens. - */ - default: - { - JSON_THROW(type_error(313, "invalid value to unflatten")); - } - } - } - - return *result; - } - - /*! - @brief return a reference to the pointed to value - - @note This version does not throw if a value is not present, but tries - to create nested values instead. For instance, calling this function - with pointer `"/this/that"` on a null value is equivalent to calling - `operator[]("this").operator[]("that")` on that value, effectively - changing the null value to an object. - - @param[in] ptr a JSON value - - @return reference to the JSON value pointed to by the JSON pointer - - @complexity Linear in the length of the JSON pointer. - - @throw parse_error.106 if an array index begins with '0' - @throw parse_error.109 if an array index was not a number - @throw out_of_range.404 if the JSON pointer can not be resolved - */ - reference get_unchecked(pointer ptr) const - { - for (const auto& reference_token : reference_tokens) - { - // convert null values to arrays or objects before continuing - if (ptr->m_type == value_t::null) - { - // check if reference token is a number - const bool nums = std::all_of(reference_token.begin(), - reference_token.end(), - [](const char x) - { - return (x >= '0' and x <= '9'); - }); - - // change value to array for numbers or "-" or to object - // otherwise - if (nums or reference_token == "-") - { - *ptr = value_t::array; - } - else - { - *ptr = value_t::object; - } - } - - switch (ptr->m_type) - { - case value_t::object: - { - // use unchecked object access - ptr = &ptr->operator[](reference_token); - break; - } - - case value_t::array: - { - // error condition (cf. RFC 6901, Sect. 4) - if (reference_token.size() > 1 and reference_token[0] == '0') - { - JSON_THROW(parse_error(106, 0, "array index '" + reference_token + "' must not begin with '0'")); - } - - if (reference_token == "-") - { - // explicitly treat "-" as index beyond the end - ptr = &ptr->operator[](ptr->m_value.array->size()); - } - else - { - // convert array index to number; unchecked access - JSON_TRY - { - ptr = &ptr->operator[](static_cast(std::stoi(reference_token))); - } - JSON_CATCH(std::invalid_argument&) - { - JSON_THROW(parse_error(109, 0, "array index '" + reference_token + "' is not a number")); - } - } - break; - } - - default: - { - JSON_THROW(out_of_range(404, "unresolved reference token '" + reference_token + "'")); - } - } - } - - return *ptr; - } - - /*! - @throw parse_error.106 if an array index begins with '0' - @throw parse_error.109 if an array index was not a number - @throw out_of_range.402 if the array index '-' is used - @throw out_of_range.404 if the JSON pointer can not be resolved - */ - reference get_checked(pointer ptr) const - { - for (const auto& reference_token : reference_tokens) - { - switch (ptr->m_type) - { - case value_t::object: - { - // note: at performs range check - ptr = &ptr->at(reference_token); - break; - } - - case value_t::array: - { - if (reference_token == "-") - { - // "-" always fails the range check - JSON_THROW(out_of_range(402, "array index '-' (" + - std::to_string(ptr->m_value.array->size()) + - ") is out of range")); - } - - // error condition (cf. RFC 6901, Sect. 4) - if (reference_token.size() > 1 and reference_token[0] == '0') - { - JSON_THROW(parse_error(106, 0, "array index '" + reference_token + "' must not begin with '0'")); - } - - // note: at performs range check - JSON_TRY - { - ptr = &ptr->at(static_cast(std::stoi(reference_token))); - } - JSON_CATCH(std::invalid_argument&) - { - JSON_THROW(parse_error(109, 0, "array index '" + reference_token + "' is not a number")); - } - break; - } - - default: - { - JSON_THROW(out_of_range(404, "unresolved reference token '" + reference_token + "'")); - } - } - } - - return *ptr; - } - - /*! - @brief return a const reference to the pointed to value - - @param[in] ptr a JSON value - - @return const reference to the JSON value pointed to by the JSON - pointer - - @throw parse_error.106 if an array index begins with '0' - @throw parse_error.109 if an array index was not a number - @throw out_of_range.402 if the array index '-' is used - @throw out_of_range.404 if the JSON pointer can not be resolved - */ - const_reference get_unchecked(const_pointer ptr) const - { - for (const auto& reference_token : reference_tokens) - { - switch (ptr->m_type) - { - case value_t::object: - { - // use unchecked object access - ptr = &ptr->operator[](reference_token); - break; - } - - case value_t::array: - { - if (reference_token == "-") - { - // "-" cannot be used for const access - JSON_THROW(out_of_range(402, "array index '-' (" + - std::to_string(ptr->m_value.array->size()) + - ") is out of range")); - } - - // error condition (cf. RFC 6901, Sect. 4) - if (reference_token.size() > 1 and reference_token[0] == '0') - { - JSON_THROW(parse_error(106, 0, "array index '" + reference_token + "' must not begin with '0'")); - } - - // use unchecked array access - JSON_TRY - { - ptr = &ptr->operator[](static_cast(std::stoi(reference_token))); - } - JSON_CATCH(std::invalid_argument&) - { - JSON_THROW(parse_error(109, 0, "array index '" + reference_token + "' is not a number")); - } - break; - } - - default: - { - JSON_THROW(out_of_range(404, "unresolved reference token '" + reference_token + "'")); - } - } - } - - return *ptr; - } - - /*! - @throw parse_error.106 if an array index begins with '0' - @throw parse_error.109 if an array index was not a number - @throw out_of_range.402 if the array index '-' is used - @throw out_of_range.404 if the JSON pointer can not be resolved - */ - const_reference get_checked(const_pointer ptr) const - { - for (const auto& reference_token : reference_tokens) - { - switch (ptr->m_type) - { - case value_t::object: - { - // note: at performs range check - ptr = &ptr->at(reference_token); - break; - } - - case value_t::array: - { - if (reference_token == "-") - { - // "-" always fails the range check - JSON_THROW(out_of_range(402, "array index '-' (" + - std::to_string(ptr->m_value.array->size()) + - ") is out of range")); - } - - // error condition (cf. RFC 6901, Sect. 4) - if (reference_token.size() > 1 and reference_token[0] == '0') - { - JSON_THROW(parse_error(106, 0, "array index '" + reference_token + "' must not begin with '0'")); - } - - // note: at performs range check - JSON_TRY - { - ptr = &ptr->at(static_cast(std::stoi(reference_token))); - } - JSON_CATCH(std::invalid_argument&) - { - JSON_THROW(parse_error(109, 0, "array index '" + reference_token + "' is not a number")); - } - break; - } - - default: - { - JSON_THROW(out_of_range(404, "unresolved reference token '" + reference_token + "'")); - } - } - } - - return *ptr; - } - - /*! - @brief split the string input to reference tokens - - @note This function is only called by the json_pointer constructor. - All exceptions below are documented there. - - @throw parse_error.107 if the pointer is not empty or begins with '/' - @throw parse_error.108 if character '~' is not followed by '0' or '1' - */ - static std::vector split(const std::string& reference_string) - { - std::vector result; - - // special case: empty reference string -> no reference tokens - if (reference_string.empty()) - { - return result; - } - - // check if nonempty reference string begins with slash - if (reference_string[0] != '/') - { - JSON_THROW(parse_error(107, 1, "JSON pointer must be empty or begin with '/' - was: '" + reference_string + "'")); - } - - // extract the reference tokens: - // - slash: position of the last read slash (or end of string) - // - start: position after the previous slash - for ( - // search for the first slash after the first character - size_t slash = reference_string.find_first_of('/', 1), - // set the beginning of the first reference token - start = 1; - // we can stop if start == string::npos+1 = 0 - start != 0; - // set the beginning of the next reference token - // (will eventually be 0 if slash == std::string::npos) - start = slash + 1, - // find next slash - slash = reference_string.find_first_of('/', start)) - { - // use the text between the beginning of the reference token - // (start) and the last slash (slash). - auto reference_token = reference_string.substr(start, slash - start); - - // check reference tokens are properly escaped - for (size_t pos = reference_token.find_first_of('~'); - pos != std::string::npos; - pos = reference_token.find_first_of('~', pos + 1)) - { - assert(reference_token[pos] == '~'); - - // ~ must be followed by 0 or 1 - if (pos == reference_token.size() - 1 or - (reference_token[pos + 1] != '0' and - reference_token[pos + 1] != '1')) - { - JSON_THROW(parse_error(108, 0, "escape character '~' must be followed with '0' or '1'")); - } - } - - // finally, store the reference token - unescape(reference_token); - result.push_back(reference_token); - } - - return result; - } - - /*! - @brief replace all occurrences of a substring by another string - - @param[in,out] s the string to manipulate; changed so that all - occurrences of @a f are replaced with @a t - @param[in] f the substring to replace with @a t - @param[in] t the string to replace @a f - - @pre The search string @a f must not be empty. **This precondition is - enforced with an assertion.** - - @since version 2.0.0 - */ - static void replace_substring(std::string& s, - const std::string& f, - const std::string& t) - { - assert(not f.empty()); - - for ( - size_t pos = s.find(f); // find first occurrence of f - pos != std::string::npos; // make sure f was found - s.replace(pos, f.size(), t), // replace with t - pos = s.find(f, pos + t.size()) // find next occurrence of f - ); - } - - /// escape tilde and slash - static std::string escape(std::string s) - { - // escape "~"" to "~0" and "/" to "~1" - replace_substring(s, "~", "~0"); - replace_substring(s, "/", "~1"); - return s; - } - - /// unescape tilde and slash - static void unescape(std::string& s) - { - // first transform any occurrence of the sequence '~1' to '/' - replace_substring(s, "~1", "/"); - // then transform any occurrence of the sequence '~0' to '~' - replace_substring(s, "~0", "~"); - } - - /*! - @param[in] reference_string the reference string to the current value - @param[in] value the value to consider - @param[in,out] result the result object to insert values to - - @note Empty objects or arrays are flattened to `null`. - */ - static void flatten(const std::string& reference_string, - const basic_json& value, - basic_json& result) - { - switch (value.m_type) - { - case value_t::array: - { - if (value.m_value.array->empty()) - { - // flatten empty array as null - result[reference_string] = nullptr; - } - else - { - // iterate array and use index as reference string - for (size_t i = 0; i < value.m_value.array->size(); ++i) - { - flatten(reference_string + "/" + std::to_string(i), - value.m_value.array->operator[](i), result); - } - } - break; - } - - case value_t::object: - { - if (value.m_value.object->empty()) - { - // flatten empty object as null - result[reference_string] = nullptr; - } - else - { - // iterate object and use keys as reference string - for (const auto& element : *value.m_value.object) - { - flatten(reference_string + "/" + escape(element.first), - element.second, result); - } - } - break; - } - - default: - { - // add primitive value with its reference string - result[reference_string] = value; - break; - } - } - } - - /*! - @param[in] value flattened JSON - - @return unflattened JSON - - @throw parse_error.109 if array index is not a number - @throw type_error.314 if value is not an object - @throw type_error.315 if object values are not primitive - @throw type_error.313 if value cannot be unflattened - */ - static basic_json unflatten(const basic_json& value) - { - if (not value.is_object()) - { - JSON_THROW(type_error(314, "only objects can be unflattened")); - } - - basic_json result; - - // iterate the JSON object values - for (const auto& element : *value.m_value.object) - { - if (not element.second.is_primitive()) - { - JSON_THROW(type_error(315, "values in object must be primitive")); - } - - // assign value to reference pointed to by JSON pointer; Note - // that if the JSON pointer is "" (i.e., points to the whole - // value), function get_and_create returns a reference to - // result itself. An assignment will then create a primitive - // value. - json_pointer(element.first).get_and_create(result) = element.second; - } - - return result; - } - - friend bool operator==(json_pointer const& lhs, - json_pointer const& rhs) noexcept - { - return lhs.reference_tokens == rhs.reference_tokens; - } - - friend bool operator!=(json_pointer const& lhs, - json_pointer const& rhs) noexcept - { - return !(lhs == rhs); - } - - /// the reference tokens - std::vector reference_tokens {}; - }; - - ////////////////////////// - // JSON Pointer support // - ////////////////////////// - - /// @name JSON Pointer functions - /// @{ - - /*! - @brief access specified element via JSON Pointer - - Uses a JSON pointer to retrieve a reference to the respective JSON value. - No bound checking is performed. Similar to @ref operator[](const typename - object_t::key_type&), `null` values are created in arrays and objects if - necessary. - - In particular: - - If the JSON pointer points to an object key that does not exist, it - is created an filled with a `null` value before a reference to it - is returned. - - If the JSON pointer points to an array index that does not exist, it - is created an filled with a `null` value before a reference to it - is returned. All indices between the current maximum and the given - index are also filled with `null`. - - The special value `-` is treated as a synonym for the index past the - end. - - @param[in] ptr a JSON pointer - - @return reference to the element pointed to by @a ptr - - @complexity Constant. - - @throw parse_error.106 if an array index begins with '0' - @throw parse_error.109 if an array index was not a number - @throw out_of_range.404 if the JSON pointer can not be resolved - - @liveexample{The behavior is shown in the example.,operatorjson_pointer} - - @since version 2.0.0 - */ - reference operator[](const json_pointer& ptr) - { - return ptr.get_unchecked(this); - } - - /*! - @brief access specified element via JSON Pointer - - Uses a JSON pointer to retrieve a reference to the respective JSON value. - No bound checking is performed. The function does not change the JSON - value; no `null` values are created. In particular, the the special value - `-` yields an exception. - - @param[in] ptr JSON pointer to the desired element - - @return const reference to the element pointed to by @a ptr - - @complexity Constant. - - @throw parse_error.106 if an array index begins with '0' - @throw parse_error.109 if an array index was not a number - @throw out_of_range.402 if the array index '-' is used - @throw out_of_range.404 if the JSON pointer can not be resolved - - @liveexample{The behavior is shown in the example.,operatorjson_pointer_const} - - @since version 2.0.0 - */ - const_reference operator[](const json_pointer& ptr) const - { - return ptr.get_unchecked(this); - } - - /*! - @brief access specified element via JSON Pointer - - Returns a reference to the element at with specified JSON pointer @a ptr, - with bounds checking. - - @param[in] ptr JSON pointer to the desired element - - @return reference to the element pointed to by @a ptr - - @throw parse_error.106 if an array index in the passed JSON pointer @a ptr - begins with '0'. See example below. - - @throw parse_error.109 if an array index in the passed JSON pointer @a ptr - is not a number. See example below. - - @throw out_of_range.401 if an array index in the passed JSON pointer @a ptr - is out of range. See example below. - - @throw out_of_range.402 if the array index '-' is used in the passed JSON - pointer @a ptr. As `at` provides checked access (and no elements are - implicitly inserted), the index '-' is always invalid. See example below. - - @throw out_of_range.404 if the JSON pointer @a ptr can not be resolved. - See example below. - - @exceptionsafety Strong guarantee: if an exception is thrown, there are no - changes in the JSON value. - - @complexity Constant. - - @since version 2.0.0 - - @liveexample{The behavior is shown in the example.,at_json_pointer} - */ - reference at(const json_pointer& ptr) - { - return ptr.get_checked(this); - } - - /*! - @brief access specified element via JSON Pointer - - Returns a const reference to the element at with specified JSON pointer @a - ptr, with bounds checking. - - @param[in] ptr JSON pointer to the desired element - - @return reference to the element pointed to by @a ptr - - @throw parse_error.106 if an array index in the passed JSON pointer @a ptr - begins with '0'. See example below. - - @throw parse_error.109 if an array index in the passed JSON pointer @a ptr - is not a number. See example below. - - @throw out_of_range.401 if an array index in the passed JSON pointer @a ptr - is out of range. See example below. - - @throw out_of_range.402 if the array index '-' is used in the passed JSON - pointer @a ptr. As `at` provides checked access (and no elements are - implicitly inserted), the index '-' is always invalid. See example below. - - @throw out_of_range.404 if the JSON pointer @a ptr can not be resolved. - See example below. - - @exceptionsafety Strong guarantee: if an exception is thrown, there are no - changes in the JSON value. - - @complexity Constant. - - @since version 2.0.0 - - @liveexample{The behavior is shown in the example.,at_json_pointer_const} - */ - const_reference at(const json_pointer& ptr) const - { - return ptr.get_checked(this); - } - - /*! - @brief return flattened JSON value - - The function creates a JSON object whose keys are JSON pointers (see [RFC - 6901](https://tools.ietf.org/html/rfc6901)) and whose values are all - primitive. The original JSON value can be restored using the @ref - unflatten() function. - - @return an object that maps JSON pointers to primitive values - - @note Empty objects and arrays are flattened to `null` and will not be - reconstructed correctly by the @ref unflatten() function. - - @complexity Linear in the size the JSON value. - - @liveexample{The following code shows how a JSON object is flattened to an - object whose keys consist of JSON pointers.,flatten} - - @sa @ref unflatten() for the reverse function - - @since version 2.0.0 - */ - basic_json flatten() const - { - basic_json result(value_t::object); - json_pointer::flatten("", *this, result); - return result; - } - - /*! - @brief unflatten a previously flattened JSON value - - The function restores the arbitrary nesting of a JSON value that has been - flattened before using the @ref flatten() function. The JSON value must - meet certain constraints: - 1. The value must be an object. - 2. The keys must be JSON pointers (see - [RFC 6901](https://tools.ietf.org/html/rfc6901)) - 3. The mapped values must be primitive JSON types. - - @return the original JSON from a flattened version - - @note Empty objects and arrays are flattened by @ref flatten() to `null` - values and can not unflattened to their original type. Apart from - this example, for a JSON value `j`, the following is always true: - `j == j.flatten().unflatten()`. - - @complexity Linear in the size the JSON value. - - @throw type_error.314 if value is not an object - @throw type_error.315 if object values are not primitve - - @liveexample{The following code shows how a flattened JSON object is - unflattened into the original nested JSON object.,unflatten} - - @sa @ref flatten() for the reverse function - - @since version 2.0.0 - */ - basic_json unflatten() const - { - return json_pointer::unflatten(*this); - } - - /// @} - - ////////////////////////// - // JSON Patch functions // - ////////////////////////// - - /// @name JSON Patch functions - /// @{ - - /*! - @brief applies a JSON patch - - [JSON Patch](http://jsonpatch.com) defines a JSON document structure for - expressing a sequence of operations to apply to a JSON) document. With - this function, a JSON Patch is applied to the current JSON value by - executing all operations from the patch. - - @param[in] json_patch JSON patch document - @return patched document - - @note The application of a patch is atomic: Either all operations succeed - and the patched document is returned or an exception is thrown. In - any case, the original value is not changed: the patch is applied - to a copy of the value. - - @throw parse_error.104 if the JSON patch does not consist of an array of - objects - - @throw parse_error.105 if the JSON patch is malformed (e.g., mandatory - attributes are missing); example: `"operation add must have member path"` - - @throw out_of_range.401 if an array index is out of range. - - @throw out_of_range.403 if a JSON pointer inside the patch could not be - resolved successfully in the current JSON value; example: `"key baz not - found"` - - @throw out_of_range.405 if JSON pointer has no parent ("add", "remove", - "move") - - @throw other_error.501 if "test" operation was unsuccessful - - @complexity Linear in the size of the JSON value and the length of the - JSON patch. As usually only a fraction of the JSON value is affected by - the patch, the complexity can usually be neglected. - - @liveexample{The following code shows how a JSON patch is applied to a - value.,patch} - - @sa @ref diff -- create a JSON patch by comparing two JSON values - - @sa [RFC 6902 (JSON Patch)](https://tools.ietf.org/html/rfc6902) - @sa [RFC 6901 (JSON Pointer)](https://tools.ietf.org/html/rfc6901) - - @since version 2.0.0 - */ - basic_json patch(const basic_json& json_patch) const - { - // make a working copy to apply the patch to - basic_json result = *this; - - // the valid JSON Patch operations - enum class patch_operations {add, remove, replace, move, copy, test, invalid}; - - const auto get_op = [](const std::string & op) - { - if (op == "add") - { - return patch_operations::add; - } - if (op == "remove") - { - return patch_operations::remove; - } - if (op == "replace") - { - return patch_operations::replace; - } - if (op == "move") - { - return patch_operations::move; - } - if (op == "copy") - { - return patch_operations::copy; - } - if (op == "test") - { - return patch_operations::test; - } - - return patch_operations::invalid; - }; - - // wrapper for "add" operation; add value at ptr - const auto operation_add = [&result](json_pointer & ptr, basic_json val) - { - // adding to the root of the target document means replacing it - if (ptr.is_root()) - { - result = val; - } - else - { - // make sure the top element of the pointer exists - json_pointer top_pointer = ptr.top(); - if (top_pointer != ptr) - { - result.at(top_pointer); - } - - // get reference to parent of JSON pointer ptr - const auto last_path = ptr.pop_back(); - basic_json& parent = result[ptr]; - - switch (parent.m_type) - { - case value_t::null: - case value_t::object: - { - // use operator[] to add value - parent[last_path] = val; - break; - } - - case value_t::array: - { - if (last_path == "-") - { - // special case: append to back - parent.push_back(val); - } - else - { - const auto idx = std::stoi(last_path); - if (static_cast(idx) > parent.size()) - { - // avoid undefined behavior - JSON_THROW(out_of_range(401, "array index " + std::to_string(idx) + " is out of range")); - } - else - { - // default case: insert add offset - parent.insert(parent.begin() + static_cast(idx), val); - } - } - break; - } - - default: - { - // if there exists a parent it cannot be primitive - assert(false); // LCOV_EXCL_LINE - } - } - } - }; - - // wrapper for "remove" operation; remove value at ptr - const auto operation_remove = [&result](json_pointer & ptr) - { - // get reference to parent of JSON pointer ptr - const auto last_path = ptr.pop_back(); - basic_json& parent = result.at(ptr); - - // remove child - if (parent.is_object()) - { - // perform range check - auto it = parent.find(last_path); - if (it != parent.end()) - { - parent.erase(it); - } - else - { - JSON_THROW(out_of_range(403, "key '" + last_path + "' not found")); - } - } - else if (parent.is_array()) - { - // note erase performs range check - parent.erase(static_cast(std::stoi(last_path))); - } - }; - - // type check: top level value must be an array - if (not json_patch.is_array()) - { - JSON_THROW(parse_error(104, 0, "JSON patch must be an array of objects")); - } - - // iterate and apply the operations - for (const auto& val : json_patch) - { - // wrapper to get a value for an operation - const auto get_value = [&val](const std::string & op, - const std::string & member, - bool string_type) -> basic_json& - { - // find value - auto it = val.m_value.object->find(member); - - // context-sensitive error message - const auto error_msg = (op == "op") ? "operation" : "operation '" + op + "'"; - - // check if desired value is present - if (it == val.m_value.object->end()) - { - JSON_THROW(parse_error(105, 0, error_msg + " must have member '" + member + "'")); - } - - // check if result is of type string - if (string_type and not it->second.is_string()) - { - JSON_THROW(parse_error(105, 0, error_msg + " must have string member '" + member + "'")); - } - - // no error: return value - return it->second; - }; - - // type check: every element of the array must be an object - if (not val.is_object()) - { - JSON_THROW(parse_error(104, 0, "JSON patch must be an array of objects")); - } - - // collect mandatory members - const std::string op = get_value("op", "op", true); - const std::string path = get_value(op, "path", true); - json_pointer ptr(path); - - switch (get_op(op)) - { - case patch_operations::add: - { - operation_add(ptr, get_value("add", "value", false)); - break; - } - - case patch_operations::remove: - { - operation_remove(ptr); - break; - } - - case patch_operations::replace: - { - // the "path" location must exist - use at() - result.at(ptr) = get_value("replace", "value", false); - break; - } - - case patch_operations::move: - { - const std::string from_path = get_value("move", "from", true); - json_pointer from_ptr(from_path); - - // the "from" location must exist - use at() - basic_json v = result.at(from_ptr); - - // The move operation is functionally identical to a - // "remove" operation on the "from" location, followed - // immediately by an "add" operation at the target - // location with the value that was just removed. - operation_remove(from_ptr); - operation_add(ptr, v); - break; - } - - case patch_operations::copy: - { - const std::string from_path = get_value("copy", "from", true);; - const json_pointer from_ptr(from_path); - - // the "from" location must exist - use at() - result[ptr] = result.at(from_ptr); - break; - } - - case patch_operations::test: - { - bool success = false; - JSON_TRY - { - // check if "value" matches the one at "path" - // the "path" location must exist - use at() - success = (result.at(ptr) == get_value("test", "value", false)); - } - JSON_CATCH (out_of_range&) - { - // ignore out of range errors: success remains false - } - - // throw an exception if test fails - if (not success) - { - JSON_THROW(other_error(501, "unsuccessful: " + val.dump())); - } - - break; - } - - case patch_operations::invalid: - { - // op must be "add", "remove", "replace", "move", "copy", or - // "test" - JSON_THROW(parse_error(105, 0, "operation value '" + op + "' is invalid")); - } - } - } - - return result; - } - - /*! - @brief creates a diff as a JSON patch - - Creates a [JSON Patch](http://jsonpatch.com) so that value @a source can - be changed into the value @a target by calling @ref patch function. - - @invariant For two JSON values @a source and @a target, the following code - yields always `true`: - @code {.cpp} - source.patch(diff(source, target)) == target; - @endcode - - @note Currently, only `remove`, `add`, and `replace` operations are - generated. - - @param[in] source JSON value to compare from - @param[in] target JSON value to compare against - @param[in] path helper value to create JSON pointers - - @return a JSON patch to convert the @a source to @a target - - @complexity Linear in the lengths of @a source and @a target. - - @liveexample{The following code shows how a JSON patch is created as a - diff for two JSON values.,diff} - - @sa @ref patch -- apply a JSON patch - - @sa [RFC 6902 (JSON Patch)](https://tools.ietf.org/html/rfc6902) - - @since version 2.0.0 - */ - static basic_json diff(const basic_json& source, - const basic_json& target, - const std::string& path = "") - { - // the patch - basic_json result(value_t::array); - - // if the values are the same, return empty patch - if (source == target) - { - return result; - } - - if (source.type() != target.type()) - { - // different types: replace value - result.push_back( - { - {"op", "replace"}, - {"path", path}, - {"value", target} - }); - } - else - { - switch (source.type()) - { - case value_t::array: - { - // first pass: traverse common elements - size_t i = 0; - while (i < source.size() and i < target.size()) - { - // recursive call to compare array values at index i - auto temp_diff = diff(source[i], target[i], path + "/" + std::to_string(i)); - result.insert(result.end(), temp_diff.begin(), temp_diff.end()); - ++i; - } - - // i now reached the end of at least one array - // in a second pass, traverse the remaining elements - - // remove my remaining elements - const auto end_index = static_cast(result.size()); - while (i < source.size()) - { - // add operations in reverse order to avoid invalid - // indices - result.insert(result.begin() + end_index, object( - { - {"op", "remove"}, - {"path", path + "/" + std::to_string(i)} - })); - ++i; - } - - // add other remaining elements - while (i < target.size()) - { - result.push_back( - { - {"op", "add"}, - {"path", path + "/" + std::to_string(i)}, - {"value", target[i]} - }); - ++i; - } - - break; - } - - case value_t::object: - { - // first pass: traverse this object's elements - for (auto it = source.begin(); it != source.end(); ++it) - { - // escape the key name to be used in a JSON patch - const auto key = json_pointer::escape(it.key()); - - if (target.find(it.key()) != target.end()) - { - // recursive call to compare object values at key it - auto temp_diff = diff(it.value(), target[it.key()], path + "/" + key); - result.insert(result.end(), temp_diff.begin(), temp_diff.end()); - } - else - { - // found a key that is not in o -> remove it - result.push_back(object( - { - {"op", "remove"}, - {"path", path + "/" + key} - })); - } - } - - // second pass: traverse other object's elements - for (auto it = target.begin(); it != target.end(); ++it) - { - if (source.find(it.key()) == source.end()) - { - // found a key that is not in this -> add it - const auto key = json_pointer::escape(it.key()); - result.push_back( - { - {"op", "add"}, - {"path", path + "/" + key}, - {"value", it.value()} - }); - } - } - - break; - } - - default: - { - // both primitive type: replace value - result.push_back( - { - {"op", "replace"}, - {"path", path}, - {"value", target} - }); - break; - } - } - } - - return result; - } - - /// @} -}; - -///////////// -// presets // -///////////// - -/*! -@brief default JSON class - -This type is the default specialization of the @ref basic_json class which -uses the standard template types. - -@since version 1.0.0 -*/ -using json = basic_json<>; -} // namespace nlohmann - - -/////////////////////// -// nonmember support // -/////////////////////// - -// specialization of std::swap, and std::hash -namespace std -{ -/*! -@brief exchanges the values of two JSON objects - -@since version 1.0.0 -*/ -template<> -inline void swap(nlohmann::json& j1, - nlohmann::json& j2) noexcept( - is_nothrow_move_constructible::value and - is_nothrow_move_assignable::value - ) -{ - j1.swap(j2); -} - -/// hash value for JSON objects -template<> -struct hash -{ - /*! - @brief return a hash value for a JSON object - - @since version 1.0.0 - */ - std::size_t operator()(const nlohmann::json& j) const - { - // a naive hashing via the string representation - const auto& h = hash(); - return h(j.dump()); - } -}; - -/// specialization for std::less -template <> -struct less<::nlohmann::detail::value_t> -{ - /*! - @brief compare two value_t enum values - @since version 3.0.0 - */ - bool operator()(nlohmann::detail::value_t lhs, - nlohmann::detail::value_t rhs) const noexcept - { - return nlohmann::detail::operator<(lhs, rhs); - } -}; - -} // namespace std - -/*! -@brief user-defined string literal for JSON values - -This operator implements a user-defined string literal for JSON objects. It -can be used by adding `"_json"` to a string literal and returns a JSON object -if no parse error occurred. - -@param[in] s a string representation of a JSON object -@param[in] n the length of string @a s -@return a JSON object - -@since version 1.0.0 -*/ -inline nlohmann::json operator "" _json(const char* s, std::size_t n) -{ - return nlohmann::json::parse(s, s + n); -} - -/*! -@brief user-defined string literal for JSON pointer - -This operator implements a user-defined string literal for JSON Pointers. It -can be used by adding `"_json_pointer"` to a string literal and returns a JSON pointer -object if no parse error occurred. - -@param[in] s a string representation of a JSON Pointer -@param[in] n the length of string @a s -@return a JSON pointer object - -@since version 2.0.0 -*/ -inline nlohmann::json::json_pointer operator "" _json_pointer(const char* s, std::size_t n) -{ - return nlohmann::json::json_pointer(std::string(s, n)); -} - -// restore GCC/clang diagnostic settings -#if defined(__clang__) || defined(__GNUC__) || defined(__GNUG__) - #pragma GCC diagnostic pop -#endif - -// clean up -#undef JSON_CATCH -#undef JSON_THROW -#undef JSON_TRY - -#endif diff --git a/test/src/unit-class_lexer.cpp b/test/src/unit-class_lexer.cpp index b88a5291..d4bf2dd2 100644 --- a/test/src/unit-class_lexer.cpp +++ b/test/src/unit-class_lexer.cpp @@ -38,79 +38,50 @@ TEST_CASE("lexer class") { SECTION("structural characters") { - CHECK((json::lexer(reinterpret_cast("["), - 1).scan() == json::lexer::token_type::begin_array)); - CHECK((json::lexer(reinterpret_cast("]"), - 1).scan() == json::lexer::token_type::end_array)); - CHECK((json::lexer(reinterpret_cast("{"), - 1).scan() == json::lexer::token_type::begin_object)); - CHECK((json::lexer(reinterpret_cast("}"), - 1).scan() == json::lexer::token_type::end_object)); - CHECK((json::lexer(reinterpret_cast(","), - 1).scan() == json::lexer::token_type::value_separator)); - CHECK((json::lexer(reinterpret_cast(":"), - 1).scan() == json::lexer::token_type::name_separator)); + CHECK((json::lexer("[", 1).scan() == json::lexer::token_type::begin_array)); + CHECK((json::lexer("]", 1).scan() == json::lexer::token_type::end_array)); + CHECK((json::lexer("{", 1).scan() == json::lexer::token_type::begin_object)); + CHECK((json::lexer("}", 1).scan() == json::lexer::token_type::end_object)); + CHECK((json::lexer(",", 1).scan() == json::lexer::token_type::value_separator)); + CHECK((json::lexer(":", 1).scan() == json::lexer::token_type::name_separator)); } SECTION("literal names") { - CHECK((json::lexer(reinterpret_cast("null"), - 4).scan() == json::lexer::token_type::literal_null)); - CHECK((json::lexer(reinterpret_cast("true"), - 4).scan() == json::lexer::token_type::literal_true)); - CHECK((json::lexer(reinterpret_cast("false"), - 5).scan() == json::lexer::token_type::literal_false)); + CHECK((json::lexer("null", 4).scan() == json::lexer::token_type::literal_null)); + CHECK((json::lexer("true", 4).scan() == json::lexer::token_type::literal_true)); + CHECK((json::lexer("false", 5).scan() == json::lexer::token_type::literal_false)); } SECTION("numbers") { - CHECK((json::lexer(reinterpret_cast("0"), - 1).scan() == json::lexer::token_type::value_unsigned)); - CHECK((json::lexer(reinterpret_cast("1"), - 1).scan() == json::lexer::token_type::value_unsigned)); - CHECK((json::lexer(reinterpret_cast("2"), - 1).scan() == json::lexer::token_type::value_unsigned)); - CHECK((json::lexer(reinterpret_cast("3"), - 1).scan() == json::lexer::token_type::value_unsigned)); - CHECK((json::lexer(reinterpret_cast("4"), - 1).scan() == json::lexer::token_type::value_unsigned)); - CHECK((json::lexer(reinterpret_cast("5"), - 1).scan() == json::lexer::token_type::value_unsigned)); - CHECK((json::lexer(reinterpret_cast("6"), - 1).scan() == json::lexer::token_type::value_unsigned)); - CHECK((json::lexer(reinterpret_cast("7"), - 1).scan() == json::lexer::token_type::value_unsigned)); - CHECK((json::lexer(reinterpret_cast("8"), - 1).scan() == json::lexer::token_type::value_unsigned)); - CHECK((json::lexer(reinterpret_cast("9"), - 1).scan() == json::lexer::token_type::value_unsigned)); + CHECK((json::lexer("0", 1).scan() == json::lexer::token_type::value_unsigned)); + CHECK((json::lexer("1", 1).scan() == json::lexer::token_type::value_unsigned)); + CHECK((json::lexer("2", 1).scan() == json::lexer::token_type::value_unsigned)); + CHECK((json::lexer("3", 1).scan() == json::lexer::token_type::value_unsigned)); + CHECK((json::lexer("4", 1).scan() == json::lexer::token_type::value_unsigned)); + CHECK((json::lexer("5", 1).scan() == json::lexer::token_type::value_unsigned)); + CHECK((json::lexer("6", 1).scan() == json::lexer::token_type::value_unsigned)); + CHECK((json::lexer("7", 1).scan() == json::lexer::token_type::value_unsigned)); + CHECK((json::lexer("8", 1).scan() == json::lexer::token_type::value_unsigned)); + CHECK((json::lexer("9", 1).scan() == json::lexer::token_type::value_unsigned)); - CHECK((json::lexer(reinterpret_cast("-0"), - 2).scan() == json::lexer::token_type::value_integer)); - CHECK((json::lexer(reinterpret_cast("-1"), - 2).scan() == json::lexer::token_type::value_integer)); + CHECK((json::lexer("-0", 2).scan() == json::lexer::token_type::value_integer)); + CHECK((json::lexer("-1", 2).scan() == json::lexer::token_type::value_integer)); - CHECK((json::lexer(reinterpret_cast("1.1"), - 3).scan() == json::lexer::token_type::value_float)); - CHECK((json::lexer(reinterpret_cast("-1.1"), - 4).scan() == json::lexer::token_type::value_float)); - CHECK((json::lexer(reinterpret_cast("1E10"), - 4).scan() == json::lexer::token_type::value_float)); + CHECK((json::lexer("1.1", 3).scan() == json::lexer::token_type::value_float)); + CHECK((json::lexer("-1.1", 4).scan() == json::lexer::token_type::value_float)); + CHECK((json::lexer("1E10", 4).scan() == json::lexer::token_type::value_float)); } SECTION("whitespace") { // result is end_of_input, because not token is following - CHECK((json::lexer(reinterpret_cast(" "), - 1).scan() == json::lexer::token_type::end_of_input)); - CHECK((json::lexer(reinterpret_cast("\t"), - 1).scan() == json::lexer::token_type::end_of_input)); - CHECK((json::lexer(reinterpret_cast("\n"), - 1).scan() == json::lexer::token_type::end_of_input)); - CHECK((json::lexer(reinterpret_cast("\r"), - 1).scan() == json::lexer::token_type::end_of_input)); - CHECK((json::lexer(reinterpret_cast(" \t\n\r\n\t "), - 7).scan() == json::lexer::token_type::end_of_input)); + CHECK((json::lexer(" ", 1).scan() == json::lexer::token_type::end_of_input)); + CHECK((json::lexer("\t", 1).scan() == json::lexer::token_type::end_of_input)); + CHECK((json::lexer("\n", 1).scan() == json::lexer::token_type::end_of_input)); + CHECK((json::lexer("\r", 1).scan() == json::lexer::token_type::end_of_input)); + CHECK((json::lexer(" \t\n\r\n\t ", 7).scan() == json::lexer::token_type::end_of_input)); } } @@ -141,8 +112,7 @@ TEST_CASE("lexer class") // create string from the ASCII code const auto s = std::string(1, static_cast(c)); // store scan() result - const auto res = json::lexer(reinterpret_cast(s.c_str()), - 1).scan(); + const auto res = json::lexer(s.c_str(), 1).scan(); switch (c) { @@ -188,12 +158,14 @@ TEST_CASE("lexer class") } } + /* NOTE: to_unicode function has been removed SECTION("to_unicode") { // lexer to call to_unicode on - json::lexer dummy_lexer(reinterpret_cast(""), 0); + json::lexer dummy_lexer("", 0); CHECK(dummy_lexer.to_unicode(0x1F4A9) == "💩"); CHECK_THROWS_AS(dummy_lexer.to_unicode(0x200000), json::parse_error); CHECK_THROWS_WITH(dummy_lexer.to_unicode(0x200000), "[json.exception.parse_error.103] parse error: code points above 0x10FFFF are invalid"); } + */ } diff --git a/test/src/unit-class_parser.cpp b/test/src/unit-class_parser.cpp index 2a5c07cf..fe2a8fbe 100644 --- a/test/src/unit-class_parser.cpp +++ b/test/src/unit-class_parser.cpp @@ -91,18 +91,18 @@ TEST_CASE("parser class") // error: tab in string CHECK_THROWS_AS(json::parser("\"\t\"").parse(), json::parse_error); CHECK_THROWS_WITH(json::parser("\"\t\"").parse(), - "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '\"'"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '\"'"); // error: newline in string CHECK_THROWS_AS(json::parser("\"\n\"").parse(), json::parse_error); CHECK_THROWS_AS(json::parser("\"\r\"").parse(), json::parse_error); CHECK_THROWS_WITH(json::parser("\"\n\"").parse(), - "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '\"'"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '\"'"); CHECK_THROWS_WITH(json::parser("\"\r\"").parse(), - "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '\"'"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '\"'"); // error: backspace in string CHECK_THROWS_AS(json::parser("\"\b\"").parse(), json::parse_error); CHECK_THROWS_WITH(json::parser("\"\b\"").parse(), - "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '\"'"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '\"'"); // improve code coverage CHECK_THROWS_AS(json::parser("\uFF01").parse(), json::parse_error); CHECK_THROWS_AS(json::parser("[-4:1,]").parse(), json::parse_error); @@ -306,39 +306,39 @@ TEST_CASE("parser class") CHECK_THROWS_AS(json::parser("+0").parse(), json::parse_error); CHECK_THROWS_WITH(json::parser("01").parse(), - "[json.exception.parse_error.101] parse error at 2: parse error - unexpected '01'"); + "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected '01'"); CHECK_THROWS_WITH(json::parser("-01").parse(), - "[json.exception.parse_error.101] parse error at 3: parse error - unexpected '-01'"); + "[json.exception.parse_error.101] parse error at 3: syntax error - unexpected '-01'"); CHECK_THROWS_WITH(json::parser("--1").parse(), - "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '-'"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '-'"); CHECK_THROWS_WITH(json::parser("1.").parse(), - "[json.exception.parse_error.101] parse error at 2: parse error - unexpected '.'; expected end of input"); + "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected '.'; expected end of input"); CHECK_THROWS_WITH(json::parser("1E").parse(), - "[json.exception.parse_error.101] parse error at 2: parse error - unexpected 'E'; expected end of input"); + "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected 'E'; expected end of input"); CHECK_THROWS_WITH(json::parser("1E-").parse(), - "[json.exception.parse_error.101] parse error at 2: parse error - unexpected 'E'; expected end of input"); + "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected 'E'; expected end of input"); CHECK_THROWS_WITH(json::parser("1.E1").parse(), - "[json.exception.parse_error.101] parse error at 2: parse error - unexpected '.'; expected end of input"); + "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected '.'; expected end of input"); CHECK_THROWS_WITH(json::parser("-1E").parse(), - "[json.exception.parse_error.101] parse error at 3: parse error - unexpected 'E'; expected end of input"); + "[json.exception.parse_error.101] parse error at 3: syntax error - unexpected 'E'; expected end of input"); CHECK_THROWS_WITH(json::parser("-0E#").parse(), - "[json.exception.parse_error.101] parse error at 3: parse error - unexpected 'E'; expected end of input"); + "[json.exception.parse_error.101] parse error at 3: syntax error - unexpected 'E'; expected end of input"); CHECK_THROWS_WITH(json::parser("-0E-#").parse(), - "[json.exception.parse_error.101] parse error at 3: parse error - unexpected 'E'; expected end of input"); + "[json.exception.parse_error.101] parse error at 3: syntax error - unexpected 'E'; expected end of input"); CHECK_THROWS_WITH(json::parser("-0#").parse(), - "[json.exception.parse_error.101] parse error at 3: parse error - unexpected '#'; expected end of input"); + "[json.exception.parse_error.101] parse error at 3: syntax error - unexpected '#'; expected end of input"); CHECK_THROWS_WITH(json::parser("-0.0:").parse(), - "[json.exception.parse_error.101] parse error at 5: parse error - unexpected ':'; expected end of input"); + "[json.exception.parse_error.101] parse error at 5: syntax error - unexpected ':'; expected end of input"); CHECK_THROWS_WITH(json::parser("-0.0Z").parse(), - "[json.exception.parse_error.101] parse error at 5: parse error - unexpected 'Z'; expected end of input"); + "[json.exception.parse_error.101] parse error at 5: syntax error - unexpected 'Z'; expected end of input"); CHECK_THROWS_WITH(json::parser("-0E123:").parse(), - "[json.exception.parse_error.101] parse error at 7: parse error - unexpected ':'; expected end of input"); + "[json.exception.parse_error.101] parse error at 7: syntax error - unexpected ':'; expected end of input"); CHECK_THROWS_WITH(json::parser("-0e0-:").parse(), - "[json.exception.parse_error.101] parse error at 5: parse error - unexpected '-'; expected end of input"); + "[json.exception.parse_error.101] parse error at 5: syntax error - unexpected '-'; expected end of input"); CHECK_THROWS_WITH(json::parser("-0e-:").parse(), - "[json.exception.parse_error.101] parse error at 3: parse error - unexpected 'e'; expected end of input"); + "[json.exception.parse_error.101] parse error at 3: syntax error - unexpected 'e'; expected end of input"); CHECK_THROWS_WITH(json::parser("-0f").parse(), - "[json.exception.parse_error.101] parse error at 3: parse error - unexpected 'f'; expected end of input"); + "[json.exception.parse_error.101] parse error at 3: syntax error - unexpected 'f'; expected end of input"); } } } @@ -361,66 +361,66 @@ TEST_CASE("parser class") CHECK_THROWS_AS(json::parser("1E/").parse(), json::parse_error); CHECK_THROWS_AS(json::parser("1E:").parse(), json::parse_error); CHECK_THROWS_WITH(json::parser("0.").parse(), - "[json.exception.parse_error.101] parse error at 2: parse error - unexpected '.'; expected end of input"); + "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected '.'; expected end of input"); CHECK_THROWS_WITH(json::parser("-").parse(), - "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '-'"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '-'"); CHECK_THROWS_WITH(json::parser("--").parse(), - "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '-'"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '-'"); CHECK_THROWS_WITH(json::parser("-0.").parse(), - "[json.exception.parse_error.101] parse error at 3: parse error - unexpected '.'; expected end of input"); + "[json.exception.parse_error.101] parse error at 3: syntax error - unexpected '.'; expected end of input"); CHECK_THROWS_WITH(json::parser("-.").parse(), - "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '-'"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '-'"); CHECK_THROWS_WITH(json::parser("-:").parse(), - "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '-'"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '-'"); CHECK_THROWS_WITH(json::parser("0.:").parse(), - "[json.exception.parse_error.101] parse error at 2: parse error - unexpected '.'; expected end of input"); + "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected '.'; expected end of input"); CHECK_THROWS_WITH(json::parser("e.").parse(), - "[json.exception.parse_error.101] parse error at 1: parse error - unexpected 'e'"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected 'e'"); CHECK_THROWS_WITH(json::parser("1e.").parse(), - "[json.exception.parse_error.101] parse error at 2: parse error - unexpected 'e'; expected end of input"); + "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected 'e'; expected end of input"); CHECK_THROWS_WITH(json::parser("1e/").parse(), - "[json.exception.parse_error.101] parse error at 2: parse error - unexpected 'e'; expected end of input"); + "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected 'e'; expected end of input"); CHECK_THROWS_WITH(json::parser("1e:").parse(), - "[json.exception.parse_error.101] parse error at 2: parse error - unexpected 'e'; expected end of input"); + "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected 'e'; expected end of input"); CHECK_THROWS_WITH(json::parser("1E.").parse(), - "[json.exception.parse_error.101] parse error at 2: parse error - unexpected 'E'; expected end of input"); + "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected 'E'; expected end of input"); CHECK_THROWS_WITH(json::parser("1E/").parse(), - "[json.exception.parse_error.101] parse error at 2: parse error - unexpected 'E'; expected end of input"); + "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected 'E'; expected end of input"); CHECK_THROWS_WITH(json::parser("1E:").parse(), - "[json.exception.parse_error.101] parse error at 2: parse error - unexpected 'E'; expected end of input"); + "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected 'E'; expected end of input"); // unexpected end of null CHECK_THROWS_AS(json::parser("n").parse(), json::parse_error); CHECK_THROWS_AS(json::parser("nu").parse(), json::parse_error); CHECK_THROWS_AS(json::parser("nul").parse(), json::parse_error); - CHECK_THROWS_WITH(json::parser("n").parse(), "[json.exception.parse_error.101] parse error at 1: parse error - unexpected 'n'"); + CHECK_THROWS_WITH(json::parser("n").parse(), "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected 'n'"); CHECK_THROWS_WITH(json::parser("nu").parse(), - "[json.exception.parse_error.101] parse error at 1: parse error - unexpected 'n'"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected 'n'"); CHECK_THROWS_WITH(json::parser("nul").parse(), - "[json.exception.parse_error.101] parse error at 1: parse error - unexpected 'n'"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected 'n'"); // unexpected end of true CHECK_THROWS_AS(json::parser("t").parse(), json::parse_error); CHECK_THROWS_AS(json::parser("tr").parse(), json::parse_error); CHECK_THROWS_AS(json::parser("tru").parse(), json::parse_error); - CHECK_THROWS_WITH(json::parser("t").parse(), "[json.exception.parse_error.101] parse error at 1: parse error - unexpected 't'"); + CHECK_THROWS_WITH(json::parser("t").parse(), "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected 't'"); CHECK_THROWS_WITH(json::parser("tr").parse(), - "[json.exception.parse_error.101] parse error at 1: parse error - unexpected 't'"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected 't'"); CHECK_THROWS_WITH(json::parser("tru").parse(), - "[json.exception.parse_error.101] parse error at 1: parse error - unexpected 't'"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected 't'"); // unexpected end of false CHECK_THROWS_AS(json::parser("f").parse(), json::parse_error); CHECK_THROWS_AS(json::parser("fa").parse(), json::parse_error); CHECK_THROWS_AS(json::parser("fal").parse(), json::parse_error); CHECK_THROWS_AS(json::parser("fals").parse(), json::parse_error); - CHECK_THROWS_WITH(json::parser("f").parse(), "[json.exception.parse_error.101] parse error at 1: parse error - unexpected 'f'"); + CHECK_THROWS_WITH(json::parser("f").parse(), "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected 'f'"); CHECK_THROWS_WITH(json::parser("fa").parse(), - "[json.exception.parse_error.101] parse error at 1: parse error - unexpected 'f'"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected 'f'"); CHECK_THROWS_WITH(json::parser("fal").parse(), - "[json.exception.parse_error.101] parse error at 1: parse error - unexpected 'f'"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected 'f'"); CHECK_THROWS_WITH(json::parser("fals").parse(), - "[json.exception.parse_error.101] parse error at 1: parse error - unexpected 'f'"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected 'f'"); // missing/unexpected end of array CHECK_THROWS_AS(json::parser("[").parse(), json::parse_error); @@ -429,15 +429,15 @@ TEST_CASE("parser class") CHECK_THROWS_AS(json::parser("[1,]").parse(), json::parse_error); CHECK_THROWS_AS(json::parser("]").parse(), json::parse_error); CHECK_THROWS_WITH(json::parser("[").parse(), - "[json.exception.parse_error.101] parse error at 2: parse error - unexpected end of input"); + "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected end of input"); CHECK_THROWS_WITH(json::parser("[1").parse(), - "[json.exception.parse_error.101] parse error at 3: parse error - unexpected end of input; expected ']'"); + "[json.exception.parse_error.101] parse error at 3: syntax error - unexpected end of input; expected ']'"); CHECK_THROWS_WITH(json::parser("[1,").parse(), - "[json.exception.parse_error.101] parse error at 4: parse error - unexpected end of input"); + "[json.exception.parse_error.101] parse error at 4: syntax error - unexpected end of input"); CHECK_THROWS_WITH(json::parser("[1,]").parse(), - "[json.exception.parse_error.101] parse error at 4: parse error - unexpected ']'"); + "[json.exception.parse_error.101] parse error at 4: syntax error - unexpected ']'"); CHECK_THROWS_WITH(json::parser("]").parse(), - "[json.exception.parse_error.101] parse error at 1: parse error - unexpected ']'"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected ']'"); // missing/unexpected end of object CHECK_THROWS_AS(json::parser("{").parse(), json::parse_error); @@ -447,17 +447,17 @@ TEST_CASE("parser class") CHECK_THROWS_AS(json::parser("{\"foo\":1,}").parse(), json::parse_error); CHECK_THROWS_AS(json::parser("}").parse(), json::parse_error); CHECK_THROWS_WITH(json::parser("{").parse(), - "[json.exception.parse_error.101] parse error at 2: parse error - unexpected end of input; expected string literal"); + "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected end of input; expected string literal"); CHECK_THROWS_WITH(json::parser("{\"foo\"").parse(), - "[json.exception.parse_error.101] parse error at 7: parse error - unexpected end of input; expected ':'"); + "[json.exception.parse_error.101] parse error at 7: syntax error - unexpected end of input; expected ':'"); CHECK_THROWS_WITH(json::parser("{\"foo\":").parse(), - "[json.exception.parse_error.101] parse error at 8: parse error - unexpected end of input"); + "[json.exception.parse_error.101] parse error at 8: syntax error - unexpected end of input"); CHECK_THROWS_WITH(json::parser("{\"foo\":}").parse(), - "[json.exception.parse_error.101] parse error at 8: parse error - unexpected '}'"); + "[json.exception.parse_error.101] parse error at 8: syntax error - unexpected '}'"); CHECK_THROWS_WITH(json::parser("{\"foo\":1,}").parse(), - "[json.exception.parse_error.101] parse error at 10: parse error - unexpected '}'; expected string literal"); + "[json.exception.parse_error.101] parse error at 10: syntax error - unexpected '}'; expected string literal"); CHECK_THROWS_WITH(json::parser("}").parse(), - "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '}'"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '}'"); // missing/unexpected end of string CHECK_THROWS_AS(json::parser("\"").parse(), json::parse_error); @@ -471,25 +471,25 @@ TEST_CASE("parser class") CHECK_THROWS_AS(json::parser("\"\\u01").parse(), json::parse_error); CHECK_THROWS_AS(json::parser("\"\\u012").parse(), json::parse_error); CHECK_THROWS_WITH(json::parser("\"").parse(), - "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '\"'"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '\"'"); CHECK_THROWS_WITH(json::parser("\"\\\"").parse(), - "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '\"'"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '\"'"); CHECK_THROWS_WITH(json::parser("\"\\u\"").parse(), - "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '\"'"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '\"'"); CHECK_THROWS_WITH(json::parser("\"\\u0\"").parse(), - "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '\"'"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '\"'"); CHECK_THROWS_WITH(json::parser("\"\\u01\"").parse(), - "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '\"'"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '\"'"); CHECK_THROWS_WITH(json::parser("\"\\u012\"").parse(), - "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '\"'"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '\"'"); CHECK_THROWS_WITH(json::parser("\"\\u").parse(), - "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '\"'"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '\"'"); CHECK_THROWS_WITH(json::parser("\"\\u0").parse(), - "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '\"'"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '\"'"); CHECK_THROWS_WITH(json::parser("\"\\u01").parse(), - "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '\"'"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '\"'"); CHECK_THROWS_WITH(json::parser("\"\\u012").parse(), - "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '\"'"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '\"'"); // invalid escapes for (int c = 1; c < 128; ++c) @@ -523,7 +523,7 @@ TEST_CASE("parser class") { CHECK_THROWS_AS(json::parser(s.c_str()).parse(), json::parse_error); CHECK_THROWS_WITH(json::parser(s.c_str()).parse(), - "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '\"'"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '\"'"); break; } } @@ -594,13 +594,13 @@ TEST_CASE("parser class") CHECK_THROWS_AS(json::parser(s4.c_str()).parse(), json::parse_error); CHECK_THROWS_WITH(json::parser(s1.c_str()).parse(), - "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '\"'"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '\"'"); CHECK_THROWS_WITH(json::parser(s2.c_str()).parse(), - "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '\"'"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '\"'"); CHECK_THROWS_WITH(json::parser(s3.c_str()).parse(), - "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '\"'"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '\"'"); CHECK_THROWS_WITH(json::parser(s4.c_str()).parse(), - "[json.exception.parse_error.101] parse error at 1: parse error - unexpected '\"'"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '\"'"); } } } @@ -626,11 +626,11 @@ TEST_CASE("parser class") // test case to make sure no comma preceeds the first key CHECK_THROWS_AS(json::parser("{,\"key\": false}").parse(), json::parse_error); CHECK_THROWS_WITH(json::parser("{,\"key\": false}").parse(), - "[json.exception.parse_error.101] parse error at 2: parse error - unexpected ','"); + "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected ','"); // test case to make sure an object is properly closed CHECK_THROWS_AS(json::parser("[{\"key\": false true]").parse(), json::parse_error); CHECK_THROWS_WITH(json::parser("[{\"key\": false true]").parse(), - "[json.exception.parse_error.101] parse error at 19: parse error - unexpected true literal; expected '}'"); + "[json.exception.parse_error.101] parse error at 19: syntax error - unexpected true literal; expected '}'"); // test case to make sure the callback is properly evaluated after reading a key { diff --git a/test/src/unit-deserialization.cpp b/test/src/unit-deserialization.cpp index cacf6687..49a642b1 100644 --- a/test/src/unit-deserialization.cpp +++ b/test/src/unit-deserialization.cpp @@ -92,7 +92,7 @@ TEST_CASE("deserialization") ss2 << "[\"foo\",1,2,3,false,{\"one\":1}"; CHECK_THROWS_AS(json::parse(ss1), json::parse_error); CHECK_THROWS_WITH(json::parse(ss2), - "[json.exception.parse_error.101] parse error at 30: parse error - unexpected end of input; expected ']'"); + "[json.exception.parse_error.101] parse error at 29: syntax error - unexpected end of input; expected ']'"); } SECTION("string") @@ -100,7 +100,7 @@ TEST_CASE("deserialization") json::string_t s = "[\"foo\",1,2,3,false,{\"one\":1}"; CHECK_THROWS_AS(json::parse(s), json::parse_error); CHECK_THROWS_WITH(json::parse(s), - "[json.exception.parse_error.101] parse error at 29: parse error - unexpected end of input; expected ']'"); + "[json.exception.parse_error.101] parse error at 29: syntax error - unexpected end of input; expected ']'"); } SECTION("operator<<") @@ -111,7 +111,7 @@ TEST_CASE("deserialization") json j; CHECK_THROWS_AS(j << ss1, json::parse_error); CHECK_THROWS_WITH(j << ss2, - "[json.exception.parse_error.101] parse error at 30: parse error - unexpected end of input; expected ']'"); + "[json.exception.parse_error.101] parse error at 29: syntax error - unexpected end of input; expected ']'"); } SECTION("operator>>") @@ -122,14 +122,14 @@ TEST_CASE("deserialization") json j; CHECK_THROWS_AS(ss1 >> j, json::parse_error); CHECK_THROWS_WITH(ss2 >> j, - "[json.exception.parse_error.101] parse error at 30: parse error - unexpected end of input; expected ']'"); + "[json.exception.parse_error.101] parse error at 29: syntax error - unexpected end of input; expected ']'"); } SECTION("user-defined string literal") { CHECK_THROWS_AS("[\"foo\",1,2,3,false,{\"one\":1}"_json, json::parse_error); CHECK_THROWS_WITH("[\"foo\",1,2,3,false,{\"one\":1}"_json, - "[json.exception.parse_error.101] parse error at 29: parse error - unexpected end of input; expected ']'"); + "[json.exception.parse_error.101] parse error at 29: syntax error - unexpected end of input; expected ']'"); } } diff --git a/test/src/unit-regression.cpp b/test/src/unit-regression.cpp index f7f4c350..2c09abaa 100644 --- a/test/src/unit-regression.cpp +++ b/test/src/unit-regression.cpp @@ -594,7 +594,7 @@ TEST_CASE("regression tests") // a parse error because of the EOF. CHECK_THROWS_AS(j << ss, json::parse_error); CHECK_THROWS_WITH(j << ss, - "[json.exception.parse_error.101] parse error at 1: parse error - unexpected end of input"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected end of input"); } SECTION("issue #389 - Integer-overflow (OSS-Fuzz issue 267)") @@ -911,6 +911,7 @@ TEST_CASE("regression tests") CHECK(j["bool_vector"].dump() == "[false,true,false,false]"); } + /* NOTE: m_line_buffer is not used any more SECTION("issue #495 - fill_line_buffer incorrectly tests m_stream for eof but not fail or bad bits") { SECTION("setting failbit") @@ -943,6 +944,7 @@ TEST_CASE("regression tests") CHECK_THROWS_WITH(l.fill_line_buffer(), "[json.exception.parse_error.111] parse error: bad input stream"); } } + */ SECTION("issue #504 - assertion error (OSS-Fuzz 856)") { diff --git a/test/src/unit-unicode.cpp b/test/src/unit-unicode.cpp index f7cf0ada..805e4c23 100644 --- a/test/src/unit-unicode.cpp +++ b/test/src/unit-unicode.cpp @@ -36,10 +36,11 @@ using nlohmann::json; TEST_CASE("Unicode", "[hide]") { + /* NOTE: to_unicode is not used any more SECTION("full enumeration of Unicode code points") { // lexer to call to_unicode on - json::lexer dummy_lexer(reinterpret_cast(""), 0); + json::lexer dummy_lexer("", 0); // create an escaped string from a code point const auto codepoint_to_unicode = [](std::size_t cp) @@ -118,6 +119,7 @@ TEST_CASE("Unicode", "[hide]") CHECK(j3 == j4); } } + */ SECTION("read all unicode characters") { From 3d597dda5d9fc758fd3d1aeb0ba19ef03625dfec Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Fri, 24 Mar 2017 19:51:11 +0100 Subject: [PATCH 099/530] :memo: update after #534 --- README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.md b/README.md index 4bf678ef..3d912305 100644 --- a/README.md +++ b/README.md @@ -828,7 +828,7 @@ I deeply appreciate the help of the following people. - [rswanson-ihi](https://github.com/rswanson-ihi) noted a typo in the README. - [Mihai Stan](https://github.com/stanmihai4) fixed a bug in the comparison with `nullptr`s. - [Tushar Maheshwari](https://github.com/tusharpm) added [cotire](https://github.com/sakra/cotire) support to speed up the compilation. -- [TedLyngmo](https://github.com/TedLyngmo) noted a typo in the README and fixed some `-Weffc++` warnings. +- [TedLyngmo](https://github.com/TedLyngmo) noted a typo in the README, removed unnecessary bit arithmetic, and fixed some `-Weffc++` warnings. - [Krzysztof Woś](https://github.com/krzysztofwos) made exceptions more visible. - [ftillier](https://github.com/ftillier) fixed a compiler warning. From 8edb0cd15c40eecbc411cf8ea5d4bb94f713819a Mon Sep 17 00:00:00 2001 From: Vladimir Petrigo Date: Sat, 25 Mar 2017 14:57:58 +0300 Subject: [PATCH 100/530] Add VS17 image to appveyor matrix --- appveyor.yml | 8 +++++--- 1 file changed, 5 insertions(+), 3 deletions(-) diff --git a/appveyor.yml b/appveyor.yml index 3960f4db..fc410e45 100644 --- a/appveyor.yml +++ b/appveyor.yml @@ -1,10 +1,12 @@ version: '{build}' -os: Visual Studio 2015 +os: +- Visual Studio 2015 +- Visual Studio 2017 init: [] install: [] build_script: -- set PATH=C:\Program Files (x86)\MSBuild\14.0\Bin;%PATH% -- cmake . -G "Visual Studio 14 2015" +- IF "%APPVEYOR_BUILD_WORKER_IMAGE%" == "Visual Studio 2015" ( SET GEN="Visual Studio 14 2015") ELSE (SET GEN="Visual Studio 15 2017") +- cmake . -G%GEN% - cmake --build . --config Release test_script: - ctest -C Release -V From 20bcf1015bbd55c7a8c9663ac87929b68d3d5869 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sat, 25 Mar 2017 14:19:05 +0100 Subject: [PATCH 101/530] :memo: cleanup after #536 --- README.md | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/README.md b/README.md index 3d912305..d0fdeb80 100644 --- a/README.md +++ b/README.md @@ -706,6 +706,7 @@ Though it's 2016 already, the support for C++11 is still a bit sparse. Currently - GCC 4.9 - 6.0 (and possibly later) - Clang 3.4 - 3.9 (and possibly later) - Microsoft Visual C++ 2015 / Build Tools 14.0.25123.0 (and possibly later) +- Microsoft Visual C++ 2017 / Build Tools 15.1.548.43366 (and possibly later) I would be happy to learn about other compilers/versions. @@ -744,7 +745,7 @@ The following compilers are currently used in continuous integration at [Travis] | Clang Xcode 8.1 | Darwin Kernel Version 16.1.0 (macOS 10.12.1) | Apple LLVM version 8.0.0 (clang-800.0.42.1) | | Clang Xcode 8.2 | Darwin Kernel Version 16.1.0 (macOS 10.12.1) | Apple LLVM version 8.0.0 (clang-800.0.42.1) | | Visual Studio 14 2015 | Windows Server 2012 R2 (x64) | Microsoft (R) Build Engine version 14.0.25123.0 | - +| Visual Studio 2017 | Windows Server 2016 | Microsoft (R) Build Engine version 15.1.548.43366 | ## License @@ -809,7 +810,7 @@ I deeply appreciate the help of the following people. - [Stefan](https://github.com/5tefan) fixed a minor issue in the documentation. - [Vasil Dimov](https://github.com/vasild) fixed the documentation regarding conversions from `std::multiset`. - [ChristophJud](https://github.com/ChristophJud) overworked the CMake files to ease project inclusion. -- [Vladimir Petrigo](https://github.com/vpetrigo) made a SFINAE hack more readable. +- [Vladimir Petrigo](https://github.com/vpetrigo) made a SFINAE hack more readable and added Visual Studio 17 to the build matrix. - [Denis Andrejew](https://github.com/seeekr) fixed a grammar issue in the README file. - [Pierre-Antoine Lacaze](https://github.com/palacaze) found a subtle bug in the `dump()` function. - [TurpentineDistillery](https://github.com/TurpentineDistillery) pointed to [`std::locale::classic()`](http://en.cppreference.com/w/cpp/locale/locale/classic) to avoid too much locale joggling, found some nice performance improvements in the parser, improved the benchmarking code, and realized locale-independent number parsing and printing. From a58ed3cd178c89e3ae2fafe0af7a799f6ee0c234 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sat, 25 Mar 2017 14:47:23 +0100 Subject: [PATCH 102/530] :lipstick: cleanup --- src/json.hpp | 10 +++++----- src/json.hpp.re2c | 10 +++++----- 2 files changed, 10 insertions(+), 10 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 6ec045aa..cc69d542 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -12486,7 +12486,7 @@ basic_json_parser_74: { result = &result->operator[](static_cast(std::stoi(reference_token))); } - JSON_CATCH(std::invalid_argument&) + JSON_CATCH (std::invalid_argument&) { JSON_THROW(parse_error(109, 0, "array index '" + reference_token + "' is not a number")); } @@ -12585,7 +12585,7 @@ basic_json_parser_74: { ptr = &ptr->operator[](static_cast(std::stoi(reference_token))); } - JSON_CATCH(std::invalid_argument&) + JSON_CATCH (std::invalid_argument&) { JSON_THROW(parse_error(109, 0, "array index '" + reference_token + "' is not a number")); } @@ -12643,7 +12643,7 @@ basic_json_parser_74: { ptr = &ptr->at(static_cast(std::stoi(reference_token))); } - JSON_CATCH(std::invalid_argument&) + JSON_CATCH (std::invalid_argument&) { JSON_THROW(parse_error(109, 0, "array index '" + reference_token + "' is not a number")); } @@ -12707,7 +12707,7 @@ basic_json_parser_74: { ptr = &ptr->operator[](static_cast(std::stoi(reference_token))); } - JSON_CATCH(std::invalid_argument&) + JSON_CATCH (std::invalid_argument&) { JSON_THROW(parse_error(109, 0, "array index '" + reference_token + "' is not a number")); } @@ -12764,7 +12764,7 @@ basic_json_parser_74: { ptr = &ptr->at(static_cast(std::stoi(reference_token))); } - JSON_CATCH(std::invalid_argument&) + JSON_CATCH (std::invalid_argument&) { JSON_THROW(parse_error(109, 0, "array index '" + reference_token + "' is not a number")); } diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 7d21f76b..1480a5f5 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -11519,7 +11519,7 @@ class basic_json { result = &result->operator[](static_cast(std::stoi(reference_token))); } - JSON_CATCH(std::invalid_argument&) + JSON_CATCH (std::invalid_argument&) { JSON_THROW(parse_error(109, 0, "array index '" + reference_token + "' is not a number")); } @@ -11618,7 +11618,7 @@ class basic_json { ptr = &ptr->operator[](static_cast(std::stoi(reference_token))); } - JSON_CATCH(std::invalid_argument&) + JSON_CATCH (std::invalid_argument&) { JSON_THROW(parse_error(109, 0, "array index '" + reference_token + "' is not a number")); } @@ -11676,7 +11676,7 @@ class basic_json { ptr = &ptr->at(static_cast(std::stoi(reference_token))); } - JSON_CATCH(std::invalid_argument&) + JSON_CATCH (std::invalid_argument&) { JSON_THROW(parse_error(109, 0, "array index '" + reference_token + "' is not a number")); } @@ -11740,7 +11740,7 @@ class basic_json { ptr = &ptr->operator[](static_cast(std::stoi(reference_token))); } - JSON_CATCH(std::invalid_argument&) + JSON_CATCH (std::invalid_argument&) { JSON_THROW(parse_error(109, 0, "array index '" + reference_token + "' is not a number")); } @@ -11797,7 +11797,7 @@ class basic_json { ptr = &ptr->at(static_cast(std::stoi(reference_token))); } - JSON_CATCH(std::invalid_argument&) + JSON_CATCH (std::invalid_argument&) { JSON_THROW(parse_error(109, 0, "array index '" + reference_token + "' is not a number")); } From cc36c65a8907f67519c2024b4ed29ea8ad178798 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sat, 25 Mar 2017 16:22:52 +0100 Subject: [PATCH 103/530] :ambulance: made exceptions nothrow-copy-constructible #531 To have nothrow-copy-constructible exceptions, we inherit from std::runtime_error which can cope with arbitrary-length error messages. Intermediate strings are built with static functions and then passed to the actual constructor. --- README.md | 2 +- src/json.hpp | 341 +++++++++++++++++++++---------------- src/json.hpp.re2c | 341 +++++++++++++++++++++---------------- test/src/unit-noexcept.cpp | 8 + 4 files changed, 391 insertions(+), 301 deletions(-) diff --git a/README.md b/README.md index d0fdeb80..e23e3c94 100644 --- a/README.md +++ b/README.md @@ -894,7 +894,7 @@ $ make json_unit -Ctest $ ./test/json_unit "*" =============================================================================== -All tests passed (11202597 assertions in 47 test cases) +All tests passed (11203022 assertions in 48 test cases) ``` Alternatively, you can use [CMake](https://cmake.org) and run diff --git a/src/json.hpp b/src/json.hpp index cc69d542..bdada276 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -116,29 +116,34 @@ namespace detail Extension of std::exception objects with a member @a id for exception ids. +@note To have nothrow-copy-constructible exceptions, we inherit from + std::runtime_error which can cope with arbitrary-length error messages. + Intermediate strings are built with static functions and then passed to + the actual constructor. + @since version 3.0.0 */ -class exception : public std::exception +class exception : public std::runtime_error { public: - /// create exception with id an explanatory string - exception(int id_, const std::string& ename, const std::string& what_arg_) - : id(id_), - what_arg("[json.exception." + ename + "." + std::to_string(id_) + "] " + what_arg_) - {} - /// returns the explanatory string virtual const char* what() const noexcept override { - return what_arg.c_str(); + return std::runtime_error::what(); } /// the id of the exception const int id; - private: - /// the explanatory string - const std::string what_arg; + protected: + exception(int id_, const char* what_arg) + : std::runtime_error(what_arg), id(id_) + {} + + static std::string name(const std::string& ename, int id_) + { + return "[json.exception." + ename + "." + std::to_string(id_) + "] "; + } }; /*! @@ -184,14 +189,16 @@ class parse_error : public exception @param[in] id_ the id of the exception @param[in] byte_ the byte index where the error occured (or 0 if the position cannot be determined) - @param[in] what_arg_ the explanatory string + @param[in] what_arg the explanatory string + @return parse_error object */ - parse_error(int id_, size_t byte_, const std::string& what_arg_) - : exception(id_, "parse_error", "parse error" + - (byte_ != 0 ? (" at " + std::to_string(byte_)) : "") + - ": " + what_arg_), - byte(byte_) - {} + static parse_error create(int id_, size_t byte_, const std::string& what_arg) + { + std::string w = exception::name("parse_error", id_) + "parse error" + + (byte_ != 0 ? (" at " + std::to_string(byte_)) : "") + + ": " + what_arg; + return parse_error(id_, byte_, w.c_str()); + } /*! @brief byte index of the parse error @@ -204,6 +211,12 @@ class parse_error : public exception MessagePack). */ const size_t byte; + + private: + parse_error(int id_, size_t byte_, const char* what_arg) + : exception(id_, what_arg), + byte(byte_) + {} }; /*! @@ -233,8 +246,15 @@ json.exception.invalid_iterator.214 | cannot get value | Cannot get value for it class invalid_iterator : public exception { public: - invalid_iterator(int id_, const std::string& what_arg_) - : exception(id_, "invalid_iterator", what_arg_) + static invalid_iterator create(int id_, const std::string& what_arg) + { + std::string w = exception::name("invalid_iterator", id_) + what_arg; + return invalid_iterator(id_, w.c_str()); + } + + private: + invalid_iterator(int id_, const char* what_arg) + : exception(id_, what_arg) {} }; @@ -265,8 +285,15 @@ json.exception.type_error.315 | values in object must be primitive | The @ref un class type_error : public exception { public: - type_error(int id_, const std::string& what_arg_) - : exception(id_, "type_error", what_arg_) + static type_error create(int id_, const std::string& what_arg) + { + std::string w = exception::name("type_error", id_) + what_arg; + return type_error(id_, w.c_str()); + } + + private: + type_error(int id_, const char* what_arg) + : exception(id_, what_arg) {} }; @@ -289,8 +316,15 @@ json.exception.out_of_range.406 | number overflow parsing '10E1000' | A parsed n class out_of_range : public exception { public: - out_of_range(int id_, const std::string& what_arg_) - : exception(id_, "out_of_range", what_arg_) + static out_of_range create(int id_, const std::string& what_arg) + { + std::string w = exception::name("out_of_range", id_) + what_arg; + return out_of_range(id_, w.c_str()); + } + + private: + out_of_range(int id_, const char* what_arg) + : exception(id_, what_arg) {} }; @@ -308,8 +342,15 @@ json.exception.other_error.501 | unsuccessful: {"op":"test","path":"/baz", "valu class other_error : public exception { public: - other_error(int id_, const std::string& what_arg_) - : exception(id_, "other_error", what_arg_) + static other_error create(int id_, const std::string& what_arg) + { + std::string w = exception::name("other_error", id_) + what_arg; + return other_error(id_, w.c_str()); + } + + private: + other_error(int id_, const char* what_arg) + : exception(id_, what_arg) {} }; @@ -841,7 +882,7 @@ void get_arithmetic_value(const BasicJsonType& j, ArithmeticType& val) } default: { - JSON_THROW(type_error(302, "type must be number, but is " + j.type_name())); + JSON_THROW(type_error::create(302, "type must be number, but is " + j.type_name())); } } } @@ -851,7 +892,7 @@ void from_json(const BasicJsonType& j, typename BasicJsonType::boolean_t& b) { if (not j.is_boolean()) { - JSON_THROW(type_error(302, "type must be boolean, but is " + j.type_name())); + JSON_THROW(type_error::create(302, "type must be boolean, but is " + j.type_name())); } b = *j.template get_ptr(); } @@ -861,7 +902,7 @@ void from_json(const BasicJsonType& j, typename BasicJsonType::string_t& s) { if (not j.is_string()) { - JSON_THROW(type_error(302, "type must be string, but is " + j.type_name())); + JSON_THROW(type_error::create(302, "type must be string, but is " + j.type_name())); } s = *j.template get_ptr(); } @@ -898,7 +939,7 @@ void from_json(const BasicJsonType& j, typename BasicJsonType::array_t& arr) { if (not j.is_array()) { - JSON_THROW(type_error(302, "type must be array, but is " + j.type_name())); + JSON_THROW(type_error::create(302, "type must be array, but is " + j.type_name())); } arr = *j.template get_ptr(); } @@ -910,7 +951,7 @@ void from_json(const BasicJsonType& j, std::forward_list& l) { if (not j.is_array()) { - JSON_THROW(type_error(302, "type must be array, but is " + j.type_name())); + JSON_THROW(type_error::create(302, "type must be array, but is " + j.type_name())); } for (auto it = j.rbegin(), end = j.rend(); it != end; ++it) @@ -961,7 +1002,7 @@ void from_json(const BasicJsonType& j, CompatibleArrayType& arr) { if (not j.is_array()) { - JSON_THROW(type_error(302, "type must be array, but is " + j.type_name())); + JSON_THROW(type_error::create(302, "type must be array, but is " + j.type_name())); } from_json_array_impl(j, arr, priority_tag<1> {}); @@ -973,7 +1014,7 @@ void from_json(const BasicJsonType& j, CompatibleObjectType& obj) { if (not j.is_object()) { - JSON_THROW(type_error(302, "type must be object, but is " + j.type_name())); + JSON_THROW(type_error::create(302, "type must be object, but is " + j.type_name())); } auto inner_object = j.template get_ptr(); @@ -1023,7 +1064,7 @@ void from_json(const BasicJsonType& j, ArithmeticType& val) } default: { - JSON_THROW(type_error(302, "type must be number, but is " + j.type_name())); + JSON_THROW(type_error::create(302, "type must be number, but is " + j.type_name())); } } } @@ -1970,7 +2011,7 @@ class basic_json { if (t == value_t::null) { - JSON_THROW(other_error(500, "961c151d2e87f2686a955a9be24d316f1362bf21 2.1.1")); // LCOV_EXCL_LINE + JSON_THROW(other_error::create(500, "961c151d2e87f2686a955a9be24d316f1362bf21 2.1.1")); // LCOV_EXCL_LINE } break; } @@ -2325,7 +2366,7 @@ class basic_json // if object is wanted but impossible, throw an exception if (manual_type == value_t::object and not is_an_object) { - JSON_THROW(type_error(301, "cannot create object from initializer list")); + JSON_THROW(type_error::create(301, "cannot create object from initializer list")); } } @@ -2509,7 +2550,7 @@ class basic_json // make sure iterator fits the current value if (first.m_object != last.m_object) { - JSON_THROW(invalid_iterator(201, "iterators are not compatible")); + JSON_THROW(invalid_iterator::create(201, "iterators are not compatible")); } // copy type from first iterator @@ -2526,7 +2567,7 @@ class basic_json { if (not first.m_it.primitive_iterator.is_begin() or not last.m_it.primitive_iterator.is_end()) { - JSON_THROW(invalid_iterator(204, "iterators out of range")); + JSON_THROW(invalid_iterator::create(204, "iterators out of range")); } break; } @@ -2585,8 +2626,8 @@ class basic_json default: { - JSON_THROW(invalid_iterator(206, "cannot construct with iterators from " + - first.m_object->type_name())); + JSON_THROW(invalid_iterator::create(206, "cannot construct with iterators from " + + first.m_object->type_name())); } } @@ -3224,7 +3265,7 @@ class basic_json return m_value.boolean; } - JSON_THROW(type_error(302, "type must be boolean, but is " + type_name())); + JSON_THROW(type_error::create(302, "type must be boolean, but is " + type_name())); } /// get a pointer to the value (object) @@ -3336,7 +3377,7 @@ class basic_json return *ptr; } - JSON_THROW(type_error(303, "incompatible ReferenceType for get_ref, actual type is " + obj.type_name())); + JSON_THROW(type_error::create(303, "incompatible ReferenceType for get_ref, actual type is " + obj.type_name())); } public: @@ -3738,12 +3779,12 @@ class basic_json JSON_CATCH (std::out_of_range&) { // create better exception explanation - JSON_THROW(out_of_range(401, "array index " + std::to_string(idx) + " is out of range")); + JSON_THROW(out_of_range::create(401, "array index " + std::to_string(idx) + " is out of range")); } } else { - JSON_THROW(type_error(304, "cannot use at() with " + type_name())); + JSON_THROW(type_error::create(304, "cannot use at() with " + type_name())); } } @@ -3785,12 +3826,12 @@ class basic_json JSON_CATCH (std::out_of_range&) { // create better exception explanation - JSON_THROW(out_of_range(401, "array index " + std::to_string(idx) + " is out of range")); + JSON_THROW(out_of_range::create(401, "array index " + std::to_string(idx) + " is out of range")); } } else { - JSON_THROW(type_error(304, "cannot use at() with " + type_name())); + JSON_THROW(type_error::create(304, "cannot use at() with " + type_name())); } } @@ -3836,12 +3877,12 @@ class basic_json JSON_CATCH (std::out_of_range&) { // create better exception explanation - JSON_THROW(out_of_range(403, "key '" + key + "' not found")); + JSON_THROW(out_of_range::create(403, "key '" + key + "' not found")); } } else { - JSON_THROW(type_error(304, "cannot use at() with " + type_name())); + JSON_THROW(type_error::create(304, "cannot use at() with " + type_name())); } } @@ -3887,12 +3928,12 @@ class basic_json JSON_CATCH (std::out_of_range&) { // create better exception explanation - JSON_THROW(out_of_range(403, "key '" + key + "' not found")); + JSON_THROW(out_of_range::create(403, "key '" + key + "' not found")); } } else { - JSON_THROW(type_error(304, "cannot use at() with " + type_name())); + JSON_THROW(type_error::create(304, "cannot use at() with " + type_name())); } } @@ -3945,7 +3986,7 @@ class basic_json return m_value.array->operator[](idx); } - JSON_THROW(type_error(305, "cannot use operator[] with " + type_name())); + JSON_THROW(type_error::create(305, "cannot use operator[] with " + type_name())); } /*! @@ -3975,7 +4016,7 @@ class basic_json return m_value.array->operator[](idx); } - JSON_THROW(type_error(305, "cannot use operator[] with " + type_name())); + JSON_THROW(type_error::create(305, "cannot use operator[] with " + type_name())); } /*! @@ -4021,7 +4062,7 @@ class basic_json return m_value.object->operator[](key); } - JSON_THROW(type_error(305, "cannot use operator[] with " + type_name())); + JSON_THROW(type_error::create(305, "cannot use operator[] with " + type_name())); } /*! @@ -4063,7 +4104,7 @@ class basic_json return m_value.object->find(key)->second; } - JSON_THROW(type_error(305, "cannot use operator[] with " + type_name())); + JSON_THROW(type_error::create(305, "cannot use operator[] with " + type_name())); } /*! @@ -4178,7 +4219,7 @@ class basic_json return m_value.object->operator[](key); } - JSON_THROW(type_error(305, "cannot use operator[] with " + type_name())); + JSON_THROW(type_error::create(305, "cannot use operator[] with " + type_name())); } /*! @@ -4221,7 +4262,7 @@ class basic_json return m_value.object->find(key)->second; } - JSON_THROW(type_error(305, "cannot use operator[] with " + type_name())); + JSON_THROW(type_error::create(305, "cannot use operator[] with " + type_name())); } /*! @@ -4290,7 +4331,7 @@ class basic_json } else { - JSON_THROW(type_error(306, "cannot use value() with " + type_name())); + JSON_THROW(type_error::create(306, "cannot use value() with " + type_name())); } } @@ -4362,7 +4403,7 @@ class basic_json } } - JSON_THROW(type_error(306, "cannot use value() with " + type_name())); + JSON_THROW(type_error::create(306, "cannot use value() with " + type_name())); } /*! @@ -4515,7 +4556,7 @@ class basic_json // make sure iterator fits the current value if (this != pos.m_object) { - JSON_THROW(invalid_iterator(202, "iterator does not fit current value")); + JSON_THROW(invalid_iterator::create(202, "iterator does not fit current value")); } IteratorType result = end(); @@ -4530,7 +4571,7 @@ class basic_json { if (not pos.m_it.primitive_iterator.is_begin()) { - JSON_THROW(invalid_iterator(205, "iterator out of range")); + JSON_THROW(invalid_iterator::create(205, "iterator out of range")); } if (is_string()) @@ -4560,7 +4601,7 @@ class basic_json default: { - JSON_THROW(type_error(307, "cannot use erase() with " + type_name())); + JSON_THROW(type_error::create(307, "cannot use erase() with " + type_name())); } } @@ -4622,7 +4663,7 @@ class basic_json // make sure iterator fits the current value if (this != first.m_object or this != last.m_object) { - JSON_THROW(invalid_iterator(203, "iterators do not fit current value")); + JSON_THROW(invalid_iterator::create(203, "iterators do not fit current value")); } IteratorType result = end(); @@ -4637,7 +4678,7 @@ class basic_json { if (not first.m_it.primitive_iterator.is_begin() or not last.m_it.primitive_iterator.is_end()) { - JSON_THROW(invalid_iterator(204, "iterators out of range")); + JSON_THROW(invalid_iterator::create(204, "iterators out of range")); } if (is_string()) @@ -4669,7 +4710,7 @@ class basic_json default: { - JSON_THROW(type_error(307, "cannot use erase() with " + type_name())); + JSON_THROW(type_error::create(307, "cannot use erase() with " + type_name())); } } @@ -4713,7 +4754,7 @@ class basic_json return m_value.object->erase(key); } - JSON_THROW(type_error(307, "cannot use erase() with " + type_name())); + JSON_THROW(type_error::create(307, "cannot use erase() with " + type_name())); } /*! @@ -4747,14 +4788,14 @@ class basic_json { if (idx >= size()) { - JSON_THROW(out_of_range(401, "array index " + std::to_string(idx) + " is out of range")); + JSON_THROW(out_of_range::create(401, "array index " + std::to_string(idx) + " is out of range")); } m_value.array->erase(m_value.array->begin() + static_cast(idx)); } else { - JSON_THROW(type_error(307, "cannot use erase() with " + type_name())); + JSON_THROW(type_error::create(307, "cannot use erase() with " + type_name())); } } @@ -5472,7 +5513,7 @@ class basic_json // push_back only works for null objects or arrays if (not(is_null() or is_array())) { - JSON_THROW(type_error(308, "cannot use push_back() with " + type_name())); + JSON_THROW(type_error::create(308, "cannot use push_back() with " + type_name())); } // transform null object into an array @@ -5508,7 +5549,7 @@ class basic_json // push_back only works for null objects or arrays if (not(is_null() or is_array())) { - JSON_THROW(type_error(308, "cannot use push_back() with " + type_name())); + JSON_THROW(type_error::create(308, "cannot use push_back() with " + type_name())); } // transform null object into an array @@ -5558,7 +5599,7 @@ class basic_json // push_back only works for null objects or objects if (not(is_null() or is_object())) { - JSON_THROW(type_error(308, "cannot use push_back() with " + type_name())); + JSON_THROW(type_error::create(308, "cannot use push_back() with " + type_name())); } // transform null object into an object @@ -5658,7 +5699,7 @@ class basic_json // emplace_back only works for null objects or arrays if (not(is_null() or is_array())) { - JSON_THROW(type_error(311, "cannot use emplace_back() with " + type_name())); + JSON_THROW(type_error::create(311, "cannot use emplace_back() with " + type_name())); } // transform null object into an array @@ -5706,7 +5747,7 @@ class basic_json // emplace only works for null objects or arrays if (not(is_null() or is_object())) { - JSON_THROW(type_error(311, "cannot use emplace() with " + type_name())); + JSON_THROW(type_error::create(311, "cannot use emplace() with " + type_name())); } // transform null object into an object @@ -5757,7 +5798,7 @@ class basic_json // check if iterator pos fits to this JSON value if (pos.m_object != this) { - JSON_THROW(invalid_iterator(202, "iterator does not fit current value")); + JSON_THROW(invalid_iterator::create(202, "iterator does not fit current value")); } // insert to array and return iterator @@ -5766,7 +5807,7 @@ class basic_json return result; } - JSON_THROW(type_error(309, "cannot use insert() with " + type_name())); + JSON_THROW(type_error::create(309, "cannot use insert() with " + type_name())); } /*! @@ -5810,7 +5851,7 @@ class basic_json // check if iterator pos fits to this JSON value if (pos.m_object != this) { - JSON_THROW(invalid_iterator(202, "iterator does not fit current value")); + JSON_THROW(invalid_iterator::create(202, "iterator does not fit current value")); } // insert to array and return iterator @@ -5819,7 +5860,7 @@ class basic_json return result; } - JSON_THROW(type_error(309, "cannot use insert() with " + type_name())); + JSON_THROW(type_error::create(309, "cannot use insert() with " + type_name())); } /*! @@ -5857,24 +5898,24 @@ class basic_json // insert only works for arrays if (not is_array()) { - JSON_THROW(type_error(309, "cannot use insert() with " + type_name())); + JSON_THROW(type_error::create(309, "cannot use insert() with " + type_name())); } // check if iterator pos fits to this JSON value if (pos.m_object != this) { - JSON_THROW(invalid_iterator(202, "iterator does not fit current value")); + JSON_THROW(invalid_iterator::create(202, "iterator does not fit current value")); } // check if range iterators belong to the same JSON object if (first.m_object != last.m_object) { - JSON_THROW(invalid_iterator(210, "iterators do not fit")); + JSON_THROW(invalid_iterator::create(210, "iterators do not fit")); } if (first.m_object == this or last.m_object == this) { - JSON_THROW(invalid_iterator(211, "passed iterators may not belong to container")); + JSON_THROW(invalid_iterator::create(211, "passed iterators may not belong to container")); } // insert to array and return iterator @@ -5915,13 +5956,13 @@ class basic_json // insert only works for arrays if (not is_array()) { - JSON_THROW(type_error(309, "cannot use insert() with " + type_name())); + JSON_THROW(type_error::create(309, "cannot use insert() with " + type_name())); } // check if iterator pos fits to this JSON value if (pos.m_object != this) { - JSON_THROW(invalid_iterator(202, "iterator does not fit current value")); + JSON_THROW(invalid_iterator::create(202, "iterator does not fit current value")); } // insert to array and return iterator @@ -5988,7 +6029,7 @@ class basic_json } else { - JSON_THROW(type_error(310, "cannot use swap() with " + type_name())); + JSON_THROW(type_error::create(310, "cannot use swap() with " + type_name())); } } @@ -6021,7 +6062,7 @@ class basic_json } else { - JSON_THROW(type_error(310, "cannot use swap() with " + type_name())); + JSON_THROW(type_error::create(310, "cannot use swap() with " + type_name())); } } @@ -6054,7 +6095,7 @@ class basic_json } else { - JSON_THROW(type_error(310, "cannot use swap() with " + type_name())); + JSON_THROW(type_error::create(310, "cannot use swap() with " + type_name())); } } @@ -7999,19 +8040,19 @@ class basic_json // simple case: requested length is greater than the vector's length if (len > size or offset > size) { - JSON_THROW(parse_error(110, offset + 1, "cannot read " + std::to_string(len) + " bytes from vector")); + JSON_THROW(parse_error::create(110, offset + 1, "cannot read " + std::to_string(len) + " bytes from vector")); } // second case: adding offset would result in overflow if ((size > ((std::numeric_limits::max)() - offset))) { - JSON_THROW(parse_error(110, offset + 1, "cannot read " + std::to_string(len) + " bytes from vector")); + JSON_THROW(parse_error::create(110, offset + 1, "cannot read " + std::to_string(len) + " bytes from vector")); } // last case: reading past the end of the vector if (len + offset > size) { - JSON_THROW(parse_error(110, offset + 1, "cannot read " + std::to_string(len) + " bytes from vector")); + JSON_THROW(parse_error::create(110, offset + 1, "cannot read " + std::to_string(len) + " bytes from vector")); } } @@ -8043,7 +8084,7 @@ class basic_json std::stringstream ss; ss << std::hex << static_cast(v[idx]); - JSON_THROW(parse_error(113, idx + 1, "expected a MessagePack string; last byte: 0x" + ss.str())); + JSON_THROW(parse_error::create(113, idx + 1, "expected a MessagePack string; last byte: 0x" + ss.str())); } /*! @@ -8073,7 +8114,7 @@ class basic_json std::stringstream ss; ss << std::hex << static_cast(v[idx]); - JSON_THROW(parse_error(113, idx + 1, "expected a CBOR string; last byte: 0x" + ss.str())); + JSON_THROW(parse_error::create(113, idx + 1, "expected a CBOR string; last byte: 0x" + ss.str())); } /*! @@ -8316,7 +8357,7 @@ class basic_json { std::stringstream ss; ss << std::hex << static_cast(v[current_idx]); - JSON_THROW(parse_error(112, current_idx + 1, "error reading MessagePack; last byte: 0x" + ss.str())); + JSON_THROW(parse_error::create(112, current_idx + 1, "error reading MessagePack; last byte: 0x" + ss.str())); } } } @@ -8816,7 +8857,7 @@ class basic_json { std::stringstream ss; ss << std::hex << static_cast(v[current_idx]); - JSON_THROW(parse_error(112, current_idx + 1, "error reading CBOR; last byte: 0x" + ss.str())); + JSON_THROW(parse_error::create(112, current_idx + 1, "error reading CBOR; last byte: 0x" + ss.str())); } } } @@ -9697,7 +9738,7 @@ class basic_json case basic_json::value_t::null: { - JSON_THROW(invalid_iterator(214, "cannot get value")); + JSON_THROW(invalid_iterator::create(214, "cannot get value")); } default: @@ -9707,7 +9748,7 @@ class basic_json return *m_object; } - JSON_THROW(invalid_iterator(214, "cannot get value")); + JSON_THROW(invalid_iterator::create(214, "cannot get value")); } } } @@ -9741,7 +9782,7 @@ class basic_json return m_object; } - JSON_THROW(invalid_iterator(214, "cannot get value")); + JSON_THROW(invalid_iterator::create(214, "cannot get value")); } } } @@ -9841,7 +9882,7 @@ class basic_json // if objects are not the same, the comparison is undefined if (m_object != other.m_object) { - JSON_THROW(invalid_iterator(212, "cannot compare iterators of different containers")); + JSON_THROW(invalid_iterator::create(212, "cannot compare iterators of different containers")); } assert(m_object != nullptr); @@ -9883,7 +9924,7 @@ class basic_json // if objects are not the same, the comparison is undefined if (m_object != other.m_object) { - JSON_THROW(invalid_iterator(212, "cannot compare iterators of different containers")); + JSON_THROW(invalid_iterator::create(212, "cannot compare iterators of different containers")); } assert(m_object != nullptr); @@ -9892,7 +9933,7 @@ class basic_json { case basic_json::value_t::object: { - JSON_THROW(invalid_iterator(213, "cannot compare order of object iterators")); + JSON_THROW(invalid_iterator::create(213, "cannot compare order of object iterators")); } case basic_json::value_t::array: @@ -9946,7 +9987,7 @@ class basic_json { case basic_json::value_t::object: { - JSON_THROW(invalid_iterator(209, "cannot use offsets with object iterators")); + JSON_THROW(invalid_iterator::create(209, "cannot use offsets with object iterators")); } case basic_json::value_t::array: @@ -10008,7 +10049,7 @@ class basic_json { case basic_json::value_t::object: { - JSON_THROW(invalid_iterator(209, "cannot use offsets with object iterators")); + JSON_THROW(invalid_iterator::create(209, "cannot use offsets with object iterators")); } case basic_json::value_t::array: @@ -10035,7 +10076,7 @@ class basic_json { case basic_json::value_t::object: { - JSON_THROW(invalid_iterator(208, "cannot use operator[] for object iterators")); + JSON_THROW(invalid_iterator::create(208, "cannot use operator[] for object iterators")); } case basic_json::value_t::array: @@ -10045,7 +10086,7 @@ class basic_json case basic_json::value_t::null: { - JSON_THROW(invalid_iterator(214, "cannot get value")); + JSON_THROW(invalid_iterator::create(214, "cannot get value")); } default: @@ -10055,7 +10096,7 @@ class basic_json return *m_object; } - JSON_THROW(invalid_iterator(214, "cannot get value")); + JSON_THROW(invalid_iterator::create(214, "cannot get value")); } } } @@ -10073,7 +10114,7 @@ class basic_json return m_it.object_iterator->first; } - JSON_THROW(invalid_iterator(207, "cannot use key() for non-object iterators")); + JSON_THROW(invalid_iterator::create(207, "cannot use key() for non-object iterators")); } /*! @@ -10263,7 +10304,7 @@ class basic_json // immediately abort if stream is erroneous if (s.fail()) { - JSON_THROW(parse_error(111, 0, "bad input stream")); + JSON_THROW(parse_error::create(111, 0, "bad input stream")); } // fill buffer @@ -10330,7 +10371,7 @@ class basic_json } else { - JSON_THROW(parse_error(102, get_position(), "missing or wrong low surrogate")); + JSON_THROW(parse_error::create(102, get_position(), "missing or wrong low surrogate")); } } @@ -10364,7 +10405,7 @@ class basic_json } else { - JSON_THROW(parse_error(103, get_position(), "code points above 0x10FFFF are invalid")); + JSON_THROW(parse_error::create(103, get_position(), "code points above 0x10FFFF are invalid")); } return result; @@ -11560,7 +11601,7 @@ basic_json_parser_74: // check if stream is still good if (m_stream->fail()) { - JSON_THROW(parse_error(111, 0, "bad input stream")); + JSON_THROW(parse_error::create(111, 0, "bad input stream")); } std::getline(*m_stream, m_line_buffer_tmp, '\n'); @@ -11729,7 +11770,7 @@ basic_json_parser_74: // make sure there is a subsequent unicode if ((i + 6 >= m_limit) or * (i + 5) != '\\' or * (i + 6) != 'u') { - JSON_THROW(parse_error(102, get_position(), "missing low surrogate")); + JSON_THROW(parse_error::create(102, get_position(), "missing low surrogate")); } // get code yyyy from uxxxx\uyyyy @@ -11742,7 +11783,7 @@ basic_json_parser_74: else if (codepoint >= 0xDC00 and codepoint <= 0xDFFF) { // we found a lone low surrogate - JSON_THROW(parse_error(102, get_position(), "missing high surrogate")); + JSON_THROW(parse_error::create(102, get_position(), "missing high surrogate")); } else { @@ -11986,7 +12027,7 @@ basic_json_parser_74: // throw in case of infinity or NAN if (not std::isfinite(result.m_value.number_float)) { - JSON_THROW(out_of_range(406, "number overflow parsing '" + get_token_string() + "'")); + JSON_THROW(out_of_range::create(406, "number overflow parsing '" + get_token_string() + "'")); } return true; @@ -12299,7 +12340,7 @@ basic_json_parser_74: "'") : lexer::token_type_name(last_token)); error_msg += "; expected " + lexer::token_type_name(t); - JSON_THROW(parse_error(101, m_lexer.get_position(), error_msg)); + JSON_THROW(parse_error::create(101, m_lexer.get_position(), error_msg)); } } @@ -12314,7 +12355,7 @@ basic_json_parser_74: error_msg += (last_token == lexer::token_type::parse_error ? ("'" + m_lexer.get_token_string() + "'") : lexer::token_type_name(last_token)); - JSON_THROW(parse_error(101, m_lexer.get_position(), error_msg)); + JSON_THROW(parse_error::create(101, m_lexer.get_position(), error_msg)); } } @@ -12413,7 +12454,7 @@ basic_json_parser_74: { if (is_root()) { - JSON_THROW(out_of_range(405, "JSON pointer has no parent")); + JSON_THROW(out_of_range::create(405, "JSON pointer has no parent")); } auto last = reference_tokens.back(); @@ -12431,7 +12472,7 @@ basic_json_parser_74: { if (is_root()) { - JSON_THROW(out_of_range(405, "JSON pointer has no parent")); + JSON_THROW(out_of_range::create(405, "JSON pointer has no parent")); } json_pointer result = *this; @@ -12488,7 +12529,7 @@ basic_json_parser_74: } JSON_CATCH (std::invalid_argument&) { - JSON_THROW(parse_error(109, 0, "array index '" + reference_token + "' is not a number")); + JSON_THROW(parse_error::create(109, 0, "array index '" + reference_token + "' is not a number")); } break; } @@ -12502,7 +12543,7 @@ basic_json_parser_74: */ default: { - JSON_THROW(type_error(313, "invalid value to unflatten")); + JSON_THROW(type_error::create(313, "invalid value to unflatten")); } } } @@ -12570,7 +12611,7 @@ basic_json_parser_74: // error condition (cf. RFC 6901, Sect. 4) if (reference_token.size() > 1 and reference_token[0] == '0') { - JSON_THROW(parse_error(106, 0, "array index '" + reference_token + "' must not begin with '0'")); + JSON_THROW(parse_error::create(106, 0, "array index '" + reference_token + "' must not begin with '0'")); } if (reference_token == "-") @@ -12587,7 +12628,7 @@ basic_json_parser_74: } JSON_CATCH (std::invalid_argument&) { - JSON_THROW(parse_error(109, 0, "array index '" + reference_token + "' is not a number")); + JSON_THROW(parse_error::create(109, 0, "array index '" + reference_token + "' is not a number")); } } break; @@ -12595,7 +12636,7 @@ basic_json_parser_74: default: { - JSON_THROW(out_of_range(404, "unresolved reference token '" + reference_token + "'")); + JSON_THROW(out_of_range::create(404, "unresolved reference token '" + reference_token + "'")); } } } @@ -12627,15 +12668,15 @@ basic_json_parser_74: if (reference_token == "-") { // "-" always fails the range check - JSON_THROW(out_of_range(402, "array index '-' (" + - std::to_string(ptr->m_value.array->size()) + - ") is out of range")); + JSON_THROW(out_of_range::create(402, "array index '-' (" + + std::to_string(ptr->m_value.array->size()) + + ") is out of range")); } // error condition (cf. RFC 6901, Sect. 4) if (reference_token.size() > 1 and reference_token[0] == '0') { - JSON_THROW(parse_error(106, 0, "array index '" + reference_token + "' must not begin with '0'")); + JSON_THROW(parse_error::create(106, 0, "array index '" + reference_token + "' must not begin with '0'")); } // note: at performs range check @@ -12645,14 +12686,14 @@ basic_json_parser_74: } JSON_CATCH (std::invalid_argument&) { - JSON_THROW(parse_error(109, 0, "array index '" + reference_token + "' is not a number")); + JSON_THROW(parse_error::create(109, 0, "array index '" + reference_token + "' is not a number")); } break; } default: { - JSON_THROW(out_of_range(404, "unresolved reference token '" + reference_token + "'")); + JSON_THROW(out_of_range::create(404, "unresolved reference token '" + reference_token + "'")); } } } @@ -12691,15 +12732,15 @@ basic_json_parser_74: if (reference_token == "-") { // "-" cannot be used for const access - JSON_THROW(out_of_range(402, "array index '-' (" + - std::to_string(ptr->m_value.array->size()) + - ") is out of range")); + JSON_THROW(out_of_range::create(402, "array index '-' (" + + std::to_string(ptr->m_value.array->size()) + + ") is out of range")); } // error condition (cf. RFC 6901, Sect. 4) if (reference_token.size() > 1 and reference_token[0] == '0') { - JSON_THROW(parse_error(106, 0, "array index '" + reference_token + "' must not begin with '0'")); + JSON_THROW(parse_error::create(106, 0, "array index '" + reference_token + "' must not begin with '0'")); } // use unchecked array access @@ -12709,14 +12750,14 @@ basic_json_parser_74: } JSON_CATCH (std::invalid_argument&) { - JSON_THROW(parse_error(109, 0, "array index '" + reference_token + "' is not a number")); + JSON_THROW(parse_error::create(109, 0, "array index '" + reference_token + "' is not a number")); } break; } default: { - JSON_THROW(out_of_range(404, "unresolved reference token '" + reference_token + "'")); + JSON_THROW(out_of_range::create(404, "unresolved reference token '" + reference_token + "'")); } } } @@ -12748,15 +12789,15 @@ basic_json_parser_74: if (reference_token == "-") { // "-" always fails the range check - JSON_THROW(out_of_range(402, "array index '-' (" + - std::to_string(ptr->m_value.array->size()) + - ") is out of range")); + JSON_THROW(out_of_range::create(402, "array index '-' (" + + std::to_string(ptr->m_value.array->size()) + + ") is out of range")); } // error condition (cf. RFC 6901, Sect. 4) if (reference_token.size() > 1 and reference_token[0] == '0') { - JSON_THROW(parse_error(106, 0, "array index '" + reference_token + "' must not begin with '0'")); + JSON_THROW(parse_error::create(106, 0, "array index '" + reference_token + "' must not begin with '0'")); } // note: at performs range check @@ -12766,14 +12807,14 @@ basic_json_parser_74: } JSON_CATCH (std::invalid_argument&) { - JSON_THROW(parse_error(109, 0, "array index '" + reference_token + "' is not a number")); + JSON_THROW(parse_error::create(109, 0, "array index '" + reference_token + "' is not a number")); } break; } default: { - JSON_THROW(out_of_range(404, "unresolved reference token '" + reference_token + "'")); + JSON_THROW(out_of_range::create(404, "unresolved reference token '" + reference_token + "'")); } } } @@ -12803,7 +12844,7 @@ basic_json_parser_74: // check if nonempty reference string begins with slash if (reference_string[0] != '/') { - JSON_THROW(parse_error(107, 1, "JSON pointer must be empty or begin with '/' - was: '" + reference_string + "'")); + JSON_THROW(parse_error::create(107, 1, "JSON pointer must be empty or begin with '/' - was: '" + reference_string + "'")); } // extract the reference tokens: @@ -12838,7 +12879,7 @@ basic_json_parser_74: (reference_token[pos + 1] != '0' and reference_token[pos + 1] != '1')) { - JSON_THROW(parse_error(108, 0, "escape character '~' must be followed with '0' or '1'")); + JSON_THROW(parse_error::create(108, 0, "escape character '~' must be followed with '0' or '1'")); } } @@ -12969,7 +13010,7 @@ basic_json_parser_74: { if (not value.is_object()) { - JSON_THROW(type_error(314, "only objects can be unflattened")); + JSON_THROW(type_error::create(314, "only objects can be unflattened")); } basic_json result; @@ -12979,7 +13020,7 @@ basic_json_parser_74: { if (not element.second.is_primitive()) { - JSON_THROW(type_error(315, "values in object must be primitive")); + JSON_THROW(type_error::create(315, "values in object must be primitive")); } // assign value to reference pointed to by JSON pointer; Note @@ -13364,7 +13405,7 @@ basic_json_parser_74: if (static_cast(idx) > parent.size()) { // avoid undefined behavior - JSON_THROW(out_of_range(401, "array index " + std::to_string(idx) + " is out of range")); + JSON_THROW(out_of_range::create(401, "array index " + std::to_string(idx) + " is out of range")); } else { @@ -13402,7 +13443,7 @@ basic_json_parser_74: } else { - JSON_THROW(out_of_range(403, "key '" + last_path + "' not found")); + JSON_THROW(out_of_range::create(403, "key '" + last_path + "' not found")); } } else if (parent.is_array()) @@ -13415,7 +13456,7 @@ basic_json_parser_74: // type check: top level value must be an array if (not json_patch.is_array()) { - JSON_THROW(parse_error(104, 0, "JSON patch must be an array of objects")); + JSON_THROW(parse_error::create(104, 0, "JSON patch must be an array of objects")); } // iterate and apply the operations @@ -13435,13 +13476,13 @@ basic_json_parser_74: // check if desired value is present if (it == val.m_value.object->end()) { - JSON_THROW(parse_error(105, 0, error_msg + " must have member '" + member + "'")); + JSON_THROW(parse_error::create(105, 0, error_msg + " must have member '" + member + "'")); } // check if result is of type string if (string_type and not it->second.is_string()) { - JSON_THROW(parse_error(105, 0, error_msg + " must have string member '" + member + "'")); + JSON_THROW(parse_error::create(105, 0, error_msg + " must have string member '" + member + "'")); } // no error: return value @@ -13451,7 +13492,7 @@ basic_json_parser_74: // type check: every element of the array must be an object if (not val.is_object()) { - JSON_THROW(parse_error(104, 0, "JSON patch must be an array of objects")); + JSON_THROW(parse_error::create(104, 0, "JSON patch must be an array of objects")); } // collect mandatory members @@ -13524,7 +13565,7 @@ basic_json_parser_74: // throw an exception if test fails if (not success) { - JSON_THROW(other_error(501, "unsuccessful: " + val.dump())); + JSON_THROW(other_error::create(501, "unsuccessful: " + val.dump())); } break; @@ -13534,7 +13575,7 @@ basic_json_parser_74: { // op must be "add", "remove", "replace", "move", "copy", or // "test" - JSON_THROW(parse_error(105, 0, "operation value '" + op + "' is invalid")); + JSON_THROW(parse_error::create(105, 0, "operation value '" + op + "' is invalid")); } } } diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 1480a5f5..d5b12d2c 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -116,29 +116,34 @@ namespace detail Extension of std::exception objects with a member @a id for exception ids. +@note To have nothrow-copy-constructible exceptions, we inherit from + std::runtime_error which can cope with arbitrary-length error messages. + Intermediate strings are built with static functions and then passed to + the actual constructor. + @since version 3.0.0 */ -class exception : public std::exception +class exception : public std::runtime_error { public: - /// create exception with id an explanatory string - exception(int id_, const std::string& ename, const std::string& what_arg_) - : id(id_), - what_arg("[json.exception." + ename + "." + std::to_string(id_) + "] " + what_arg_) - {} - /// returns the explanatory string virtual const char* what() const noexcept override { - return what_arg.c_str(); + return std::runtime_error::what(); } /// the id of the exception const int id; - private: - /// the explanatory string - const std::string what_arg; + protected: + exception(int id_, const char* what_arg) + : std::runtime_error(what_arg), id(id_) + {} + + static std::string name(const std::string& ename, int id_) + { + return "[json.exception." + ename + "." + std::to_string(id_) + "] "; + } }; /*! @@ -184,14 +189,16 @@ class parse_error : public exception @param[in] id_ the id of the exception @param[in] byte_ the byte index where the error occured (or 0 if the position cannot be determined) - @param[in] what_arg_ the explanatory string + @param[in] what_arg the explanatory string + @return parse_error object */ - parse_error(int id_, size_t byte_, const std::string& what_arg_) - : exception(id_, "parse_error", "parse error" + - (byte_ != 0 ? (" at " + std::to_string(byte_)) : "") + - ": " + what_arg_), - byte(byte_) - {} + static parse_error create(int id_, size_t byte_, const std::string& what_arg) + { + std::string w = exception::name("parse_error", id_) + "parse error" + + (byte_ != 0 ? (" at " + std::to_string(byte_)) : "") + + ": " + what_arg; + return parse_error(id_, byte_, w.c_str()); + } /*! @brief byte index of the parse error @@ -204,6 +211,12 @@ class parse_error : public exception MessagePack). */ const size_t byte; + + private: + parse_error(int id_, size_t byte_, const char* what_arg) + : exception(id_, what_arg), + byte(byte_) + {} }; /*! @@ -233,8 +246,15 @@ json.exception.invalid_iterator.214 | cannot get value | Cannot get value for it class invalid_iterator : public exception { public: - invalid_iterator(int id_, const std::string& what_arg_) - : exception(id_, "invalid_iterator", what_arg_) + static invalid_iterator create(int id_, const std::string& what_arg) + { + std::string w = exception::name("invalid_iterator", id_) + what_arg; + return invalid_iterator(id_, w.c_str()); + } + + private: + invalid_iterator(int id_, const char* what_arg) + : exception(id_, what_arg) {} }; @@ -265,8 +285,15 @@ json.exception.type_error.315 | values in object must be primitive | The @ref un class type_error : public exception { public: - type_error(int id_, const std::string& what_arg_) - : exception(id_, "type_error", what_arg_) + static type_error create(int id_, const std::string& what_arg) + { + std::string w = exception::name("type_error", id_) + what_arg; + return type_error(id_, w.c_str()); + } + + private: + type_error(int id_, const char* what_arg) + : exception(id_, what_arg) {} }; @@ -289,8 +316,15 @@ json.exception.out_of_range.406 | number overflow parsing '10E1000' | A parsed n class out_of_range : public exception { public: - out_of_range(int id_, const std::string& what_arg_) - : exception(id_, "out_of_range", what_arg_) + static out_of_range create(int id_, const std::string& what_arg) + { + std::string w = exception::name("out_of_range", id_) + what_arg; + return out_of_range(id_, w.c_str()); + } + + private: + out_of_range(int id_, const char* what_arg) + : exception(id_, what_arg) {} }; @@ -308,8 +342,15 @@ json.exception.other_error.501 | unsuccessful: {"op":"test","path":"/baz", "valu class other_error : public exception { public: - other_error(int id_, const std::string& what_arg_) - : exception(id_, "other_error", what_arg_) + static other_error create(int id_, const std::string& what_arg) + { + std::string w = exception::name("other_error", id_) + what_arg; + return other_error(id_, w.c_str()); + } + + private: + other_error(int id_, const char* what_arg) + : exception(id_, what_arg) {} }; @@ -841,7 +882,7 @@ void get_arithmetic_value(const BasicJsonType& j, ArithmeticType& val) } default: { - JSON_THROW(type_error(302, "type must be number, but is " + j.type_name())); + JSON_THROW(type_error::create(302, "type must be number, but is " + j.type_name())); } } } @@ -851,7 +892,7 @@ void from_json(const BasicJsonType& j, typename BasicJsonType::boolean_t& b) { if (not j.is_boolean()) { - JSON_THROW(type_error(302, "type must be boolean, but is " + j.type_name())); + JSON_THROW(type_error::create(302, "type must be boolean, but is " + j.type_name())); } b = *j.template get_ptr(); } @@ -861,7 +902,7 @@ void from_json(const BasicJsonType& j, typename BasicJsonType::string_t& s) { if (not j.is_string()) { - JSON_THROW(type_error(302, "type must be string, but is " + j.type_name())); + JSON_THROW(type_error::create(302, "type must be string, but is " + j.type_name())); } s = *j.template get_ptr(); } @@ -898,7 +939,7 @@ void from_json(const BasicJsonType& j, typename BasicJsonType::array_t& arr) { if (not j.is_array()) { - JSON_THROW(type_error(302, "type must be array, but is " + j.type_name())); + JSON_THROW(type_error::create(302, "type must be array, but is " + j.type_name())); } arr = *j.template get_ptr(); } @@ -910,7 +951,7 @@ void from_json(const BasicJsonType& j, std::forward_list& l) { if (not j.is_array()) { - JSON_THROW(type_error(302, "type must be array, but is " + j.type_name())); + JSON_THROW(type_error::create(302, "type must be array, but is " + j.type_name())); } for (auto it = j.rbegin(), end = j.rend(); it != end; ++it) @@ -961,7 +1002,7 @@ void from_json(const BasicJsonType& j, CompatibleArrayType& arr) { if (not j.is_array()) { - JSON_THROW(type_error(302, "type must be array, but is " + j.type_name())); + JSON_THROW(type_error::create(302, "type must be array, but is " + j.type_name())); } from_json_array_impl(j, arr, priority_tag<1> {}); @@ -973,7 +1014,7 @@ void from_json(const BasicJsonType& j, CompatibleObjectType& obj) { if (not j.is_object()) { - JSON_THROW(type_error(302, "type must be object, but is " + j.type_name())); + JSON_THROW(type_error::create(302, "type must be object, but is " + j.type_name())); } auto inner_object = j.template get_ptr(); @@ -1023,7 +1064,7 @@ void from_json(const BasicJsonType& j, ArithmeticType& val) } default: { - JSON_THROW(type_error(302, "type must be number, but is " + j.type_name())); + JSON_THROW(type_error::create(302, "type must be number, but is " + j.type_name())); } } } @@ -1970,7 +2011,7 @@ class basic_json { if (t == value_t::null) { - JSON_THROW(other_error(500, "961c151d2e87f2686a955a9be24d316f1362bf21 2.1.1")); // LCOV_EXCL_LINE + JSON_THROW(other_error::create(500, "961c151d2e87f2686a955a9be24d316f1362bf21 2.1.1")); // LCOV_EXCL_LINE } break; } @@ -2325,7 +2366,7 @@ class basic_json // if object is wanted but impossible, throw an exception if (manual_type == value_t::object and not is_an_object) { - JSON_THROW(type_error(301, "cannot create object from initializer list")); + JSON_THROW(type_error::create(301, "cannot create object from initializer list")); } } @@ -2509,7 +2550,7 @@ class basic_json // make sure iterator fits the current value if (first.m_object != last.m_object) { - JSON_THROW(invalid_iterator(201, "iterators are not compatible")); + JSON_THROW(invalid_iterator::create(201, "iterators are not compatible")); } // copy type from first iterator @@ -2526,7 +2567,7 @@ class basic_json { if (not first.m_it.primitive_iterator.is_begin() or not last.m_it.primitive_iterator.is_end()) { - JSON_THROW(invalid_iterator(204, "iterators out of range")); + JSON_THROW(invalid_iterator::create(204, "iterators out of range")); } break; } @@ -2585,8 +2626,8 @@ class basic_json default: { - JSON_THROW(invalid_iterator(206, "cannot construct with iterators from " + - first.m_object->type_name())); + JSON_THROW(invalid_iterator::create(206, "cannot construct with iterators from " + + first.m_object->type_name())); } } @@ -3224,7 +3265,7 @@ class basic_json return m_value.boolean; } - JSON_THROW(type_error(302, "type must be boolean, but is " + type_name())); + JSON_THROW(type_error::create(302, "type must be boolean, but is " + type_name())); } /// get a pointer to the value (object) @@ -3336,7 +3377,7 @@ class basic_json return *ptr; } - JSON_THROW(type_error(303, "incompatible ReferenceType for get_ref, actual type is " + obj.type_name())); + JSON_THROW(type_error::create(303, "incompatible ReferenceType for get_ref, actual type is " + obj.type_name())); } public: @@ -3738,12 +3779,12 @@ class basic_json JSON_CATCH (std::out_of_range&) { // create better exception explanation - JSON_THROW(out_of_range(401, "array index " + std::to_string(idx) + " is out of range")); + JSON_THROW(out_of_range::create(401, "array index " + std::to_string(idx) + " is out of range")); } } else { - JSON_THROW(type_error(304, "cannot use at() with " + type_name())); + JSON_THROW(type_error::create(304, "cannot use at() with " + type_name())); } } @@ -3785,12 +3826,12 @@ class basic_json JSON_CATCH (std::out_of_range&) { // create better exception explanation - JSON_THROW(out_of_range(401, "array index " + std::to_string(idx) + " is out of range")); + JSON_THROW(out_of_range::create(401, "array index " + std::to_string(idx) + " is out of range")); } } else { - JSON_THROW(type_error(304, "cannot use at() with " + type_name())); + JSON_THROW(type_error::create(304, "cannot use at() with " + type_name())); } } @@ -3836,12 +3877,12 @@ class basic_json JSON_CATCH (std::out_of_range&) { // create better exception explanation - JSON_THROW(out_of_range(403, "key '" + key + "' not found")); + JSON_THROW(out_of_range::create(403, "key '" + key + "' not found")); } } else { - JSON_THROW(type_error(304, "cannot use at() with " + type_name())); + JSON_THROW(type_error::create(304, "cannot use at() with " + type_name())); } } @@ -3887,12 +3928,12 @@ class basic_json JSON_CATCH (std::out_of_range&) { // create better exception explanation - JSON_THROW(out_of_range(403, "key '" + key + "' not found")); + JSON_THROW(out_of_range::create(403, "key '" + key + "' not found")); } } else { - JSON_THROW(type_error(304, "cannot use at() with " + type_name())); + JSON_THROW(type_error::create(304, "cannot use at() with " + type_name())); } } @@ -3945,7 +3986,7 @@ class basic_json return m_value.array->operator[](idx); } - JSON_THROW(type_error(305, "cannot use operator[] with " + type_name())); + JSON_THROW(type_error::create(305, "cannot use operator[] with " + type_name())); } /*! @@ -3975,7 +4016,7 @@ class basic_json return m_value.array->operator[](idx); } - JSON_THROW(type_error(305, "cannot use operator[] with " + type_name())); + JSON_THROW(type_error::create(305, "cannot use operator[] with " + type_name())); } /*! @@ -4021,7 +4062,7 @@ class basic_json return m_value.object->operator[](key); } - JSON_THROW(type_error(305, "cannot use operator[] with " + type_name())); + JSON_THROW(type_error::create(305, "cannot use operator[] with " + type_name())); } /*! @@ -4063,7 +4104,7 @@ class basic_json return m_value.object->find(key)->second; } - JSON_THROW(type_error(305, "cannot use operator[] with " + type_name())); + JSON_THROW(type_error::create(305, "cannot use operator[] with " + type_name())); } /*! @@ -4178,7 +4219,7 @@ class basic_json return m_value.object->operator[](key); } - JSON_THROW(type_error(305, "cannot use operator[] with " + type_name())); + JSON_THROW(type_error::create(305, "cannot use operator[] with " + type_name())); } /*! @@ -4221,7 +4262,7 @@ class basic_json return m_value.object->find(key)->second; } - JSON_THROW(type_error(305, "cannot use operator[] with " + type_name())); + JSON_THROW(type_error::create(305, "cannot use operator[] with " + type_name())); } /*! @@ -4290,7 +4331,7 @@ class basic_json } else { - JSON_THROW(type_error(306, "cannot use value() with " + type_name())); + JSON_THROW(type_error::create(306, "cannot use value() with " + type_name())); } } @@ -4362,7 +4403,7 @@ class basic_json } } - JSON_THROW(type_error(306, "cannot use value() with " + type_name())); + JSON_THROW(type_error::create(306, "cannot use value() with " + type_name())); } /*! @@ -4515,7 +4556,7 @@ class basic_json // make sure iterator fits the current value if (this != pos.m_object) { - JSON_THROW(invalid_iterator(202, "iterator does not fit current value")); + JSON_THROW(invalid_iterator::create(202, "iterator does not fit current value")); } IteratorType result = end(); @@ -4530,7 +4571,7 @@ class basic_json { if (not pos.m_it.primitive_iterator.is_begin()) { - JSON_THROW(invalid_iterator(205, "iterator out of range")); + JSON_THROW(invalid_iterator::create(205, "iterator out of range")); } if (is_string()) @@ -4560,7 +4601,7 @@ class basic_json default: { - JSON_THROW(type_error(307, "cannot use erase() with " + type_name())); + JSON_THROW(type_error::create(307, "cannot use erase() with " + type_name())); } } @@ -4622,7 +4663,7 @@ class basic_json // make sure iterator fits the current value if (this != first.m_object or this != last.m_object) { - JSON_THROW(invalid_iterator(203, "iterators do not fit current value")); + JSON_THROW(invalid_iterator::create(203, "iterators do not fit current value")); } IteratorType result = end(); @@ -4637,7 +4678,7 @@ class basic_json { if (not first.m_it.primitive_iterator.is_begin() or not last.m_it.primitive_iterator.is_end()) { - JSON_THROW(invalid_iterator(204, "iterators out of range")); + JSON_THROW(invalid_iterator::create(204, "iterators out of range")); } if (is_string()) @@ -4669,7 +4710,7 @@ class basic_json default: { - JSON_THROW(type_error(307, "cannot use erase() with " + type_name())); + JSON_THROW(type_error::create(307, "cannot use erase() with " + type_name())); } } @@ -4713,7 +4754,7 @@ class basic_json return m_value.object->erase(key); } - JSON_THROW(type_error(307, "cannot use erase() with " + type_name())); + JSON_THROW(type_error::create(307, "cannot use erase() with " + type_name())); } /*! @@ -4747,14 +4788,14 @@ class basic_json { if (idx >= size()) { - JSON_THROW(out_of_range(401, "array index " + std::to_string(idx) + " is out of range")); + JSON_THROW(out_of_range::create(401, "array index " + std::to_string(idx) + " is out of range")); } m_value.array->erase(m_value.array->begin() + static_cast(idx)); } else { - JSON_THROW(type_error(307, "cannot use erase() with " + type_name())); + JSON_THROW(type_error::create(307, "cannot use erase() with " + type_name())); } } @@ -5472,7 +5513,7 @@ class basic_json // push_back only works for null objects or arrays if (not(is_null() or is_array())) { - JSON_THROW(type_error(308, "cannot use push_back() with " + type_name())); + JSON_THROW(type_error::create(308, "cannot use push_back() with " + type_name())); } // transform null object into an array @@ -5508,7 +5549,7 @@ class basic_json // push_back only works for null objects or arrays if (not(is_null() or is_array())) { - JSON_THROW(type_error(308, "cannot use push_back() with " + type_name())); + JSON_THROW(type_error::create(308, "cannot use push_back() with " + type_name())); } // transform null object into an array @@ -5558,7 +5599,7 @@ class basic_json // push_back only works for null objects or objects if (not(is_null() or is_object())) { - JSON_THROW(type_error(308, "cannot use push_back() with " + type_name())); + JSON_THROW(type_error::create(308, "cannot use push_back() with " + type_name())); } // transform null object into an object @@ -5658,7 +5699,7 @@ class basic_json // emplace_back only works for null objects or arrays if (not(is_null() or is_array())) { - JSON_THROW(type_error(311, "cannot use emplace_back() with " + type_name())); + JSON_THROW(type_error::create(311, "cannot use emplace_back() with " + type_name())); } // transform null object into an array @@ -5706,7 +5747,7 @@ class basic_json // emplace only works for null objects or arrays if (not(is_null() or is_object())) { - JSON_THROW(type_error(311, "cannot use emplace() with " + type_name())); + JSON_THROW(type_error::create(311, "cannot use emplace() with " + type_name())); } // transform null object into an object @@ -5757,7 +5798,7 @@ class basic_json // check if iterator pos fits to this JSON value if (pos.m_object != this) { - JSON_THROW(invalid_iterator(202, "iterator does not fit current value")); + JSON_THROW(invalid_iterator::create(202, "iterator does not fit current value")); } // insert to array and return iterator @@ -5766,7 +5807,7 @@ class basic_json return result; } - JSON_THROW(type_error(309, "cannot use insert() with " + type_name())); + JSON_THROW(type_error::create(309, "cannot use insert() with " + type_name())); } /*! @@ -5810,7 +5851,7 @@ class basic_json // check if iterator pos fits to this JSON value if (pos.m_object != this) { - JSON_THROW(invalid_iterator(202, "iterator does not fit current value")); + JSON_THROW(invalid_iterator::create(202, "iterator does not fit current value")); } // insert to array and return iterator @@ -5819,7 +5860,7 @@ class basic_json return result; } - JSON_THROW(type_error(309, "cannot use insert() with " + type_name())); + JSON_THROW(type_error::create(309, "cannot use insert() with " + type_name())); } /*! @@ -5857,24 +5898,24 @@ class basic_json // insert only works for arrays if (not is_array()) { - JSON_THROW(type_error(309, "cannot use insert() with " + type_name())); + JSON_THROW(type_error::create(309, "cannot use insert() with " + type_name())); } // check if iterator pos fits to this JSON value if (pos.m_object != this) { - JSON_THROW(invalid_iterator(202, "iterator does not fit current value")); + JSON_THROW(invalid_iterator::create(202, "iterator does not fit current value")); } // check if range iterators belong to the same JSON object if (first.m_object != last.m_object) { - JSON_THROW(invalid_iterator(210, "iterators do not fit")); + JSON_THROW(invalid_iterator::create(210, "iterators do not fit")); } if (first.m_object == this or last.m_object == this) { - JSON_THROW(invalid_iterator(211, "passed iterators may not belong to container")); + JSON_THROW(invalid_iterator::create(211, "passed iterators may not belong to container")); } // insert to array and return iterator @@ -5915,13 +5956,13 @@ class basic_json // insert only works for arrays if (not is_array()) { - JSON_THROW(type_error(309, "cannot use insert() with " + type_name())); + JSON_THROW(type_error::create(309, "cannot use insert() with " + type_name())); } // check if iterator pos fits to this JSON value if (pos.m_object != this) { - JSON_THROW(invalid_iterator(202, "iterator does not fit current value")); + JSON_THROW(invalid_iterator::create(202, "iterator does not fit current value")); } // insert to array and return iterator @@ -5988,7 +6029,7 @@ class basic_json } else { - JSON_THROW(type_error(310, "cannot use swap() with " + type_name())); + JSON_THROW(type_error::create(310, "cannot use swap() with " + type_name())); } } @@ -6021,7 +6062,7 @@ class basic_json } else { - JSON_THROW(type_error(310, "cannot use swap() with " + type_name())); + JSON_THROW(type_error::create(310, "cannot use swap() with " + type_name())); } } @@ -6054,7 +6095,7 @@ class basic_json } else { - JSON_THROW(type_error(310, "cannot use swap() with " + type_name())); + JSON_THROW(type_error::create(310, "cannot use swap() with " + type_name())); } } @@ -7999,19 +8040,19 @@ class basic_json // simple case: requested length is greater than the vector's length if (len > size or offset > size) { - JSON_THROW(parse_error(110, offset + 1, "cannot read " + std::to_string(len) + " bytes from vector")); + JSON_THROW(parse_error::create(110, offset + 1, "cannot read " + std::to_string(len) + " bytes from vector")); } // second case: adding offset would result in overflow if ((size > ((std::numeric_limits::max)() - offset))) { - JSON_THROW(parse_error(110, offset + 1, "cannot read " + std::to_string(len) + " bytes from vector")); + JSON_THROW(parse_error::create(110, offset + 1, "cannot read " + std::to_string(len) + " bytes from vector")); } // last case: reading past the end of the vector if (len + offset > size) { - JSON_THROW(parse_error(110, offset + 1, "cannot read " + std::to_string(len) + " bytes from vector")); + JSON_THROW(parse_error::create(110, offset + 1, "cannot read " + std::to_string(len) + " bytes from vector")); } } @@ -8043,7 +8084,7 @@ class basic_json std::stringstream ss; ss << std::hex << static_cast(v[idx]); - JSON_THROW(parse_error(113, idx + 1, "expected a MessagePack string; last byte: 0x" + ss.str())); + JSON_THROW(parse_error::create(113, idx + 1, "expected a MessagePack string; last byte: 0x" + ss.str())); } /*! @@ -8073,7 +8114,7 @@ class basic_json std::stringstream ss; ss << std::hex << static_cast(v[idx]); - JSON_THROW(parse_error(113, idx + 1, "expected a CBOR string; last byte: 0x" + ss.str())); + JSON_THROW(parse_error::create(113, idx + 1, "expected a CBOR string; last byte: 0x" + ss.str())); } /*! @@ -8316,7 +8357,7 @@ class basic_json { std::stringstream ss; ss << std::hex << static_cast(v[current_idx]); - JSON_THROW(parse_error(112, current_idx + 1, "error reading MessagePack; last byte: 0x" + ss.str())); + JSON_THROW(parse_error::create(112, current_idx + 1, "error reading MessagePack; last byte: 0x" + ss.str())); } } } @@ -8816,7 +8857,7 @@ class basic_json { std::stringstream ss; ss << std::hex << static_cast(v[current_idx]); - JSON_THROW(parse_error(112, current_idx + 1, "error reading CBOR; last byte: 0x" + ss.str())); + JSON_THROW(parse_error::create(112, current_idx + 1, "error reading CBOR; last byte: 0x" + ss.str())); } } } @@ -9697,7 +9738,7 @@ class basic_json case basic_json::value_t::null: { - JSON_THROW(invalid_iterator(214, "cannot get value")); + JSON_THROW(invalid_iterator::create(214, "cannot get value")); } default: @@ -9707,7 +9748,7 @@ class basic_json return *m_object; } - JSON_THROW(invalid_iterator(214, "cannot get value")); + JSON_THROW(invalid_iterator::create(214, "cannot get value")); } } } @@ -9741,7 +9782,7 @@ class basic_json return m_object; } - JSON_THROW(invalid_iterator(214, "cannot get value")); + JSON_THROW(invalid_iterator::create(214, "cannot get value")); } } } @@ -9841,7 +9882,7 @@ class basic_json // if objects are not the same, the comparison is undefined if (m_object != other.m_object) { - JSON_THROW(invalid_iterator(212, "cannot compare iterators of different containers")); + JSON_THROW(invalid_iterator::create(212, "cannot compare iterators of different containers")); } assert(m_object != nullptr); @@ -9883,7 +9924,7 @@ class basic_json // if objects are not the same, the comparison is undefined if (m_object != other.m_object) { - JSON_THROW(invalid_iterator(212, "cannot compare iterators of different containers")); + JSON_THROW(invalid_iterator::create(212, "cannot compare iterators of different containers")); } assert(m_object != nullptr); @@ -9892,7 +9933,7 @@ class basic_json { case basic_json::value_t::object: { - JSON_THROW(invalid_iterator(213, "cannot compare order of object iterators")); + JSON_THROW(invalid_iterator::create(213, "cannot compare order of object iterators")); } case basic_json::value_t::array: @@ -9946,7 +9987,7 @@ class basic_json { case basic_json::value_t::object: { - JSON_THROW(invalid_iterator(209, "cannot use offsets with object iterators")); + JSON_THROW(invalid_iterator::create(209, "cannot use offsets with object iterators")); } case basic_json::value_t::array: @@ -10008,7 +10049,7 @@ class basic_json { case basic_json::value_t::object: { - JSON_THROW(invalid_iterator(209, "cannot use offsets with object iterators")); + JSON_THROW(invalid_iterator::create(209, "cannot use offsets with object iterators")); } case basic_json::value_t::array: @@ -10035,7 +10076,7 @@ class basic_json { case basic_json::value_t::object: { - JSON_THROW(invalid_iterator(208, "cannot use operator[] for object iterators")); + JSON_THROW(invalid_iterator::create(208, "cannot use operator[] for object iterators")); } case basic_json::value_t::array: @@ -10045,7 +10086,7 @@ class basic_json case basic_json::value_t::null: { - JSON_THROW(invalid_iterator(214, "cannot get value")); + JSON_THROW(invalid_iterator::create(214, "cannot get value")); } default: @@ -10055,7 +10096,7 @@ class basic_json return *m_object; } - JSON_THROW(invalid_iterator(214, "cannot get value")); + JSON_THROW(invalid_iterator::create(214, "cannot get value")); } } } @@ -10073,7 +10114,7 @@ class basic_json return m_it.object_iterator->first; } - JSON_THROW(invalid_iterator(207, "cannot use key() for non-object iterators")); + JSON_THROW(invalid_iterator::create(207, "cannot use key() for non-object iterators")); } /*! @@ -10263,7 +10304,7 @@ class basic_json // immediately abort if stream is erroneous if (s.fail()) { - JSON_THROW(parse_error(111, 0, "bad input stream")); + JSON_THROW(parse_error::create(111, 0, "bad input stream")); } // fill buffer @@ -10330,7 +10371,7 @@ class basic_json } else { - JSON_THROW(parse_error(102, get_position(), "missing or wrong low surrogate")); + JSON_THROW(parse_error::create(102, get_position(), "missing or wrong low surrogate")); } } @@ -10364,7 +10405,7 @@ class basic_json } else { - JSON_THROW(parse_error(103, get_position(), "code points above 0x10FFFF are invalid")); + JSON_THROW(parse_error::create(103, get_position(), "code points above 0x10FFFF are invalid")); } return result; @@ -10593,7 +10634,7 @@ class basic_json // check if stream is still good if (m_stream->fail()) { - JSON_THROW(parse_error(111, 0, "bad input stream")); + JSON_THROW(parse_error::create(111, 0, "bad input stream")); } std::getline(*m_stream, m_line_buffer_tmp, '\n'); @@ -10762,7 +10803,7 @@ class basic_json // make sure there is a subsequent unicode if ((i + 6 >= m_limit) or * (i + 5) != '\\' or * (i + 6) != 'u') { - JSON_THROW(parse_error(102, get_position(), "missing low surrogate")); + JSON_THROW(parse_error::create(102, get_position(), "missing low surrogate")); } // get code yyyy from uxxxx\uyyyy @@ -10775,7 +10816,7 @@ class basic_json else if (codepoint >= 0xDC00 and codepoint <= 0xDFFF) { // we found a lone low surrogate - JSON_THROW(parse_error(102, get_position(), "missing high surrogate")); + JSON_THROW(parse_error::create(102, get_position(), "missing high surrogate")); } else { @@ -11019,7 +11060,7 @@ class basic_json // throw in case of infinity or NAN if (not std::isfinite(result.m_value.number_float)) { - JSON_THROW(out_of_range(406, "number overflow parsing '" + get_token_string() + "'")); + JSON_THROW(out_of_range::create(406, "number overflow parsing '" + get_token_string() + "'")); } return true; @@ -11332,7 +11373,7 @@ class basic_json "'") : lexer::token_type_name(last_token)); error_msg += "; expected " + lexer::token_type_name(t); - JSON_THROW(parse_error(101, m_lexer.get_position(), error_msg)); + JSON_THROW(parse_error::create(101, m_lexer.get_position(), error_msg)); } } @@ -11347,7 +11388,7 @@ class basic_json error_msg += (last_token == lexer::token_type::parse_error ? ("'" + m_lexer.get_token_string() + "'") : lexer::token_type_name(last_token)); - JSON_THROW(parse_error(101, m_lexer.get_position(), error_msg)); + JSON_THROW(parse_error::create(101, m_lexer.get_position(), error_msg)); } } @@ -11446,7 +11487,7 @@ class basic_json { if (is_root()) { - JSON_THROW(out_of_range(405, "JSON pointer has no parent")); + JSON_THROW(out_of_range::create(405, "JSON pointer has no parent")); } auto last = reference_tokens.back(); @@ -11464,7 +11505,7 @@ class basic_json { if (is_root()) { - JSON_THROW(out_of_range(405, "JSON pointer has no parent")); + JSON_THROW(out_of_range::create(405, "JSON pointer has no parent")); } json_pointer result = *this; @@ -11521,7 +11562,7 @@ class basic_json } JSON_CATCH (std::invalid_argument&) { - JSON_THROW(parse_error(109, 0, "array index '" + reference_token + "' is not a number")); + JSON_THROW(parse_error::create(109, 0, "array index '" + reference_token + "' is not a number")); } break; } @@ -11535,7 +11576,7 @@ class basic_json */ default: { - JSON_THROW(type_error(313, "invalid value to unflatten")); + JSON_THROW(type_error::create(313, "invalid value to unflatten")); } } } @@ -11603,7 +11644,7 @@ class basic_json // error condition (cf. RFC 6901, Sect. 4) if (reference_token.size() > 1 and reference_token[0] == '0') { - JSON_THROW(parse_error(106, 0, "array index '" + reference_token + "' must not begin with '0'")); + JSON_THROW(parse_error::create(106, 0, "array index '" + reference_token + "' must not begin with '0'")); } if (reference_token == "-") @@ -11620,7 +11661,7 @@ class basic_json } JSON_CATCH (std::invalid_argument&) { - JSON_THROW(parse_error(109, 0, "array index '" + reference_token + "' is not a number")); + JSON_THROW(parse_error::create(109, 0, "array index '" + reference_token + "' is not a number")); } } break; @@ -11628,7 +11669,7 @@ class basic_json default: { - JSON_THROW(out_of_range(404, "unresolved reference token '" + reference_token + "'")); + JSON_THROW(out_of_range::create(404, "unresolved reference token '" + reference_token + "'")); } } } @@ -11660,15 +11701,15 @@ class basic_json if (reference_token == "-") { // "-" always fails the range check - JSON_THROW(out_of_range(402, "array index '-' (" + - std::to_string(ptr->m_value.array->size()) + - ") is out of range")); + JSON_THROW(out_of_range::create(402, "array index '-' (" + + std::to_string(ptr->m_value.array->size()) + + ") is out of range")); } // error condition (cf. RFC 6901, Sect. 4) if (reference_token.size() > 1 and reference_token[0] == '0') { - JSON_THROW(parse_error(106, 0, "array index '" + reference_token + "' must not begin with '0'")); + JSON_THROW(parse_error::create(106, 0, "array index '" + reference_token + "' must not begin with '0'")); } // note: at performs range check @@ -11678,14 +11719,14 @@ class basic_json } JSON_CATCH (std::invalid_argument&) { - JSON_THROW(parse_error(109, 0, "array index '" + reference_token + "' is not a number")); + JSON_THROW(parse_error::create(109, 0, "array index '" + reference_token + "' is not a number")); } break; } default: { - JSON_THROW(out_of_range(404, "unresolved reference token '" + reference_token + "'")); + JSON_THROW(out_of_range::create(404, "unresolved reference token '" + reference_token + "'")); } } } @@ -11724,15 +11765,15 @@ class basic_json if (reference_token == "-") { // "-" cannot be used for const access - JSON_THROW(out_of_range(402, "array index '-' (" + - std::to_string(ptr->m_value.array->size()) + - ") is out of range")); + JSON_THROW(out_of_range::create(402, "array index '-' (" + + std::to_string(ptr->m_value.array->size()) + + ") is out of range")); } // error condition (cf. RFC 6901, Sect. 4) if (reference_token.size() > 1 and reference_token[0] == '0') { - JSON_THROW(parse_error(106, 0, "array index '" + reference_token + "' must not begin with '0'")); + JSON_THROW(parse_error::create(106, 0, "array index '" + reference_token + "' must not begin with '0'")); } // use unchecked array access @@ -11742,14 +11783,14 @@ class basic_json } JSON_CATCH (std::invalid_argument&) { - JSON_THROW(parse_error(109, 0, "array index '" + reference_token + "' is not a number")); + JSON_THROW(parse_error::create(109, 0, "array index '" + reference_token + "' is not a number")); } break; } default: { - JSON_THROW(out_of_range(404, "unresolved reference token '" + reference_token + "'")); + JSON_THROW(out_of_range::create(404, "unresolved reference token '" + reference_token + "'")); } } } @@ -11781,15 +11822,15 @@ class basic_json if (reference_token == "-") { // "-" always fails the range check - JSON_THROW(out_of_range(402, "array index '-' (" + - std::to_string(ptr->m_value.array->size()) + - ") is out of range")); + JSON_THROW(out_of_range::create(402, "array index '-' (" + + std::to_string(ptr->m_value.array->size()) + + ") is out of range")); } // error condition (cf. RFC 6901, Sect. 4) if (reference_token.size() > 1 and reference_token[0] == '0') { - JSON_THROW(parse_error(106, 0, "array index '" + reference_token + "' must not begin with '0'")); + JSON_THROW(parse_error::create(106, 0, "array index '" + reference_token + "' must not begin with '0'")); } // note: at performs range check @@ -11799,14 +11840,14 @@ class basic_json } JSON_CATCH (std::invalid_argument&) { - JSON_THROW(parse_error(109, 0, "array index '" + reference_token + "' is not a number")); + JSON_THROW(parse_error::create(109, 0, "array index '" + reference_token + "' is not a number")); } break; } default: { - JSON_THROW(out_of_range(404, "unresolved reference token '" + reference_token + "'")); + JSON_THROW(out_of_range::create(404, "unresolved reference token '" + reference_token + "'")); } } } @@ -11836,7 +11877,7 @@ class basic_json // check if nonempty reference string begins with slash if (reference_string[0] != '/') { - JSON_THROW(parse_error(107, 1, "JSON pointer must be empty or begin with '/' - was: '" + reference_string + "'")); + JSON_THROW(parse_error::create(107, 1, "JSON pointer must be empty or begin with '/' - was: '" + reference_string + "'")); } // extract the reference tokens: @@ -11871,7 +11912,7 @@ class basic_json (reference_token[pos + 1] != '0' and reference_token[pos + 1] != '1')) { - JSON_THROW(parse_error(108, 0, "escape character '~' must be followed with '0' or '1'")); + JSON_THROW(parse_error::create(108, 0, "escape character '~' must be followed with '0' or '1'")); } } @@ -12002,7 +12043,7 @@ class basic_json { if (not value.is_object()) { - JSON_THROW(type_error(314, "only objects can be unflattened")); + JSON_THROW(type_error::create(314, "only objects can be unflattened")); } basic_json result; @@ -12012,7 +12053,7 @@ class basic_json { if (not element.second.is_primitive()) { - JSON_THROW(type_error(315, "values in object must be primitive")); + JSON_THROW(type_error::create(315, "values in object must be primitive")); } // assign value to reference pointed to by JSON pointer; Note @@ -12397,7 +12438,7 @@ class basic_json if (static_cast(idx) > parent.size()) { // avoid undefined behavior - JSON_THROW(out_of_range(401, "array index " + std::to_string(idx) + " is out of range")); + JSON_THROW(out_of_range::create(401, "array index " + std::to_string(idx) + " is out of range")); } else { @@ -12435,7 +12476,7 @@ class basic_json } else { - JSON_THROW(out_of_range(403, "key '" + last_path + "' not found")); + JSON_THROW(out_of_range::create(403, "key '" + last_path + "' not found")); } } else if (parent.is_array()) @@ -12448,7 +12489,7 @@ class basic_json // type check: top level value must be an array if (not json_patch.is_array()) { - JSON_THROW(parse_error(104, 0, "JSON patch must be an array of objects")); + JSON_THROW(parse_error::create(104, 0, "JSON patch must be an array of objects")); } // iterate and apply the operations @@ -12468,13 +12509,13 @@ class basic_json // check if desired value is present if (it == val.m_value.object->end()) { - JSON_THROW(parse_error(105, 0, error_msg + " must have member '" + member + "'")); + JSON_THROW(parse_error::create(105, 0, error_msg + " must have member '" + member + "'")); } // check if result is of type string if (string_type and not it->second.is_string()) { - JSON_THROW(parse_error(105, 0, error_msg + " must have string member '" + member + "'")); + JSON_THROW(parse_error::create(105, 0, error_msg + " must have string member '" + member + "'")); } // no error: return value @@ -12484,7 +12525,7 @@ class basic_json // type check: every element of the array must be an object if (not val.is_object()) { - JSON_THROW(parse_error(104, 0, "JSON patch must be an array of objects")); + JSON_THROW(parse_error::create(104, 0, "JSON patch must be an array of objects")); } // collect mandatory members @@ -12557,7 +12598,7 @@ class basic_json // throw an exception if test fails if (not success) { - JSON_THROW(other_error(501, "unsuccessful: " + val.dump())); + JSON_THROW(other_error::create(501, "unsuccessful: " + val.dump())); } break; @@ -12567,7 +12608,7 @@ class basic_json { // op must be "add", "remove", "replace", "move", "copy", or // "test" - JSON_THROW(parse_error(105, 0, "operation value '" + op + "' is invalid")); + JSON_THROW(parse_error::create(105, 0, "operation value '" + op + "' is invalid")); } } } diff --git a/test/src/unit-noexcept.cpp b/test/src/unit-noexcept.cpp index 898e7796..ed362f65 100644 --- a/test/src/unit-noexcept.cpp +++ b/test/src/unit-noexcept.cpp @@ -57,3 +57,11 @@ static_assert(noexcept(json(pod{})), ""); static_assert(noexcept(j.get()), ""); static_assert(not noexcept(j.get()), ""); static_assert(noexcept(json(pod{})), ""); + +// for ERR60-CPP (https://github.com/nlohmann/json/issues/531) +static_assert(std::is_nothrow_copy_constructible::value, "json::exception must be nothrow copy constructible"); +static_assert(std::is_nothrow_copy_constructible::value, "json::parse_error must be nothrow copy constructible"); +static_assert(std::is_nothrow_copy_constructible::value, "json::invalid_iterator must be nothrow copy constructible"); +static_assert(std::is_nothrow_copy_constructible::value, "json::type_error must be nothrow copy constructible"); +static_assert(std::is_nothrow_copy_constructible::value, "json::out_of_range must be nothrow copy constructible"); +static_assert(std::is_nothrow_copy_constructible::value, "json::other_error must be nothrow copy constructible"); From c333679a9653119ddf19613d3fdc2cb1d42e2a26 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sat, 25 Mar 2017 17:25:39 +0100 Subject: [PATCH 104/530] :hammer: small refactoring The solution with a std::runtime_error member is more elegant. It allows to have std::exception as base class again. However, I still have no idea why GCC thinks the copy constructor may throw... --- src/json.hpp | 9 ++++++--- src/json.hpp.re2c | 9 ++++++--- 2 files changed, 12 insertions(+), 6 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index bdada276..5ee3edad 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -123,13 +123,13 @@ Extension of std::exception objects with a member @a id for exception ids. @since version 3.0.0 */ -class exception : public std::runtime_error +class exception : public std::exception { public: /// returns the explanatory string virtual const char* what() const noexcept override { - return std::runtime_error::what(); + return m.what(); } /// the id of the exception @@ -137,13 +137,16 @@ class exception : public std::runtime_error protected: exception(int id_, const char* what_arg) - : std::runtime_error(what_arg), id(id_) + : id(id_), m(what_arg) {} static std::string name(const std::string& ename, int id_) { return "[json.exception." + ename + "." + std::to_string(id_) + "] "; } + + private: + std::runtime_error m; }; /*! diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index d5b12d2c..a51f77e7 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -123,13 +123,13 @@ Extension of std::exception objects with a member @a id for exception ids. @since version 3.0.0 */ -class exception : public std::runtime_error +class exception : public std::exception { public: /// returns the explanatory string virtual const char* what() const noexcept override { - return std::runtime_error::what(); + return m.what(); } /// the id of the exception @@ -137,13 +137,16 @@ class exception : public std::runtime_error protected: exception(int id_, const char* what_arg) - : std::runtime_error(what_arg), id(id_) + : id(id_), m(what_arg) {} static std::string name(const std::string& ename, int id_) { return "[json.exception." + ename + "." + std::to_string(id_) + "] "; } + + private: + std::runtime_error m; }; /*! From cf7786887c7ca80d86112f6b9f0aeba8cb1f2f9c Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sat, 25 Mar 2017 17:35:57 +0100 Subject: [PATCH 105/530] :hammer: fixed check for is_nothrow_copy_constructible We now only demand our exceptions to be is_nothrow_copy_constructible if std::runtime_exception is. --- test/src/unit-noexcept.cpp | 23 ++++++++++++++++------- 1 file changed, 16 insertions(+), 7 deletions(-) diff --git a/test/src/unit-noexcept.cpp b/test/src/unit-noexcept.cpp index ed362f65..f2fbc8e6 100644 --- a/test/src/unit-noexcept.cpp +++ b/test/src/unit-noexcept.cpp @@ -58,10 +58,19 @@ static_assert(noexcept(j.get()), ""); static_assert(not noexcept(j.get()), ""); static_assert(noexcept(json(pod{})), ""); -// for ERR60-CPP (https://github.com/nlohmann/json/issues/531) -static_assert(std::is_nothrow_copy_constructible::value, "json::exception must be nothrow copy constructible"); -static_assert(std::is_nothrow_copy_constructible::value, "json::parse_error must be nothrow copy constructible"); -static_assert(std::is_nothrow_copy_constructible::value, "json::invalid_iterator must be nothrow copy constructible"); -static_assert(std::is_nothrow_copy_constructible::value, "json::type_error must be nothrow copy constructible"); -static_assert(std::is_nothrow_copy_constructible::value, "json::out_of_range must be nothrow copy constructible"); -static_assert(std::is_nothrow_copy_constructible::value, "json::other_error must be nothrow copy constructible"); +TEST_CASE("runtime checks") +{ + SECTION("nothrow-copy-constructible exceptions") + { + // for ERR60-CPP (https://github.com/nlohmann/json/issues/531) + if (std::is_nothrow_copy_constructible::value) + { + CHECK(std::is_nothrow_copy_constructible::value); + CHECK(std::is_nothrow_copy_constructible::value); + CHECK(std::is_nothrow_copy_constructible::value); + CHECK(std::is_nothrow_copy_constructible::value); + CHECK(std::is_nothrow_copy_constructible::value); + CHECK(std::is_nothrow_copy_constructible::value); + } + } +} From 53b501a785f94250fe7b359b8ba6288d495aad5d Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sat, 25 Mar 2017 23:31:03 +0100 Subject: [PATCH 106/530] :hammer: cleanup --- src/json.hpp | 64 +++++++++++++++++++------------------- src/json.hpp.re2c | 62 ++++++++++++++++++------------------ test/src/unit-noexcept.cpp | 21 +++++++------ 3 files changed, 74 insertions(+), 73 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 5ee3edad..522c601d 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -116,7 +116,7 @@ namespace detail Extension of std::exception objects with a member @a id for exception ids. -@note To have nothrow-copy-constructible exceptions, we inherit from +@note To have nothrow-copy-constructible exceptions, we internally use std::runtime_error which can cope with arbitrary-length error messages. Intermediate strings are built with static functions and then passed to the actual constructor. @@ -136,16 +136,17 @@ class exception : public std::exception const int id; protected: - exception(int id_, const char* what_arg) + exception(int id_, const char* what_arg) noexcept(noexcept(std::runtime_error(what_arg))) : id(id_), m(what_arg) {} - static std::string name(const std::string& ename, int id_) + static std::string name(const std::string& ename, int id) { - return "[json.exception." + ename + "." + std::to_string(id_) + "] "; + return "[json.exception." + ename + "." + std::to_string(id) + "] "; } private: + /// an exception object as storage for error messages std::runtime_error m; }; @@ -184,23 +185,23 @@ json.exception.parse_error.113 | parse error at 2: expected a CBOR string; last @since version 3.0.0 */ -class parse_error : public exception +class parse_error : private exception { public: /*! @brief create a parse error exception - @param[in] id_ the id of the exception + @param[in] id the id of the exception @param[in] byte_ the byte index where the error occured (or 0 if the position cannot be determined) @param[in] what_arg the explanatory string @return parse_error object */ - static parse_error create(int id_, size_t byte_, const std::string& what_arg) + static parse_error create(int id, size_t byte_, const std::string& what_arg) { - std::string w = exception::name("parse_error", id_) + "parse error" + + std::string w = exception::name("parse_error", id) + "parse error" + (byte_ != 0 ? (" at " + std::to_string(byte_)) : "") + ": " + what_arg; - return parse_error(id_, byte_, w.c_str()); + return parse_error(id, byte_, w.c_str()); } /*! @@ -216,9 +217,8 @@ class parse_error : public exception const size_t byte; private: - parse_error(int id_, size_t byte_, const char* what_arg) - : exception(id_, what_arg), - byte(byte_) + parse_error(int id, size_t byte_, const char* what_arg) noexcept(noexcept(std::runtime_error(what_arg))) + : exception(id, what_arg), byte(byte_) {} }; @@ -249,15 +249,15 @@ json.exception.invalid_iterator.214 | cannot get value | Cannot get value for it class invalid_iterator : public exception { public: - static invalid_iterator create(int id_, const std::string& what_arg) + static invalid_iterator create(int id, const std::string& what_arg) { - std::string w = exception::name("invalid_iterator", id_) + what_arg; - return invalid_iterator(id_, w.c_str()); + std::string w = exception::name("invalid_iterator", id) + what_arg; + return invalid_iterator(id, w.c_str()); } private: - invalid_iterator(int id_, const char* what_arg) - : exception(id_, what_arg) + invalid_iterator(int id, const char* what_arg) noexcept(noexcept(std::runtime_error(what_arg))) + : exception(id, what_arg) {} }; @@ -288,15 +288,15 @@ json.exception.type_error.315 | values in object must be primitive | The @ref un class type_error : public exception { public: - static type_error create(int id_, const std::string& what_arg) + static type_error create(int id, const std::string& what_arg) { - std::string w = exception::name("type_error", id_) + what_arg; - return type_error(id_, w.c_str()); + std::string w = exception::name("type_error", id) + what_arg; + return type_error(id, w.c_str()); } private: - type_error(int id_, const char* what_arg) - : exception(id_, what_arg) + type_error(int id, const char* what_arg) noexcept(noexcept(std::runtime_error(what_arg))) + : exception(id, what_arg) {} }; @@ -319,15 +319,15 @@ json.exception.out_of_range.406 | number overflow parsing '10E1000' | A parsed n class out_of_range : public exception { public: - static out_of_range create(int id_, const std::string& what_arg) + static out_of_range create(int id, const std::string& what_arg) { - std::string w = exception::name("out_of_range", id_) + what_arg; - return out_of_range(id_, w.c_str()); + std::string w = exception::name("out_of_range", id) + what_arg; + return out_of_range(id, w.c_str()); } private: - out_of_range(int id_, const char* what_arg) - : exception(id_, what_arg) + out_of_range(int id, const char* what_arg) noexcept(noexcept(std::runtime_error(what_arg))) + : exception(id, what_arg) {} }; @@ -345,15 +345,15 @@ json.exception.other_error.501 | unsuccessful: {"op":"test","path":"/baz", "valu class other_error : public exception { public: - static other_error create(int id_, const std::string& what_arg) + static other_error create(int id, const std::string& what_arg) { - std::string w = exception::name("other_error", id_) + what_arg; - return other_error(id_, w.c_str()); + std::string w = exception::name("other_error", id) + what_arg; + return other_error(id, w.c_str()); } private: - other_error(int id_, const char* what_arg) - : exception(id_, what_arg) + other_error(int id, const char* what_arg) noexcept(noexcept(std::runtime_error(what_arg))) + : exception(id, what_arg) {} }; diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index a51f77e7..c7243ca4 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -116,7 +116,7 @@ namespace detail Extension of std::exception objects with a member @a id for exception ids. -@note To have nothrow-copy-constructible exceptions, we inherit from +@note To have nothrow-copy-constructible exceptions, we internally use std::runtime_error which can cope with arbitrary-length error messages. Intermediate strings are built with static functions and then passed to the actual constructor. @@ -136,16 +136,17 @@ class exception : public std::exception const int id; protected: - exception(int id_, const char* what_arg) + exception(int id_, const char* what_arg) noexcept(noexcept(std::runtime_error(what_arg))) : id(id_), m(what_arg) {} - static std::string name(const std::string& ename, int id_) + static std::string name(const std::string& ename, int id) { - return "[json.exception." + ename + "." + std::to_string(id_) + "] "; + return "[json.exception." + ename + "." + std::to_string(id) + "] "; } private: + /// an exception object as storage for error messages std::runtime_error m; }; @@ -189,18 +190,18 @@ class parse_error : public exception public: /*! @brief create a parse error exception - @param[in] id_ the id of the exception + @param[in] id the id of the exception @param[in] byte_ the byte index where the error occured (or 0 if the position cannot be determined) @param[in] what_arg the explanatory string @return parse_error object */ - static parse_error create(int id_, size_t byte_, const std::string& what_arg) + static parse_error create(int id, size_t byte_, const std::string& what_arg) { - std::string w = exception::name("parse_error", id_) + "parse error" + + std::string w = exception::name("parse_error", id) + "parse error" + (byte_ != 0 ? (" at " + std::to_string(byte_)) : "") + ": " + what_arg; - return parse_error(id_, byte_, w.c_str()); + return parse_error(id, byte_, w.c_str()); } /*! @@ -216,9 +217,8 @@ class parse_error : public exception const size_t byte; private: - parse_error(int id_, size_t byte_, const char* what_arg) - : exception(id_, what_arg), - byte(byte_) + parse_error(int id, size_t byte_, const char* what_arg) noexcept(noexcept(std::runtime_error(what_arg))) + : exception(id, what_arg), byte(byte_) {} }; @@ -249,15 +249,15 @@ json.exception.invalid_iterator.214 | cannot get value | Cannot get value for it class invalid_iterator : public exception { public: - static invalid_iterator create(int id_, const std::string& what_arg) + static invalid_iterator create(int id, const std::string& what_arg) { - std::string w = exception::name("invalid_iterator", id_) + what_arg; - return invalid_iterator(id_, w.c_str()); + std::string w = exception::name("invalid_iterator", id) + what_arg; + return invalid_iterator(id, w.c_str()); } private: - invalid_iterator(int id_, const char* what_arg) - : exception(id_, what_arg) + invalid_iterator(int id, const char* what_arg) noexcept(noexcept(std::runtime_error(what_arg))) + : exception(id, what_arg) {} }; @@ -288,15 +288,15 @@ json.exception.type_error.315 | values in object must be primitive | The @ref un class type_error : public exception { public: - static type_error create(int id_, const std::string& what_arg) + static type_error create(int id, const std::string& what_arg) { - std::string w = exception::name("type_error", id_) + what_arg; - return type_error(id_, w.c_str()); + std::string w = exception::name("type_error", id) + what_arg; + return type_error(id, w.c_str()); } private: - type_error(int id_, const char* what_arg) - : exception(id_, what_arg) + type_error(int id, const char* what_arg) noexcept(noexcept(std::runtime_error(what_arg))) + : exception(id, what_arg) {} }; @@ -319,15 +319,15 @@ json.exception.out_of_range.406 | number overflow parsing '10E1000' | A parsed n class out_of_range : public exception { public: - static out_of_range create(int id_, const std::string& what_arg) + static out_of_range create(int id, const std::string& what_arg) { - std::string w = exception::name("out_of_range", id_) + what_arg; - return out_of_range(id_, w.c_str()); + std::string w = exception::name("out_of_range", id) + what_arg; + return out_of_range(id, w.c_str()); } private: - out_of_range(int id_, const char* what_arg) - : exception(id_, what_arg) + out_of_range(int id, const char* what_arg) noexcept(noexcept(std::runtime_error(what_arg))) + : exception(id, what_arg) {} }; @@ -345,15 +345,15 @@ json.exception.other_error.501 | unsuccessful: {"op":"test","path":"/baz", "valu class other_error : public exception { public: - static other_error create(int id_, const std::string& what_arg) + static other_error create(int id, const std::string& what_arg) { - std::string w = exception::name("other_error", id_) + what_arg; - return other_error(id_, w.c_str()); + std::string w = exception::name("other_error", id) + what_arg; + return other_error(id, w.c_str()); } private: - other_error(int id_, const char* what_arg) - : exception(id_, what_arg) + other_error(int id, const char* what_arg) noexcept(noexcept(std::runtime_error(what_arg))) + : exception(id, what_arg) {} }; diff --git a/test/src/unit-noexcept.cpp b/test/src/unit-noexcept.cpp index f2fbc8e6..8269574e 100644 --- a/test/src/unit-noexcept.cpp +++ b/test/src/unit-noexcept.cpp @@ -62,15 +62,16 @@ TEST_CASE("runtime checks") { SECTION("nothrow-copy-constructible exceptions") { - // for ERR60-CPP (https://github.com/nlohmann/json/issues/531) - if (std::is_nothrow_copy_constructible::value) - { - CHECK(std::is_nothrow_copy_constructible::value); - CHECK(std::is_nothrow_copy_constructible::value); - CHECK(std::is_nothrow_copy_constructible::value); - CHECK(std::is_nothrow_copy_constructible::value); - CHECK(std::is_nothrow_copy_constructible::value); - CHECK(std::is_nothrow_copy_constructible::value); - } + // for ERR60-CPP (https://github.com/nlohmann/json/issues/531): + // Exceptions should be nothrow-copy-constructible. However, compilers + // treat std::runtime_exception differently in this regard. Therefore, + // we can only demand nothrow-copy-constructibility for our exceptions + // if std::runtime_exception is. + CHECK(std::is_nothrow_copy_constructible::value == std::is_nothrow_copy_constructible::value); + CHECK(std::is_nothrow_copy_constructible::value == std::is_nothrow_copy_constructible::value); + CHECK(std::is_nothrow_copy_constructible::value == std::is_nothrow_copy_constructible::value); + CHECK(std::is_nothrow_copy_constructible::value == std::is_nothrow_copy_constructible::value); + CHECK(std::is_nothrow_copy_constructible::value == std::is_nothrow_copy_constructible::value); + CHECK(std::is_nothrow_copy_constructible::value == std::is_nothrow_copy_constructible::value); } } From 333619430601807e2807b5d63eae831f435e7d00 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 26 Mar 2017 12:23:10 +0200 Subject: [PATCH 107/530] :hammer: reverted changes that led to Travis failures --- src/json.hpp | 14 +++++++------- src/json.hpp.re2c | 12 ++++++------ 2 files changed, 13 insertions(+), 13 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 522c601d..cae427ef 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -136,7 +136,7 @@ class exception : public std::exception const int id; protected: - exception(int id_, const char* what_arg) noexcept(noexcept(std::runtime_error(what_arg))) + exception(int id_, const char* what_arg) : id(id_), m(what_arg) {} @@ -185,7 +185,7 @@ json.exception.parse_error.113 | parse error at 2: expected a CBOR string; last @since version 3.0.0 */ -class parse_error : private exception +class parse_error : public exception { public: /*! @@ -217,7 +217,7 @@ class parse_error : private exception const size_t byte; private: - parse_error(int id, size_t byte_, const char* what_arg) noexcept(noexcept(std::runtime_error(what_arg))) + parse_error(int id, size_t byte_, const char* what_arg) : exception(id, what_arg), byte(byte_) {} }; @@ -256,7 +256,7 @@ class invalid_iterator : public exception } private: - invalid_iterator(int id, const char* what_arg) noexcept(noexcept(std::runtime_error(what_arg))) + invalid_iterator(int id, const char* what_arg) : exception(id, what_arg) {} }; @@ -295,7 +295,7 @@ class type_error : public exception } private: - type_error(int id, const char* what_arg) noexcept(noexcept(std::runtime_error(what_arg))) + type_error(int id, const char* what_arg) : exception(id, what_arg) {} }; @@ -326,7 +326,7 @@ class out_of_range : public exception } private: - out_of_range(int id, const char* what_arg) noexcept(noexcept(std::runtime_error(what_arg))) + out_of_range(int id, const char* what_arg) : exception(id, what_arg) {} }; @@ -352,7 +352,7 @@ class other_error : public exception } private: - other_error(int id, const char* what_arg) noexcept(noexcept(std::runtime_error(what_arg))) + other_error(int id, const char* what_arg) : exception(id, what_arg) {} }; diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index c7243ca4..3aabece5 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -136,7 +136,7 @@ class exception : public std::exception const int id; protected: - exception(int id_, const char* what_arg) noexcept(noexcept(std::runtime_error(what_arg))) + exception(int id_, const char* what_arg) : id(id_), m(what_arg) {} @@ -217,7 +217,7 @@ class parse_error : public exception const size_t byte; private: - parse_error(int id, size_t byte_, const char* what_arg) noexcept(noexcept(std::runtime_error(what_arg))) + parse_error(int id, size_t byte_, const char* what_arg) : exception(id, what_arg), byte(byte_) {} }; @@ -256,7 +256,7 @@ class invalid_iterator : public exception } private: - invalid_iterator(int id, const char* what_arg) noexcept(noexcept(std::runtime_error(what_arg))) + invalid_iterator(int id, const char* what_arg) : exception(id, what_arg) {} }; @@ -295,7 +295,7 @@ class type_error : public exception } private: - type_error(int id, const char* what_arg) noexcept(noexcept(std::runtime_error(what_arg))) + type_error(int id, const char* what_arg) : exception(id, what_arg) {} }; @@ -326,7 +326,7 @@ class out_of_range : public exception } private: - out_of_range(int id, const char* what_arg) noexcept(noexcept(std::runtime_error(what_arg))) + out_of_range(int id, const char* what_arg) : exception(id, what_arg) {} }; @@ -352,7 +352,7 @@ class other_error : public exception } private: - other_error(int id, const char* what_arg) noexcept(noexcept(std::runtime_error(what_arg))) + other_error(int id, const char* what_arg) : exception(id, what_arg) {} }; From 99e0d8b339fca017c0f3b7a55d85a087bcaace0a Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 26 Mar 2017 15:29:08 +0200 Subject: [PATCH 108/530] :hammer: fixed \uxxxx parsing --- src/json.hpp | 37 +++++------ test/src/unit-class_parser.cpp | 115 +++++++++++++++++++++------------ 2 files changed, 90 insertions(+), 62 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 3caa8a6c..6cbcdf28 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -42,6 +42,7 @@ SOFTWARE. #include // forward_list #include // function, hash, less #include // initializer_list +#include // hex #include // istream, ostream #include // advance, begin, back_inserter, bidirectional_iterator_tag, distance, end, inserter, iterator, iterator_traits, next, random_access_iterator_tag, reverse_iterator #include // numeric_limits @@ -10542,36 +10543,30 @@ class basic_json // must be called after \u was read; returns following xxxx as hex or -1 when error int get_codepoint() { - // read xxxx of \uxxxx - std::vector buffer(5, '\0'); + // a mapping to discover hex numbers + static int8_t ascii_to_hex[256] = {-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, -1, -1, -1, -1, -1, -1, -1, 10, 11, 12, 13, 14, 15, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, 10, 11, 12, 13, 14, 15, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1}; + + int codepoint = 0; + for (size_t i = 0; i < 4; ++i) { - get(); - if (JSON_UNLIKELY(current != std::char_traits::eof())) + const int8_t digit = ascii_to_hex[static_cast(get())]; + if (JSON_UNLIKELY(digit == -1)) { - buffer[i] = static_cast(current); + return -1; } else { - // error message will be created by caller - return -1; + codepoint += digit; + } + + if (i != 3) + { + codepoint <<= 4; } } - char* endptr; - errno = 0; - - const int codepoint = static_cast(std::strtoul(buffer.data(), &endptr, 16)); - - if (JSON_LIKELY(errno == 0 and endptr == buffer.data() + 4)) - { - return codepoint; - } - else - { - // conversion incomplete or failure - return -1; - } + return codepoint; } token_type scan_string() diff --git a/test/src/unit-class_parser.cpp b/test/src/unit-class_parser.cpp index fe2a8fbe..01812259 100644 --- a/test/src/unit-class_parser.cpp +++ b/test/src/unit-class_parser.cpp @@ -91,18 +91,18 @@ TEST_CASE("parser class") // error: tab in string CHECK_THROWS_AS(json::parser("\"\t\"").parse(), json::parse_error); CHECK_THROWS_WITH(json::parser("\"\t\"").parse(), - "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '\"'"); + "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: control characters (U+0000 through U+001f) must be escaped; last read '\"'"); // error: newline in string CHECK_THROWS_AS(json::parser("\"\n\"").parse(), json::parse_error); CHECK_THROWS_AS(json::parser("\"\r\"").parse(), json::parse_error); CHECK_THROWS_WITH(json::parser("\"\n\"").parse(), - "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '\"'"); + "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: control characters (U+0000 through U+001f) must be escaped; last read '\"'"); CHECK_THROWS_WITH(json::parser("\"\r\"").parse(), - "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '\"'"); + "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: control characters (U+0000 through U+001f) must be escaped; last read '\"'"); // error: backspace in string CHECK_THROWS_AS(json::parser("\"\b\"").parse(), json::parse_error); CHECK_THROWS_WITH(json::parser("\"\b\"").parse(), - "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '\"'"); + "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: control characters (U+0000 through U+001f) must be escaped; last read '\"'"); // improve code coverage CHECK_THROWS_AS(json::parser("\uFF01").parse(), json::parse_error); CHECK_THROWS_AS(json::parser("[-4:1,]").parse(), json::parse_error); @@ -393,34 +393,37 @@ TEST_CASE("parser class") CHECK_THROWS_AS(json::parser("n").parse(), json::parse_error); CHECK_THROWS_AS(json::parser("nu").parse(), json::parse_error); CHECK_THROWS_AS(json::parser("nul").parse(), json::parse_error); - CHECK_THROWS_WITH(json::parser("n").parse(), "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected 'n'"); + CHECK_THROWS_WITH(json::parser("n").parse(), + "[json.exception.parse_error.101] parse error at 2: syntax error - invalid literal; expected 'null'; last read 'n'"); CHECK_THROWS_WITH(json::parser("nu").parse(), - "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected 'n'"); + "[json.exception.parse_error.101] parse error at 3: syntax error - invalid literal; expected 'null'; last read 'nu'"); CHECK_THROWS_WITH(json::parser("nul").parse(), - "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected 'n'"); + "[json.exception.parse_error.101] parse error at 4: syntax error - invalid literal; expected 'null'; last read 'nul'"); // unexpected end of true CHECK_THROWS_AS(json::parser("t").parse(), json::parse_error); CHECK_THROWS_AS(json::parser("tr").parse(), json::parse_error); CHECK_THROWS_AS(json::parser("tru").parse(), json::parse_error); - CHECK_THROWS_WITH(json::parser("t").parse(), "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected 't'"); + CHECK_THROWS_WITH(json::parser("t").parse(), + "[json.exception.parse_error.101] parse error at 2: syntax error - invalid literal; expected 'true'; last read 't'"); CHECK_THROWS_WITH(json::parser("tr").parse(), - "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected 't'"); + "[json.exception.parse_error.101] parse error at 3: syntax error - invalid literal; expected 'true'; last read 'tr'"); CHECK_THROWS_WITH(json::parser("tru").parse(), - "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected 't'"); + "[json.exception.parse_error.101] parse error at 4: syntax error - invalid literal; expected 'true'; last read 'tru'"); // unexpected end of false CHECK_THROWS_AS(json::parser("f").parse(), json::parse_error); CHECK_THROWS_AS(json::parser("fa").parse(), json::parse_error); CHECK_THROWS_AS(json::parser("fal").parse(), json::parse_error); CHECK_THROWS_AS(json::parser("fals").parse(), json::parse_error); - CHECK_THROWS_WITH(json::parser("f").parse(), "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected 'f'"); + CHECK_THROWS_WITH(json::parser("f").parse(), + "[json.exception.parse_error.101] parse error at 2: syntax error - invalid literal; expected 'false'; last read 'f'"); CHECK_THROWS_WITH(json::parser("fa").parse(), - "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected 'f'"); + "[json.exception.parse_error.101] parse error at 3: syntax error - invalid literal; expected 'false'; last read 'fa'"); CHECK_THROWS_WITH(json::parser("fal").parse(), - "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected 'f'"); + "[json.exception.parse_error.101] parse error at 4: syntax error - invalid literal; expected 'false'; last read 'fal'"); CHECK_THROWS_WITH(json::parser("fals").parse(), - "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected 'f'"); + "[json.exception.parse_error.101] parse error at 5: syntax error - invalid literal; expected 'false'; last read 'fals'"); // missing/unexpected end of array CHECK_THROWS_AS(json::parser("[").parse(), json::parse_error); @@ -471,25 +474,25 @@ TEST_CASE("parser class") CHECK_THROWS_AS(json::parser("\"\\u01").parse(), json::parse_error); CHECK_THROWS_AS(json::parser("\"\\u012").parse(), json::parse_error); CHECK_THROWS_WITH(json::parser("\"").parse(), - "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '\"'"); + "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: missing closing quote; last read '\"'"); CHECK_THROWS_WITH(json::parser("\"\\\"").parse(), - "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '\"'"); + "[json.exception.parse_error.101] parse error at 4: syntax error - invalid string: missing closing quote; last read '\"\\\"'"); CHECK_THROWS_WITH(json::parser("\"\\u\"").parse(), - "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '\"'"); + "[json.exception.parse_error.101] parse error at 4: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read '\"\\u\"'"); CHECK_THROWS_WITH(json::parser("\"\\u0\"").parse(), - "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '\"'"); + "[json.exception.parse_error.101] parse error at 5: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read '\"\\u0\"'"); CHECK_THROWS_WITH(json::parser("\"\\u01\"").parse(), - "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '\"'"); + "[json.exception.parse_error.101] parse error at 6: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read '\"\\u01\"'"); CHECK_THROWS_WITH(json::parser("\"\\u012\"").parse(), - "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '\"'"); + "[json.exception.parse_error.101] parse error at 7: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read '\"\\u012\"'"); CHECK_THROWS_WITH(json::parser("\"\\u").parse(), - "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '\"'"); + "[json.exception.parse_error.101] parse error at 4: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read '\"\\u'"); CHECK_THROWS_WITH(json::parser("\"\\u0").parse(), - "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '\"'"); + "[json.exception.parse_error.101] parse error at 5: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read '\"\\u0'"); CHECK_THROWS_WITH(json::parser("\"\\u01").parse(), - "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '\"'"); + "[json.exception.parse_error.101] parse error at 6: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read '\"\\u01'"); CHECK_THROWS_WITH(json::parser("\"\\u012").parse(), - "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '\"'"); + "[json.exception.parse_error.101] parse error at 7: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read '\"\\u012'"); // invalid escapes for (int c = 1; c < 128; ++c) @@ -522,8 +525,12 @@ TEST_CASE("parser class") default: { CHECK_THROWS_AS(json::parser(s.c_str()).parse(), json::parse_error); - CHECK_THROWS_WITH(json::parser(s.c_str()).parse(), - "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '\"'"); + // only check error message if c is not a control character + if (c > 0x1f) + { + CHECK_THROWS_WITH(json::parser(s.c_str()).parse(), + "[json.exception.parse_error.101] parse error at 3: syntax error - invalid string: forbidden character after backspace; last read '\"\\" + std::string(1, c) + "'"); + } break; } } @@ -581,26 +588,52 @@ TEST_CASE("parser class") if (valid(c)) { + CAPTURE(s1); CHECK_NOTHROW(json::parser(s1.c_str()).parse()); + CAPTURE(s2); CHECK_NOTHROW(json::parser(s2.c_str()).parse()); + CAPTURE(s3); CHECK_NOTHROW(json::parser(s3.c_str()).parse()); + CAPTURE(s4); CHECK_NOTHROW(json::parser(s4.c_str()).parse()); } else { + CAPTURE(s1); CHECK_THROWS_AS(json::parser(s1.c_str()).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(s2.c_str()).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(s3.c_str()).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(s4.c_str()).parse(), json::parse_error); + // only check error message if c is not a control character + if (c > 0x1f) + { + CHECK_THROWS_WITH(json::parser(s1.c_str()).parse(), + "[json.exception.parse_error.101] parse error at 7: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read '" + s1.substr(0, 7) + "'"); + } - CHECK_THROWS_WITH(json::parser(s1.c_str()).parse(), - "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '\"'"); - CHECK_THROWS_WITH(json::parser(s2.c_str()).parse(), - "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '\"'"); - CHECK_THROWS_WITH(json::parser(s3.c_str()).parse(), - "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '\"'"); - CHECK_THROWS_WITH(json::parser(s4.c_str()).parse(), - "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '\"'"); + CAPTURE(s2); + CHECK_THROWS_AS(json::parser(s2.c_str()).parse(), json::parse_error); + // only check error message if c is not a control character + if (c > 0x1f) + { + CHECK_THROWS_WITH(json::parser(s2.c_str()).parse(), + "[json.exception.parse_error.101] parse error at 6: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read '" + s2.substr(0, 6) + "'"); + } + + CAPTURE(s3); + CHECK_THROWS_AS(json::parser(s3.c_str()).parse(), json::parse_error); + // only check error message if c is not a control character + if (c > 0x1f) + { + CHECK_THROWS_WITH(json::parser(s3.c_str()).parse(), + "[json.exception.parse_error.101] parse error at 5: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read '" + s3.substr(0, 5) + "'"); + } + + CAPTURE(s4); + CHECK_THROWS_AS(json::parser(s4.c_str()).parse(), json::parse_error); + // only check error message if c is not a control character + if (c > 0x1f) + { + CHECK_THROWS_WITH(json::parser(s4.c_str()).parse(), + "[json.exception.parse_error.101] parse error at 4: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read '" + s4.substr(0, 4) + "'"); + } } } } @@ -608,17 +641,17 @@ TEST_CASE("parser class") // missing part of a surrogate pair CHECK_THROWS_AS(json::parse("\"\\uD80C\""), json::parse_error); CHECK_THROWS_WITH(json::parse("\"\\uD80C\""), - "[json.exception.parse_error.102] parse error at 8: missing low surrogate"); + "[json.exception.parse_error.101] parse error at 8: syntax error - invalid string: missing low surrogate; last read '\"\\uD80C\"'"); // invalid surrogate pair CHECK_THROWS_AS(json::parse("\"\\uD80C\\uD80C\""), json::parse_error); CHECK_THROWS_AS(json::parse("\"\\uD80C\\u0000\""), json::parse_error); CHECK_THROWS_AS(json::parse("\"\\uD80C\\uFFFF\""), json::parse_error); CHECK_THROWS_WITH(json::parse("\"\\uD80C\\uD80C\""), - "[json.exception.parse_error.102] parse error at 14: missing or wrong low surrogate"); + "[json.exception.parse_error.101] parse error at 13: syntax error - invalid string: invalid low surrogate; last read '\"\\uD80C\\uD80C'"); CHECK_THROWS_WITH(json::parse("\"\\uD80C\\u0000\""), - "[json.exception.parse_error.102] parse error at 14: missing or wrong low surrogate"); + "[json.exception.parse_error.101] parse error at 13: syntax error - invalid string: invalid low surrogate; last read '\"\\uD80C\\u0000'"); CHECK_THROWS_WITH(json::parse("\"\\uD80C\\uFFFF\""), - "[json.exception.parse_error.102] parse error at 14: missing or wrong low surrogate"); + "[json.exception.parse_error.101] parse error at 13: syntax error - invalid string: invalid low surrogate; last read '\"\\uD80C\\uFFFF'"); } SECTION("tests found by mutate++") From 3b069b4b4c135be91a7885d6795ca1e7fcff2eb7 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 26 Mar 2017 16:19:07 +0200 Subject: [PATCH 109/530] :hammer: adjusted more expected exceptions --- test/src/unit-class_parser.cpp | 44 +++++++++++++++++----------------- 1 file changed, 22 insertions(+), 22 deletions(-) diff --git a/test/src/unit-class_parser.cpp b/test/src/unit-class_parser.cpp index 01812259..36451042 100644 --- a/test/src/unit-class_parser.cpp +++ b/test/src/unit-class_parser.cpp @@ -310,21 +310,21 @@ TEST_CASE("parser class") CHECK_THROWS_WITH(json::parser("-01").parse(), "[json.exception.parse_error.101] parse error at 3: syntax error - unexpected '-01'"); CHECK_THROWS_WITH(json::parser("--1").parse(), - "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '-'"); + "[json.exception.parse_error.101] parse error at 2: syntax error - invalid number; expected digit after '-'; last read '--'"); CHECK_THROWS_WITH(json::parser("1.").parse(), - "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected '.'; expected end of input"); + "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected digit after '.'; last read '1.'"); CHECK_THROWS_WITH(json::parser("1E").parse(), - "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected 'E'; expected end of input"); + "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected '+', '-', or digit after exponent; last read '1E'"); CHECK_THROWS_WITH(json::parser("1E-").parse(), - "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected 'E'; expected end of input"); + "[json.exception.parse_error.101] parse error at 4: syntax error - invalid number; expected digit after exponent sign; last read '1E-'"); CHECK_THROWS_WITH(json::parser("1.E1").parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected '.'; expected end of input"); CHECK_THROWS_WITH(json::parser("-1E").parse(), - "[json.exception.parse_error.101] parse error at 3: syntax error - unexpected 'E'; expected end of input"); + "[json.exception.parse_error.101] parse error at 4: syntax error - invalid number; expected '+', '-', or digit after exponent; last read '-1E'"); CHECK_THROWS_WITH(json::parser("-0E#").parse(), - "[json.exception.parse_error.101] parse error at 3: syntax error - unexpected 'E'; expected end of input"); + "[json.exception.parse_error.101] parse error at 4: syntax error - invalid number; expected '+', '-', or digit after exponent; last read '-0E#'"); CHECK_THROWS_WITH(json::parser("-0E-#").parse(), - "[json.exception.parse_error.101] parse error at 3: syntax error - unexpected 'E'; expected end of input"); + "[json.exception.parse_error.101] parse error at 5: syntax error - invalid number; expected digit after exponent sign; last read '-0E-#'"); CHECK_THROWS_WITH(json::parser("-0#").parse(), "[json.exception.parse_error.101] parse error at 3: syntax error - unexpected '#'; expected end of input"); CHECK_THROWS_WITH(json::parser("-0.0:").parse(), @@ -336,7 +336,7 @@ TEST_CASE("parser class") CHECK_THROWS_WITH(json::parser("-0e0-:").parse(), "[json.exception.parse_error.101] parse error at 5: syntax error - unexpected '-'; expected end of input"); CHECK_THROWS_WITH(json::parser("-0e-:").parse(), - "[json.exception.parse_error.101] parse error at 3: syntax error - unexpected 'e'; expected end of input"); + "[json.exception.parse_error.101] parse error at 5: syntax error - invalid number; expected digit after exponent sign; last read '-0e-:'"); CHECK_THROWS_WITH(json::parser("-0f").parse(), "[json.exception.parse_error.101] parse error at 3: syntax error - unexpected 'f'; expected end of input"); } @@ -361,33 +361,33 @@ TEST_CASE("parser class") CHECK_THROWS_AS(json::parser("1E/").parse(), json::parse_error); CHECK_THROWS_AS(json::parser("1E:").parse(), json::parse_error); CHECK_THROWS_WITH(json::parser("0.").parse(), - "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected '.'; expected end of input"); + "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected digit after '.'; last read '0.'"); CHECK_THROWS_WITH(json::parser("-").parse(), - "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '-'"); + "[json.exception.parse_error.101] parse error at 2: syntax error - invalid number; expected digit after '-'; last read '-'"); CHECK_THROWS_WITH(json::parser("--").parse(), - "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '-'"); + "[json.exception.parse_error.101] parse error at 2: syntax error - invalid number; expected digit after '-'; last read '--'"); CHECK_THROWS_WITH(json::parser("-0.").parse(), - "[json.exception.parse_error.101] parse error at 3: syntax error - unexpected '.'; expected end of input"); + "[json.exception.parse_error.101] parse error at 4: syntax error - invalid number; expected digit after '.'; last read '-0.'"); CHECK_THROWS_WITH(json::parser("-.").parse(), - "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '-'"); + "[json.exception.parse_error.101] parse error at 2: syntax error - invalid number; expected digit after '-'; last read '-.'"); CHECK_THROWS_WITH(json::parser("-:").parse(), - "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '-'"); + "[json.exception.parse_error.101] parse error at 2: syntax error - invalid number; expected digit after '-'; last read '-:'"); CHECK_THROWS_WITH(json::parser("0.:").parse(), - "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected '.'; expected end of input"); + "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected digit after '.'; last read '0.:'"); CHECK_THROWS_WITH(json::parser("e.").parse(), - "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected 'e'"); + "[json.exception.parse_error.101] parse error at 1: syntax error - invalid literal; last read 'e'"); CHECK_THROWS_WITH(json::parser("1e.").parse(), - "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected 'e'; expected end of input"); + "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected '+', '-', or digit after exponent; last read '1e.'"); CHECK_THROWS_WITH(json::parser("1e/").parse(), - "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected 'e'; expected end of input"); + "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected '+', '-', or digit after exponent; last read '1e/'"); CHECK_THROWS_WITH(json::parser("1e:").parse(), - "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected 'e'; expected end of input"); + "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected '+', '-', or digit after exponent; last read '1e:'"); CHECK_THROWS_WITH(json::parser("1E.").parse(), - "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected 'E'; expected end of input"); + "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected '+', '-', or digit after exponent; last read '1E.'"); CHECK_THROWS_WITH(json::parser("1E/").parse(), - "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected 'E'; expected end of input"); + "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected '+', '-', or digit after exponent; last read '1E/'"); CHECK_THROWS_WITH(json::parser("1E:").parse(), - "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected 'E'; expected end of input"); + "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected '+', '-', or digit after exponent; last read '1E:'"); // unexpected end of null CHECK_THROWS_AS(json::parser("n").parse(), json::parse_error); From 06e2a291b1492dbabc6929ed77a63bda7427ced0 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 26 Mar 2017 17:26:41 +0200 Subject: [PATCH 110/530] :hammer: fixed number parsing --- src/json.hpp | 65 +++++++++++++++++++++------------- test/src/unit-class_parser.cpp | 2 +- test/src/unit-regression.cpp | 1 + test/src/unit-testsuites.cpp | 1 + 4 files changed, 43 insertions(+), 26 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 6cbcdf28..e47abab4 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -222,8 +222,8 @@ class parse_error : public exception const size_t byte; private: - parse_error(int id, size_t byte_, const char* what_arg) - : exception(id, what_arg), byte(byte_) + parse_error(int id_, size_t byte_, const char* what_arg) + : exception(id_, what_arg), byte(byte_) {} }; @@ -261,8 +261,8 @@ class invalid_iterator : public exception } private: - invalid_iterator(int id, const char* what_arg) - : exception(id, what_arg) + invalid_iterator(int id_, const char* what_arg) + : exception(id_, what_arg) {} }; @@ -300,8 +300,8 @@ class type_error : public exception } private: - type_error(int id, const char* what_arg) - : exception(id, what_arg) + type_error(int id_, const char* what_arg) + : exception(id_, what_arg) {} }; @@ -331,8 +331,8 @@ class out_of_range : public exception } private: - out_of_range(int id, const char* what_arg) - : exception(id, what_arg) + out_of_range(int id_, const char* what_arg) + : exception(id_, what_arg) {} }; @@ -357,8 +357,8 @@ class other_error : public exception } private: - other_error(int id, const char* what_arg) - : exception(id, what_arg) + other_error(int id_, const char* what_arg) + : exception(id_, what_arg) {} }; @@ -10828,24 +10828,33 @@ class basic_json add('\0'); --yylen; - if (has_exp or has_point) + // the conversion + char* endptr = nullptr; + + // try to parse integers first and fall back to floats + if (not has_exp and not has_point) { - value_float = std::strtod(yytext.data(), nullptr); - return token_type::value_float; + errno = 0; + if (has_sign) + { + value_integer = std::strtoll(yytext.data(), &endptr, 10); + if (JSON_LIKELY(errno == 0 and endptr == yytext.data() + yylen)) + { + return token_type::value_integer; + } + } + else + { + value_unsigned = std::strtoull(yytext.data(), &endptr, 10); + if (JSON_LIKELY(errno == 0 and endptr == yytext.data() + yylen)) + { + return token_type::value_unsigned; + } + } } - if (has_sign) - { - char* endptr = nullptr; - value_integer = std::strtoll(yytext.data(), &endptr, 10); - return token_type::value_integer; - } - else - { - char* endptr = nullptr; - value_unsigned = std::strtoull(yytext.data(), &endptr, 10); - return token_type::value_unsigned; - } + value_float = std::strtod(yytext.data(), nullptr); + return token_type::value_float; } token_type scan_true() @@ -10986,6 +10995,12 @@ class basic_json case lexer::token_type::value_float: { + // throw in case of infinity or NAN + if (not std::isfinite(value_float)) + { + JSON_THROW(out_of_range::create(406, "number overflow parsing '" + get_token_string() + "'")); + } + result.m_type = value_t::number_float; result.m_value = static_cast(value_float); return true; diff --git a/test/src/unit-class_parser.cpp b/test/src/unit-class_parser.cpp index 36451042..3e5b2871 100644 --- a/test/src/unit-class_parser.cpp +++ b/test/src/unit-class_parser.cpp @@ -529,7 +529,7 @@ TEST_CASE("parser class") if (c > 0x1f) { CHECK_THROWS_WITH(json::parser(s.c_str()).parse(), - "[json.exception.parse_error.101] parse error at 3: syntax error - invalid string: forbidden character after backspace; last read '\"\\" + std::string(1, c) + "'"); + "[json.exception.parse_error.101] parse error at 3: syntax error - invalid string: forbidden character after backspace; last read '\"\\" + std::string(1, static_cast(c)) + "'"); } break; } diff --git a/test/src/unit-regression.cpp b/test/src/unit-regression.cpp index 2c09abaa..df685d2c 100644 --- a/test/src/unit-regression.cpp +++ b/test/src/unit-regression.cpp @@ -216,6 +216,7 @@ TEST_CASE("regression tests") { json a = {1, 2, 3}; json::reverse_iterator rit = ++a.rbegin(); + CHECK(*rit == json(2)); } { json a = {1, 2, 3}; diff --git a/test/src/unit-testsuites.cpp b/test/src/unit-testsuites.cpp index 8d6a8162..c46a4ffa 100644 --- a/test/src/unit-testsuites.cpp +++ b/test/src/unit-testsuites.cpp @@ -305,6 +305,7 @@ TEST_CASE("compliance tests from nativejson-benchmark") std::string json_string( (std::istreambuf_iterator(f) ), (std::istreambuf_iterator()) ); + CAPTURE(json_string); json j = json::parse(json_string); CHECK(j.dump() == json_string); } From 342552797093683ec76316b8d588056720a0813d Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 26 Mar 2017 18:12:58 +0200 Subject: [PATCH 111/530] :hammer: added code for locale-independent number parsing --- errors.txt | 7 ------- src/json.hpp | 25 +++++++++++++++++++++---- 2 files changed, 21 insertions(+), 11 deletions(-) delete mode 100644 errors.txt diff --git a/errors.txt b/errors.txt deleted file mode 100644 index d3a6c6db..00000000 --- a/errors.txt +++ /dev/null @@ -1,7 +0,0 @@ -- test/test-class_parser - - 617 failed -- test/test-regression - - 11 failed -- test/test-testsuites - - 43 failed - diff --git a/src/json.hpp b/src/json.hpp index e47abab4..2ab32885 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -10517,12 +10517,13 @@ class basic_json } explicit lexer(std::istream& i) - // : ia(new input_stream_adapter(i)) - : ia(new cached_input_stream_adapter(i)) + : ia(new cached_input_stream_adapter(i)), + decimal_point_char(get_decimal_point()) {} lexer(const char* buff, const size_t len) - : ia(new input_buffer_adapter(buff, len)) + : ia(new input_buffer_adapter(buff, len)), + decimal_point_char(get_decimal_point()) {} ~lexer() @@ -10536,6 +10537,18 @@ class basic_json lexer operator=(const lexer&) = delete; private: + ///////////////////// + // locales + ///////////////////// + + /// return the locale-dependent decimal point + static char get_decimal_point() noexcept + { + const auto loc = localeconv(); + assert(loc != nullptr); + return (loc->decimal_point == nullptr) ? '.' : loc->decimal_point[0]; + } + ///////////////////// // scan functions ///////////////////// @@ -10815,7 +10828,8 @@ class basic_json return token_type::parse_error; } - add(current); + // add current character and fix decimal point + add((state == 4) ? decimal_point_char : current); get(); old_state = state; state = lookup[state][static_cast(current)]; @@ -11103,6 +11117,9 @@ class basic_json long long value_integer = 0; unsigned long long value_unsigned = 0; double value_float = 0; + + // the decimal point + const char decimal_point_char = '\0'; }; /*! From f75e195db3a057295c296b8aae66099fdf462d11 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 26 Mar 2017 18:45:04 +0200 Subject: [PATCH 112/530] :hammer: added code for user-defined number types --- src/json.hpp | 41 ++++++++++++++++++++++++++++++----------- 1 file changed, 30 insertions(+), 11 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 2ab32885..22f65979 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -10771,6 +10771,23 @@ class basic_json } } + // overloaded wrappers for strtod/strtof/strtold + // that will be called from parse + static void strtof(float& f, const char* str, char** endptr) noexcept + { + f = std::strtof(str, endptr); + } + + static void strtof(double& f, const char* str, char** endptr) noexcept + { + f = std::strtod(str, endptr); + } + + static void strtof(long double& f, const char* str, char** endptr) noexcept + { + f = std::strtold(str, endptr); + } + token_type scan_number() { static unsigned char lookup[9][256] = @@ -10851,23 +10868,25 @@ class basic_json errno = 0; if (has_sign) { - value_integer = std::strtoll(yytext.data(), &endptr, 10); - if (JSON_LIKELY(errno == 0 and endptr == yytext.data() + yylen)) + const auto x = std::strtoll(yytext.data(), &endptr, 10); + value_integer = static_cast(x); + if (JSON_LIKELY(errno == 0 and endptr == yytext.data() + yylen and value_integer == x)) { return token_type::value_integer; } } else { - value_unsigned = std::strtoull(yytext.data(), &endptr, 10); - if (JSON_LIKELY(errno == 0 and endptr == yytext.data() + yylen)) + const auto x = std::strtoull(yytext.data(), &endptr, 10); + value_unsigned = static_cast(x); + if (JSON_LIKELY(errno == 0 and endptr == yytext.data() + yylen and value_unsigned == x)) { return token_type::value_unsigned; } } } - value_float = std::strtod(yytext.data(), nullptr); + strtof(value_float, yytext.data(), nullptr); return token_type::value_float; } @@ -10996,14 +11015,14 @@ class basic_json case lexer::token_type::value_unsigned: { result.m_type = value_t::number_unsigned; - result.m_value = static_cast(value_unsigned); + result.m_value = value_unsigned; return true; } case lexer::token_type::value_integer: { result.m_type = value_t::number_integer; - result.m_value = static_cast(value_integer); + result.m_value = value_integer; return true; } @@ -11016,7 +11035,7 @@ class basic_json } result.m_type = value_t::number_float; - result.m_value = static_cast(value_float); + result.m_value = value_float; return true; } @@ -11114,9 +11133,9 @@ class basic_json std::string error_message = ""; // number values - long long value_integer = 0; - unsigned long long value_unsigned = 0; - double value_float = 0; + number_integer_t value_integer = 0; + number_unsigned_t value_unsigned = 0; + number_float_t value_float = 0; // the decimal point const char decimal_point_char = '\0'; From b0c47abd889d47c89deeeb8632b2fd194b6fe9b3 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Mon, 27 Mar 2017 22:58:44 +0200 Subject: [PATCH 113/530] :hammer: rewrote string parser to cope with UTF-8 --- src/json.hpp | 281 +++++++++++++++++++++++++++++++++++++++++++++++---- 1 file changed, 264 insertions(+), 17 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 22f65979..9dd50447 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -10587,8 +10587,20 @@ class basic_json // reset yytext (ignore opening quote) reset(); + // we entered the function by reading an open quote + assert (current == '\"'); + + static unsigned char next[256] = {17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 0, 0, 16, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 4, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 5, 2, 2, 6, 3, 3, 3, 7, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18}; + + // state variable + int state = -1; + + // whether the state is already set + bool state_set = false; + while (true) { + // get next character get(); // end of file while parsing string @@ -10598,25 +10610,240 @@ class basic_json return token_type::parse_error; } - // control character - if (JSON_UNLIKELY('\x00' <= current and current <= '\x1f')) - { - error_message = "invalid string: control characters (U+0000 through U+001f) must be escaped"; - return token_type::parse_error; - } + // after coping with EOF, we only cope with bytes + //assert(0 <= current and current <= 255); + unsigned char ch = static_cast(current); - switch (current) + // get next state + state = state_set ? state : next[ch]; + // reset variable + state_set = false; + + // 'add': 0, + // 'add_check1': 1, + // 'add_check2': 2, + // 'add_check3': 3, + // 'add_check_e0': 4, + // 'add_check_ed': 5, + // 'add_check_f0': 6, + // 'add_check_f4': 7, + // 'check1': 8, + // 'check2': 9, + // 'check3': 10, + // 'check_e0': 11, + // 'check_ed': 12, + // 'check_f0': 13, + // 'check_f4': 14, + // 'escape': 15, + // 'end': 16, + // 'error_invalid': 17, + // 'error_utf8': 18 + assert(0 <= state and state <= 18); + + switch (state) { - // closing quote - case '\"': + // add + case 0: { - add('\0'); - --yylen; - return token_type::value_string; + add(current); + break; } - // escape sequence - case '\\': + // add_check1 + case 1: + { + add(current); + // next state is check1 + state = 8; + state_set = true; + break; + } + + // add_check2 + case 2: + { + add(current); + // next state is check2 + state = 9; + state_set = true; + break; + } + + // add_check3 + case 3: + { + add(current); + // next state is check3 + state = 10; + state_set = true; + break; + } + + // add_check_e0 + case 4: + { + add(current); + // next state is check_e0 + state = 11; + state_set = true; + break; + } + + // add_check_ed + case 5: + { + add(current); + // next state is check_ed + state = 12; + state_set = true; + break; + } + + // add_check_f0 + case 6: + { + add(current); + // next state is check_f0 + state = 13; + state_set = true; + break; + } + + // add_check_f4 + case 7: + { + add(current); + // next state is check_f4 + state = 14; + state_set = true; + break; + } + + // check1 + case 8: + { + if (JSON_LIKELY(0x80 <= ch and ch <= 0xBF)) + { + add(current); + break; + } + else + { + error_message = "invalid string: not well-formed UTF-8 byte"; + return token_type::parse_error; + } + } + + // check2 + case 9: + { + if (JSON_LIKELY(0x80 <= ch and ch <= 0xBF)) + { + add(current); + // next state is check1 + state = 8; + state_set = true; + break; + } + else + { + error_message = "invalid string: not well-formed UTF-8 byte"; + return token_type::parse_error; + } + } + + // check3 + case 10: + { + if (JSON_LIKELY(0x80 <= ch and ch <= 0xBF)) + { + add(current); + // next state is check2 + state = 9; + state_set = true; + break; + } + else + { + error_message = "invalid string: not well-formed UTF-8 byte"; + return token_type::parse_error; + } + } + + // check_e0 + case 11: + { + if (JSON_LIKELY(0xA0 <= ch and ch <= 0xBF)) + { + add(current); + // next state is check1 + state = 8; + state_set = true; + break; + } + else + { + error_message = "invalid string: not well-formed UTF-8 byte"; + return token_type::parse_error; + } + } + + // check_ed + case 12: + { + if (JSON_LIKELY(0x80 <= ch and ch <= 0x9F)) + { + add(current); + // next state is check1 + state = 8; + state_set = true; + break; + } + else + { + error_message = "invalid string: not well-formed UTF-8 byte"; + return token_type::parse_error; + } + } + + // check_f0 + case 13: + { + if (JSON_LIKELY(0x90 <= ch and ch <= 0xBF)) + { + add(current); + // next state is check2 + state = 9; + state_set = true; + break; + } + else + { + error_message = "invalid string: not well-formed UTF-8 byte"; + return token_type::parse_error; + } + } + + // check_f4 + case 14: + { + if (JSON_LIKELY(0x80 <= ch and ch <= 0x8F)) + { + add(current); + // next state is check2 + state = 9; + state_set = true; + break; + } + else + { + error_message = "invalid string: not well-formed UTF-8 byte"; + return token_type::parse_error; + } + } + + // escape + case 15: { switch (get()) { @@ -10761,11 +10988,31 @@ class basic_json break; } - // any other character + // end + case 16: + { + add('\0'); + --yylen; + return token_type::value_string; + } + + // error_invalid + case 17: + { + error_message = "invalid string: control characters (U+0000 through U+001f) must be escaped"; + return token_type::parse_error; + } + + // error_utf8 + case 18: + { + error_message = "invalid string: not well-formed UTF-8 byte"; + return token_type::parse_error; + } + default: { - add(current); - break; + assert(false); } } } From bbb6bd9de560a300aa9fd6d19e07876c687a7868 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 28 Mar 2017 17:00:03 +0200 Subject: [PATCH 114/530] :hammer: fixing last failing test cases --- src/json.hpp | 27 +++++++++++++--------- test/src/unit-class_parser.cpp | 14 ++++++------ test/src/unit-testsuites.cpp | 41 ++++++++++++---------------------- 3 files changed, 37 insertions(+), 45 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 9dd50447..cbd630e4 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -7179,7 +7179,7 @@ class basic_json static basic_json parse(const CharT s, const parser_callback_t cb = nullptr) { - return parser(reinterpret_cast(s), cb).parse(); + return parser(reinterpret_cast(s), cb).parse(true); } /*! @@ -7214,7 +7214,7 @@ class basic_json static basic_json parse(std::istream& i, const parser_callback_t cb = nullptr) { - return parser(i, cb).parse(); + return parser(i, cb).parse(true); } /*! @@ -7223,7 +7223,7 @@ class basic_json static basic_json parse(std::istream&& i, const parser_callback_t cb = nullptr) { - return parser(i, cb).parse(); + return parser(i, cb).parse(true); } /*! @@ -7295,10 +7295,10 @@ class basic_json // to generate "unexpected EOF" error message if (std::distance(first, last) <= 0) { - return parser("").parse(); + return parser("").parse(true); } - return parser(first, last, cb).parse(); + return parser(first, last, cb).parse(true); } /*! @@ -7386,7 +7386,7 @@ class basic_json */ friend std::istream& operator<<(basic_json& j, std::istream& i) { - j = parser(i).parse(); + j = parser(i).parse(false); return i; } @@ -7396,7 +7396,7 @@ class basic_json */ friend std::istream& operator>>(std::istream& i, basic_json& j) { - j = parser(i).parse(); + j = parser(i).parse(false); return i; } @@ -11043,7 +11043,7 @@ class basic_json {10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 3, 2, 2, 2, 2, 2, 2, 2, 2, 2, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10}, {9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 4, 9, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 5, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 5, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9}, {9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 4, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 5, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 5, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9}, - {10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 5, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 5, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10}, + {10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10}, {10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 8, 10, 8, 10, 10, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10}, {9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 5, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 5, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9}, {9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9}, @@ -11423,11 +11423,14 @@ class basic_json /*! @brief public parser interface + + @param[in] strict whether to expect the last token to be EOF + @throw parse_error.101 in case of an unexpected token @throw parse_error.102 if to_unicode fails or surrogate error @throw parse_error.103 if to_unicode fails */ - basic_json parse() + basic_json parse(const bool strict = true) { // read first token get_token(); @@ -11435,8 +11438,10 @@ class basic_json basic_json result = parse_internal(true); result.assert_invariant(); - // FIXME: this is new behavior - //expect(lexer::token_type::end_of_input); + if (strict) + { + expect(lexer::token_type::end_of_input); + } // return parser result and replace it with null in case the // top-level value was discarded by the callback function diff --git a/test/src/unit-class_parser.cpp b/test/src/unit-class_parser.cpp index 3e5b2871..f36eb900 100644 --- a/test/src/unit-class_parser.cpp +++ b/test/src/unit-class_parser.cpp @@ -306,9 +306,9 @@ TEST_CASE("parser class") CHECK_THROWS_AS(json::parser("+0").parse(), json::parse_error); CHECK_THROWS_WITH(json::parser("01").parse(), - "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected '01'"); + "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected number literal; expected end of input"); CHECK_THROWS_WITH(json::parser("-01").parse(), - "[json.exception.parse_error.101] parse error at 3: syntax error - unexpected '-01'"); + "[json.exception.parse_error.101] parse error at 3: syntax error - unexpected number literal; expected end of input"); CHECK_THROWS_WITH(json::parser("--1").parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - invalid number; expected digit after '-'; last read '--'"); CHECK_THROWS_WITH(json::parser("1.").parse(), @@ -318,7 +318,7 @@ TEST_CASE("parser class") CHECK_THROWS_WITH(json::parser("1E-").parse(), "[json.exception.parse_error.101] parse error at 4: syntax error - invalid number; expected digit after exponent sign; last read '1E-'"); CHECK_THROWS_WITH(json::parser("1.E1").parse(), - "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected '.'; expected end of input"); + "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected digit after '.'; last read '1.E'"); CHECK_THROWS_WITH(json::parser("-1E").parse(), "[json.exception.parse_error.101] parse error at 4: syntax error - invalid number; expected '+', '-', or digit after exponent; last read '-1E'"); CHECK_THROWS_WITH(json::parser("-0E#").parse(), @@ -326,19 +326,19 @@ TEST_CASE("parser class") CHECK_THROWS_WITH(json::parser("-0E-#").parse(), "[json.exception.parse_error.101] parse error at 5: syntax error - invalid number; expected digit after exponent sign; last read '-0E-#'"); CHECK_THROWS_WITH(json::parser("-0#").parse(), - "[json.exception.parse_error.101] parse error at 3: syntax error - unexpected '#'; expected end of input"); + "[json.exception.parse_error.101] parse error at 3: syntax error - invalid literal; last read: '-0#'; expected end of input"); CHECK_THROWS_WITH(json::parser("-0.0:").parse(), "[json.exception.parse_error.101] parse error at 5: syntax error - unexpected ':'; expected end of input"); CHECK_THROWS_WITH(json::parser("-0.0Z").parse(), - "[json.exception.parse_error.101] parse error at 5: syntax error - unexpected 'Z'; expected end of input"); + "[json.exception.parse_error.101] parse error at 5: syntax error - invalid literal; last read: '-0.0Z'; expected end of input"); CHECK_THROWS_WITH(json::parser("-0E123:").parse(), "[json.exception.parse_error.101] parse error at 7: syntax error - unexpected ':'; expected end of input"); CHECK_THROWS_WITH(json::parser("-0e0-:").parse(), - "[json.exception.parse_error.101] parse error at 5: syntax error - unexpected '-'; expected end of input"); + "[json.exception.parse_error.101] parse error at 6: syntax error - invalid number; expected digit after '-'; last read: '-:'; expected end of input"); CHECK_THROWS_WITH(json::parser("-0e-:").parse(), "[json.exception.parse_error.101] parse error at 5: syntax error - invalid number; expected digit after exponent sign; last read '-0e-:'"); CHECK_THROWS_WITH(json::parser("-0f").parse(), - "[json.exception.parse_error.101] parse error at 3: syntax error - unexpected 'f'; expected end of input"); + "[json.exception.parse_error.101] parse error at 4: syntax error - invalid literal; expected 'false'; last read: '-0f'; expected end of input"); } } } diff --git a/test/src/unit-testsuites.cpp b/test/src/unit-testsuites.cpp index c46a4ffa..b0c721d2 100644 --- a/test/src/unit-testsuites.cpp +++ b/test/src/unit-testsuites.cpp @@ -77,9 +77,8 @@ TEST_CASE("compliance tests from json.org") }) { CAPTURE(filename); - json j; std::ifstream f(filename); - CHECK_THROWS_AS(j << f, json::parse_error); + CHECK_THROWS_AS(json::parse(f), json::parse_error); } } @@ -93,9 +92,8 @@ TEST_CASE("compliance tests from json.org") }) { CAPTURE(filename); - json j; std::ifstream f(filename); - CHECK_NOTHROW(j << f); + CHECK_NOTHROW(json::parse(f)); } } } @@ -320,7 +318,7 @@ TEST_CASE("test suite from json-test-suite") // strings in a JSON array std::ifstream f("test/data/json_testsuite/sample.json"); json j; - CHECK_NOTHROW(j << f); + CHECK_NOTHROW(j = json::parse(f)); // the array has 3 elements CHECK(j.size() == 3); @@ -334,36 +332,31 @@ TEST_CASE("json.org examples") SECTION("1.json") { std::ifstream f("test/data/json.org/1.json"); - json j; - CHECK_NOTHROW(j << f); + CHECK_NOTHROW(json::parse(f)); } SECTION("2.json") { std::ifstream f("test/data/json.org/2.json"); - json j; - CHECK_NOTHROW(j << f); + CHECK_NOTHROW(json::parse(f)); } SECTION("3.json") { std::ifstream f("test/data/json.org/3.json"); - json j; - CHECK_NOTHROW(j << f); + CHECK_NOTHROW(json::parse(f)); } SECTION("4.json") { std::ifstream f("test/data/json.org/4.json"); - json j; - CHECK_NOTHROW(j << f); + CHECK_NOTHROW(json::parse(f)); } SECTION("5.json") { std::ifstream f("test/data/json.org/5.json"); - json j; - CHECK_NOTHROW(j << f); + CHECK_NOTHROW(json::parse(f)); } } @@ -545,8 +538,7 @@ TEST_CASE("nst's JSONTestSuite") { CAPTURE(filename); std::ifstream f(filename); - json j; - CHECK_NOTHROW(j << f); + CHECK_NOTHROW(json::parse(f)); } } @@ -754,8 +746,7 @@ TEST_CASE("nst's JSONTestSuite") { CAPTURE(filename); std::ifstream f(filename); - json j; - CHECK_THROWS_AS(j << f, json::parse_error); + CHECK_THROWS_AS(json::parse(f), json::parse_error); } } @@ -777,8 +768,7 @@ TEST_CASE("nst's JSONTestSuite") { CAPTURE(filename); std::ifstream f(filename); - json j; - CHECK_NOTHROW(j << f); + CHECK_NOTHROW(json::parse(f)); } } @@ -797,8 +787,7 @@ TEST_CASE("nst's JSONTestSuite") { CAPTURE(filename); std::ifstream f(filename); - json j; - CHECK_THROWS_AS(j << f, json::out_of_range); + CHECK_THROWS_AS(json::parse(f), json::out_of_range); } } @@ -824,8 +813,7 @@ TEST_CASE("nst's JSONTestSuite") { CAPTURE(filename); std::ifstream f(filename); - json j; - CHECK_THROWS_AS(j << f, json::parse_error); + CHECK_THROWS_AS(json::parse(f), json::parse_error); } } } @@ -851,8 +839,7 @@ TEST_CASE("Big List of Naughty Strings") SECTION("parsing blns.json") { std::ifstream f("test/data/big-list-of-naughty-strings/blns.json"); - json j; - CHECK_NOTHROW(j << f); + CHECK_NOTHROW(json::parse(f)); } // check if parsed strings roundtrip From 1e495945f1f20dcb986e27b4f55f1350fb2d10c6 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 28 Mar 2017 18:06:43 +0200 Subject: [PATCH 115/530] :hammer: fixed buffer overflow --- src/json.hpp | 10 ++++++---- 1 file changed, 6 insertions(+), 4 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index cbd630e4..bdb3861a 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -8575,7 +8575,7 @@ class basic_json case 0x7f: // UTF-8 string (indefinite length) { std::string result; - while (check_length(v.size(), 1, idx), v[idx] != 0xff) + while (static_cast(check_length(v.size(), 1, idx)), v[idx] != 0xff) { string_t s = from_cbor_internal(v, idx); result += s; @@ -8671,7 +8671,7 @@ class basic_json case 0x9f: // array (indefinite length) { basic_json result = value_t::array; - while (check_length(v.size(), 1, idx), v[idx] != 0xff) + while (static_cast(check_length(v.size(), 1, idx)), v[idx] != 0xff) { result.push_back(from_cbor_internal(v, idx)); } @@ -8776,7 +8776,7 @@ class basic_json case 0xbf: // map (indefinite length) { basic_json result = value_t::object; - while (check_length(v.size(), 1, idx), v[idx] != 0xff) + while (static_cast(check_length(v.size(), 1, idx)), v[idx] != 0xff) { cbor_expect_string(v, idx); std::string key = from_cbor_internal(v, idx); @@ -10442,7 +10442,9 @@ class basic_json std::string read(size_t offset, size_t length) override { - return std::string(start + offset, length); + // avoid reading too many characters + const size_t max_length = static_cast(limit-start); + return std::string(start + offset, std::min({length, max_length})); } private: From 6a6fbea62c61fb0416cbc891100bc4acd976d95e Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 28 Mar 2017 20:08:08 +0200 Subject: [PATCH 116/530] :hammer: some cleanup --- src/json.hpp | 117 ++++++++++++++++++++++++++------------------------- 1 file changed, 59 insertions(+), 58 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index bdb3861a..48607bb8 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -10443,7 +10443,7 @@ class basic_json std::string read(size_t offset, size_t length) override { // avoid reading too many characters - const size_t max_length = static_cast(limit-start); + const size_t max_length = static_cast(limit - start); return std::string(start + offset, std::min({length, max_length})); } @@ -10462,9 +10462,9 @@ class basic_json literal_false, ///< the `false` literal literal_null, ///< the `null` literal value_string, ///< a string -- use get_string() for actual value - value_unsigned, ///< an unsigned integer -- use get_number() for actual value - value_integer, ///< a signed integer -- use get_number() for actual value - value_float, ///< an floating point number -- use get_number() for actual value + value_unsigned, ///< an unsigned integer -- use get_number_unsigned() for actual value + value_integer, ///< a signed integer -- use get_number_integer() for actual value + value_float, ///< an floating point number -- use get_number_float() for actual value begin_array, ///< the character for array begin `[` begin_object, ///< the character for object begin `{` end_array, ///< the character for array end `]` @@ -10476,7 +10476,7 @@ class basic_json }; /// return name of values of type token_type (only used for errors) - static std::string token_type_name(const token_type t) + static const char* token_type_name(const token_type t) noexcept { switch (t) { @@ -10563,6 +10563,7 @@ class basic_json int codepoint = 0; + // check the next 4 bytes for (size_t i = 0; i < 4; ++i) { const int8_t digit = ascii_to_hex[static_cast(get())]; @@ -10575,6 +10576,7 @@ class basic_json codepoint += digit; } + // except the last byte, result must be multiplied by 16 if (i != 3) { codepoint <<= 4; @@ -10895,7 +10897,7 @@ class basic_json } // check if code point is a high surrogate - if (codepoint1 >= 0xD800 and codepoint1 <= 0xDBFF) + if (0xD800 <= codepoint1 and codepoint1 <= 0xDBFF) { // expect next \uxxxx entry if (JSON_LIKELY(get() == '\\' and get() == 'u')) @@ -10909,7 +10911,7 @@ class basic_json } // check if codepoint2 is a low surrogate - if (codepoint2 >= 0xDC00 and codepoint2 <= 0xDFFF) + if (JSON_LIKELY(0xDC00 <= codepoint2 and codepoint2 <= 0xDFFF)) { codepoint = // high surrogate occupies the most significant 22 bits @@ -10935,7 +10937,7 @@ class basic_json } else { - if (JSON_UNLIKELY(codepoint1 >= 0xDC00 and codepoint1 <= 0xDFFF)) + if (JSON_UNLIKELY(0xDC00 <= codepoint1 and codepoint1 <= 0xDFFF)) { error_message = "invalid string: missing high surrogate"; return token_type::parse_error; @@ -10993,6 +10995,7 @@ class basic_json // end case 16: { + // terminate yytext add('\0'); --yylen; return token_type::value_string; @@ -11108,15 +11111,13 @@ class basic_json add('\0'); --yylen; - // the conversion - char* endptr = nullptr; - // try to parse integers first and fall back to floats if (not has_exp and not has_point) { errno = 0; if (has_sign) { + char* endptr = nullptr; const auto x = std::strtoll(yytext.data(), &endptr, 10); value_integer = static_cast(x); if (JSON_LIKELY(errno == 0 and endptr == yytext.data() + yylen and value_integer == x)) @@ -11126,6 +11127,7 @@ class basic_json } else { + char* endptr = nullptr; const auto x = std::strtoull(yytext.data(), &endptr, 10); value_unsigned = static_cast(x); if (JSON_LIKELY(errno == 0 and endptr == yytext.data() + yylen and value_unsigned == x)) @@ -11218,11 +11220,26 @@ class basic_json } public: - constexpr size_t get_position() const + constexpr size_t get_position() const noexcept { return chars_read; } + constexpr number_integer_t get_number_integer() const noexcept + { + return value_integer; + } + + constexpr number_unsigned_t get_number_unsigned() const noexcept + { + return value_unsigned; + } + + constexpr number_float_t get_number_float() const noexcept + { + return value_float; + } + const std::string get_string() { return std::string(yytext.data(), yylen); @@ -11252,49 +11269,11 @@ class basic_json return ss.str(); } - const std::string& get_error_message() const + const std::string& get_error_message() const noexcept { return error_message; } - bool get_number(basic_json& result, const token_type token) const - { - switch (token) - { - case lexer::token_type::value_unsigned: - { - result.m_type = value_t::number_unsigned; - result.m_value = value_unsigned; - return true; - } - - case lexer::token_type::value_integer: - { - result.m_type = value_t::number_integer; - result.m_value = value_integer; - return true; - } - - case lexer::token_type::value_float: - { - // throw in case of infinity or NAN - if (not std::isfinite(value_float)) - { - JSON_THROW(out_of_range::create(406, "number overflow parsing '" + get_token_string() + "'")); - } - - result.m_type = value_t::number_float; - result.m_value = value_float; - return true; - } - - default: - { - return false; - } - } - } - token_type scan() { // read next character and ignore whitespace @@ -11602,8 +11581,8 @@ class basic_json case lexer::token_type::literal_null: { - get_token(); result.m_type = value_t::null; + get_token(); break; } @@ -11616,25 +11595,47 @@ class basic_json case lexer::token_type::literal_true: { - get_token(); result.m_type = value_t::boolean; result.m_value = true; + get_token(); break; } case lexer::token_type::literal_false: { - get_token(); result.m_type = value_t::boolean; result.m_value = false; + get_token(); break; } case lexer::token_type::value_unsigned: + { + result.m_type = value_t::number_unsigned; + result.m_value = m_lexer.get_number_unsigned(); + get_token(); + break; + } + case lexer::token_type::value_integer: + { + result.m_type = value_t::number_integer; + result.m_value = m_lexer.get_number_integer(); + get_token(); + break; + } + case lexer::token_type::value_float: { - m_lexer.get_number(result, last_token); + result.m_type = value_t::number_float; + result.m_value = m_lexer.get_number_float(); + + // throw in case of infinity or NAN + if (JSON_UNLIKELY(not std::isfinite(result.m_value.number_float))) + { + JSON_THROW(out_of_range::create(406, "number overflow parsing '" + m_lexer.get_token_string() + "'")); + } + get_token(); break; } @@ -11674,10 +11675,10 @@ class basic_json } else { - error_msg += "unexpected " + lexer::token_type_name(last_token); + error_msg += "unexpected " + std::string(lexer::token_type_name(last_token)); } - error_msg += "; expected " + lexer::token_type_name(t); + error_msg += "; expected " + std::string(lexer::token_type_name(t)); JSON_THROW(parse_error::create(101, m_lexer.get_position(), error_msg)); } } @@ -11696,7 +11697,7 @@ class basic_json } else { - error_msg += "unexpected " + lexer::token_type_name(last_token); + error_msg += "unexpected " + std::string(lexer::token_type_name(last_token)); } JSON_THROW(parse_error::create(101, m_lexer.get_position(), error_msg)); From 50e251f5f644ca9fd2d2fd58a62f0eff2f31ab18 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 28 Mar 2017 20:17:00 +0200 Subject: [PATCH 117/530] :hammer: adjusted test case to new signature of token_type_name() --- test/src/unit-class_lexer.cpp | 32 ++++++++++++++++---------------- 1 file changed, 16 insertions(+), 16 deletions(-) diff --git a/test/src/unit-class_lexer.cpp b/test/src/unit-class_lexer.cpp index d4bf2dd2..214cae75 100644 --- a/test/src/unit-class_lexer.cpp +++ b/test/src/unit-class_lexer.cpp @@ -87,22 +87,22 @@ TEST_CASE("lexer class") SECTION("token_type_name") { - CHECK((json::lexer::token_type_name(json::lexer::token_type::uninitialized) == "")); - CHECK((json::lexer::token_type_name(json::lexer::token_type::literal_true) == "true literal")); - CHECK((json::lexer::token_type_name(json::lexer::token_type::literal_false) == "false literal")); - CHECK((json::lexer::token_type_name(json::lexer::token_type::literal_null) == "null literal")); - CHECK((json::lexer::token_type_name(json::lexer::token_type::value_string) == "string literal")); - CHECK((json::lexer::token_type_name(json::lexer::token_type::value_unsigned) == "number literal")); - CHECK((json::lexer::token_type_name(json::lexer::token_type::value_integer) == "number literal")); - CHECK((json::lexer::token_type_name(json::lexer::token_type::value_float) == "number literal")); - CHECK((json::lexer::token_type_name(json::lexer::token_type::begin_array) == "'['")); - CHECK((json::lexer::token_type_name(json::lexer::token_type::begin_object) == "'{'")); - CHECK((json::lexer::token_type_name(json::lexer::token_type::end_array) == "']'")); - CHECK((json::lexer::token_type_name(json::lexer::token_type::end_object) == "'}'")); - CHECK((json::lexer::token_type_name(json::lexer::token_type::name_separator) == "':'")); - CHECK((json::lexer::token_type_name(json::lexer::token_type::value_separator) == "','")); - CHECK((json::lexer::token_type_name(json::lexer::token_type::parse_error) == "")); - CHECK((json::lexer::token_type_name(json::lexer::token_type::end_of_input) == "end of input")); + CHECK((json::lexer::token_type_name(std::string(json::lexer::token_type::uninitialized)) == "")); + CHECK((json::lexer::token_type_name(std::string(json::lexer::token_type::literal_true)) == "true literal")); + CHECK((json::lexer::token_type_name(std::string(json::lexer::token_type::literal_false)) == "false literal")); + CHECK((json::lexer::token_type_name(std::string(json::lexer::token_type::literal_null)) == "null literal")); + CHECK((json::lexer::token_type_name(std::string(json::lexer::token_type::value_string)) == "string literal")); + CHECK((json::lexer::token_type_name(std::string(json::lexer::token_type::value_unsigned)) == "number literal")); + CHECK((json::lexer::token_type_name(std::string(json::lexer::token_type::value_integer)) == "number literal")); + CHECK((json::lexer::token_type_name(std::string(json::lexer::token_type::value_float)) == "number literal")); + CHECK((json::lexer::token_type_name(std::string(json::lexer::token_type::begin_array)) == "'['")); + CHECK((json::lexer::token_type_name(std::string(json::lexer::token_type::begin_object)) == "'{'")); + CHECK((json::lexer::token_type_name(std::string(json::lexer::token_type::end_array)) == "']'")); + CHECK((json::lexer::token_type_name(std::string(json::lexer::token_type::end_object)) == "'}'")); + CHECK((json::lexer::token_type_name(std::string(json::lexer::token_type::name_separator)) == "':'")); + CHECK((json::lexer::token_type_name(std::string(json::lexer::token_type::value_separator)) == "','")); + CHECK((json::lexer::token_type_name(std::string(json::lexer::token_type::parse_error)) == "")); + CHECK((json::lexer::token_type_name(std::string(json::lexer::token_type::end_of_input)) == "end of input")); } SECTION("parse errors on first character") From d37ca2eba8363a294469b066dc137236c8a13802 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 28 Mar 2017 20:29:27 +0200 Subject: [PATCH 118/530] :hammer: fixed test case --- test/src/unit-class_lexer.cpp | 32 ++++++++++++++++---------------- 1 file changed, 16 insertions(+), 16 deletions(-) diff --git a/test/src/unit-class_lexer.cpp b/test/src/unit-class_lexer.cpp index 214cae75..468e1f52 100644 --- a/test/src/unit-class_lexer.cpp +++ b/test/src/unit-class_lexer.cpp @@ -87,22 +87,22 @@ TEST_CASE("lexer class") SECTION("token_type_name") { - CHECK((json::lexer::token_type_name(std::string(json::lexer::token_type::uninitialized)) == "")); - CHECK((json::lexer::token_type_name(std::string(json::lexer::token_type::literal_true)) == "true literal")); - CHECK((json::lexer::token_type_name(std::string(json::lexer::token_type::literal_false)) == "false literal")); - CHECK((json::lexer::token_type_name(std::string(json::lexer::token_type::literal_null)) == "null literal")); - CHECK((json::lexer::token_type_name(std::string(json::lexer::token_type::value_string)) == "string literal")); - CHECK((json::lexer::token_type_name(std::string(json::lexer::token_type::value_unsigned)) == "number literal")); - CHECK((json::lexer::token_type_name(std::string(json::lexer::token_type::value_integer)) == "number literal")); - CHECK((json::lexer::token_type_name(std::string(json::lexer::token_type::value_float)) == "number literal")); - CHECK((json::lexer::token_type_name(std::string(json::lexer::token_type::begin_array)) == "'['")); - CHECK((json::lexer::token_type_name(std::string(json::lexer::token_type::begin_object)) == "'{'")); - CHECK((json::lexer::token_type_name(std::string(json::lexer::token_type::end_array)) == "']'")); - CHECK((json::lexer::token_type_name(std::string(json::lexer::token_type::end_object)) == "'}'")); - CHECK((json::lexer::token_type_name(std::string(json::lexer::token_type::name_separator)) == "':'")); - CHECK((json::lexer::token_type_name(std::string(json::lexer::token_type::value_separator)) == "','")); - CHECK((json::lexer::token_type_name(std::string(json::lexer::token_type::parse_error)) == "")); - CHECK((json::lexer::token_type_name(std::string(json::lexer::token_type::end_of_input)) == "end of input")); + CHECK((std::string(json::lexer::token_type_name(json::lexer::token_type::uninitialized)) == "")); + CHECK((std::string(json::lexer::token_type_name(json::lexer::token_type::literal_true)) == "true literal")); + CHECK((std::string(json::lexer::token_type_name(json::lexer::token_type::literal_false)) == "false literal")); + CHECK((std::string(json::lexer::token_type_name(json::lexer::token_type::literal_null)) == "null literal")); + CHECK((std::string(json::lexer::token_type_name(json::lexer::token_type::value_string)) == "string literal")); + CHECK((std::string(json::lexer::token_type_name(json::lexer::token_type::value_unsigned)) == "number literal")); + CHECK((std::string(json::lexer::token_type_name(json::lexer::token_type::value_integer)) == "number literal")); + CHECK((std::string(json::lexer::token_type_name(json::lexer::token_type::value_float)) == "number literal")); + CHECK((std::string(json::lexer::token_type_name(json::lexer::token_type::begin_array)) == "'['")); + CHECK((std::string(json::lexer::token_type_name(json::lexer::token_type::begin_object)) == "'{'")); + CHECK((std::string(json::lexer::token_type_name(json::lexer::token_type::end_array)) == "']'")); + CHECK((std::string(json::lexer::token_type_name(json::lexer::token_type::end_object)) == "'}'")); + CHECK((std::string(json::lexer::token_type_name(json::lexer::token_type::name_separator)) == "':'")); + CHECK((std::string(json::lexer::token_type_name(json::lexer::token_type::value_separator)) == "','")); + CHECK((std::string(json::lexer::token_type_name(json::lexer::token_type::parse_error)) == "")); + CHECK((std::string(json::lexer::token_type_name(json::lexer::token_type::end_of_input)) == "end of input")); } SECTION("parse errors on first character") From a0aa6ad2b2d7f58682f549ea5ca2496578f5267c Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 28 Mar 2017 20:42:12 +0200 Subject: [PATCH 119/530] :construction_worker: added Xcode 8.3 https://blog.travis-ci.com/2017-03-28-xcode-83-ga-is-here --- .travis.yml | 3 +++ 1 file changed, 3 insertions(+) diff --git a/.travis.yml b/.travis.yml index dcf7b6a2..8b9ba53e 100644 --- a/.travis.yml +++ b/.travis.yml @@ -148,6 +148,9 @@ matrix: - os: osx osx_image: xcode8.2 + - os: osx + osx_image: xcode8.3 + # Linux / GCC - os: linux From 19cd206d99756bb8aebb284a0d984677dc39faa6 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 28 Mar 2017 21:39:16 +0200 Subject: [PATCH 120/530] :memo: documented Xcode 8.3 --- README.md | 1 + 1 file changed, 1 insertion(+) diff --git a/README.md b/README.md index e23e3c94..803234d8 100644 --- a/README.md +++ b/README.md @@ -744,6 +744,7 @@ The following compilers are currently used in continuous integration at [Travis] | Clang Xcode 8.0 | Darwin Kernel Version 15.6.0 | Apple LLVM version 8.0.0 (clang-800.0.38) | | Clang Xcode 8.1 | Darwin Kernel Version 16.1.0 (macOS 10.12.1) | Apple LLVM version 8.0.0 (clang-800.0.42.1) | | Clang Xcode 8.2 | Darwin Kernel Version 16.1.0 (macOS 10.12.1) | Apple LLVM version 8.0.0 (clang-800.0.42.1) | +| Clang Xcode 8.3 | Darwin Kernel Version 16.5.0 (macOS 10.12.4) | Apple LLVM version 8.1.0 (clang-802.0.38) | | Visual Studio 14 2015 | Windows Server 2012 R2 (x64) | Microsoft (R) Build Engine version 14.0.25123.0 | | Visual Studio 2017 | Windows Server 2016 | Microsoft (R) Build Engine version 15.1.548.43366 | From 6b12e4047863dfb2060f92c42bd7ac8960dfd672 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 28 Mar 2017 21:55:26 +0200 Subject: [PATCH 121/530] :checkered_flag: removed __builtin_expect for MSVC --- src/json.hpp | 9 +++++++-- 1 file changed, 7 insertions(+), 2 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 48607bb8..c10c4037 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -91,8 +91,13 @@ SOFTWARE. #endif // manual branch prediction -#define JSON_LIKELY(x) __builtin_expect(!!(x), 1) -#define JSON_UNLIKELY(x) __builtin_expect(!!(x), 0) +#if defined(__clang__) || defined(__GNUC__) || defined(__GNUG__) + #define JSON_LIKELY(x) __builtin_expect(!!(x), 1) + #define JSON_UNLIKELY(x) __builtin_expect(!!(x), 0) +#else + #define JSON_LIKELY(x) x + #define JSON_UNLIKELY(x) x +#endif /*! @brief namespace for Niels Lohmann From 19d119e18c3ff13fc85c822c52c78399f9256617 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 28 Mar 2017 22:10:24 +0200 Subject: [PATCH 122/530] :hammer: fixed a warning in MSVC --- src/json.hpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/json.hpp b/src/json.hpp index c10c4037..cb52a6ef 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -10323,7 +10323,7 @@ class basic_json // refill is.read(reinterpret_cast(buffer.data()), static_cast(buffer.size())); // set unfilled characters to EOF - std::fill_n(buffer.begin() + is.gcount(), + std::fill_n(buffer.begin() + static_cast(is.gcount()), buffer.size() - static_cast(is.gcount()), std::char_traits::eof()); // the buffer is ready From 5d586b01928e329606d705f45d7012b1c0ec7527 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 28 Mar 2017 23:20:45 +0200 Subject: [PATCH 123/530] :hammer: cleanup --- src/json.hpp | 84 +++++++++++++++++++++++++++++++++++++--------------- 1 file changed, 60 insertions(+), 24 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index cb52a6ef..e9b621f6 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -10296,10 +10296,11 @@ class basic_json JSON_THROW(parse_error::create(111, 0, "bad input stream")); } - // initial fill; unfilled buffer charaters remain EOF + // initial fill; unfilled buffer characters remain EOF is.read(buffer.data(), static_cast(buffer.size())); - // ignore byte-order mark + // skip byte-order mark + assert(buffer.size() >= 3); if (buffer[0] == '\xEF' and buffer[1] == '\xBB' and buffer[2] == '\xBF') { buffer_pos += 3; @@ -10317,7 +10318,7 @@ class basic_json int get_character() override { - // check if refilling is neccessary + // check if refilling is necessary if (JSON_UNLIKELY(buffer_pos == buffer.size())) { // refill @@ -10360,17 +10361,18 @@ class basic_json } private: + /// the associated input stream std::istream& is; - // chars returned via get_character() + /// chars returned via get_character() size_t processed_chars = 0; - // chars processed in the current buffer + /// chars processed in the current buffer size_t buffer_pos = 0; - // position of the stream when we started + /// position of the stream when we started const std::streampos start_position; - // internal buffer + /// internal buffer std::vector buffer; }; @@ -10419,6 +10421,7 @@ class basic_json } private: + /// the associated input stream std::istream& is; }; @@ -10430,6 +10433,7 @@ class basic_json : input_adapter(), cursor(b), limit(b + l), start(b) {} + // delete because of pointer members input_buffer_adapter(const input_buffer_adapter&) = delete; input_buffer_adapter& operator=(input_buffer_adapter&) = delete; @@ -10453,8 +10457,11 @@ class basic_json } private: + /// pointer to the current character const char* cursor; + /// pointer past the last character const char* limit; + /// pointer to the first character const char* start; }; @@ -10597,7 +10604,7 @@ class basic_json reset(); // we entered the function by reading an open quote - assert (current == '\"'); + assert(current == '\"'); static unsigned char next[256] = {17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 0, 0, 16, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 4, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 5, 2, 2, 6, 3, 3, 3, 7, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18}; @@ -11022,14 +11029,12 @@ class basic_json default: { - assert(false); + assert(false); // LCOV_EXCL_LINE } } } } - // overloaded wrappers for strtod/strtof/strtold - // that will be called from parse static void strtof(float& f, const char* str, char** endptr) noexcept { f = std::strtof(str, endptr); @@ -11125,7 +11130,7 @@ class basic_json char* endptr = nullptr; const auto x = std::strtoll(yytext.data(), &endptr, 10); value_integer = static_cast(x); - if (JSON_LIKELY(errno == 0 and endptr == yytext.data() + yylen and value_integer == x)) + if (errno == 0 and endptr == yytext.data() + yylen and value_integer == x) { return token_type::value_integer; } @@ -11135,7 +11140,7 @@ class basic_json char* endptr = nullptr; const auto x = std::strtoull(yytext.data(), &endptr, 10); value_unsigned = static_cast(x); - if (JSON_LIKELY(errno == 0 and endptr == yytext.data() + yylen and value_unsigned == x)) + if (errno == 0 and endptr == yytext.data() + yylen and value_unsigned == x) { return token_type::value_unsigned; } @@ -11148,6 +11153,7 @@ class basic_json token_type scan_true() { + assert(current == 't'); if (JSON_LIKELY((get() == 'r' and get() == 'u' and get() == 'e'))) { return token_type::literal_true; @@ -11159,6 +11165,7 @@ class basic_json token_type scan_false() { + assert(current == 'f'); if (JSON_LIKELY((get() == 'a' and get() == 'l' and get() == 's' and get() == 'e'))) { return token_type::literal_false; @@ -11170,6 +11177,7 @@ class basic_json token_type scan_null() { + assert(current == 'n'); if (JSON_LIKELY((get() == 'u' and get() == 'l' and get() == 'l'))) { return token_type::literal_null; @@ -11183,13 +11191,14 @@ class basic_json // input management ///////////////////// - void reset() + /// reset yytext + void reset() noexcept { yylen = 0; start_pos = chars_read - 1; } - // get a character from the input + /// get a character from the input int get() { ++chars_read; @@ -11206,14 +11215,14 @@ class basic_json return current; } - // unget a character to the input - void unget() + /// unget a character to the input + void unget() noexcept { --chars_read; next_unget = true; } - // add a character to yytext + /// add a character to yytext void add(int c) { // resize yytext if necessary @@ -11225,48 +11234,70 @@ class basic_json } public: - constexpr size_t get_position() const noexcept - { - return chars_read; - } + ///////////////////// + // value getters + ///////////////////// + /// return integer value constexpr number_integer_t get_number_integer() const noexcept { return value_integer; } + /// return unsigned integer value constexpr number_unsigned_t get_number_unsigned() const noexcept { return value_unsigned; } + /// return floating-point value constexpr number_float_t get_number_float() const noexcept { return value_float; } + /// return string value const std::string get_string() { + // yytext cannot be returned as char*, because it may contain a + // null byte return std::string(yytext.data(), yylen); } + ///////////////////// + // diagnostics + ///////////////////// + + /// return position of last read token + constexpr size_t get_position() const noexcept + { + return chars_read; + } + + /// return the last read token (for errors only) std::string get_token_string() const { + // get the raw byte sequence of the last token std::string s = ia->read(start_pos, chars_read - start_pos); - std::stringstream ss; + // escape control characters + std::stringstream ss; for (auto c : s) { if (c == '\0' or c == std::char_traits::eof()) { + // ignore EOF continue; } else if ('\x00' <= c and c <= '\x1f') { + // escape control characters ss << ""; } else { + + // add character as is ss << c; } } @@ -11274,11 +11305,16 @@ class basic_json return ss.str(); } + /// return syntax error message const std::string& get_error_message() const noexcept { return error_message; } + ///////////////////// + // actual scanner + ///////////////////// + token_type scan() { // read next character and ignore whitespace @@ -11370,7 +11406,7 @@ class basic_json number_unsigned_t value_unsigned = 0; number_float_t value_float = 0; - // the decimal point + /// the decimal point const char decimal_point_char = '\0'; }; From 62dfdf3f5d47711db35cc18f7f6500e27a0298a3 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 28 Mar 2017 23:24:59 +0200 Subject: [PATCH 124/530] :memo: addressed #514 NaN values never compare equal to themselves or to other NaN values. --- src/json.hpp | 10 +++++++--- src/json.hpp.re2c | 10 +++++++--- 2 files changed, 14 insertions(+), 6 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index cae427ef..37db5c45 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -5640,7 +5640,7 @@ class basic_json @ref push_back(const typename object_t::value_type&). Otherwise, @a init is converted to a JSON value and added using @ref push_back(basic_json&&). - @param init an initializer list + @param[in] init an initializer list @complexity Linear in the size of the initializer list @a init. @@ -6117,13 +6117,17 @@ class basic_json Compares two JSON values for equality according to the following rules: - Two JSON values are equal if (1) they are from the same type and (2) - their stored values are the same. + their stored values are the same according to their respective + `operator==`. - Integer and floating-point numbers are automatically converted before comparison. Floating-point numbers are compared indirectly: two floating-point numbers `f1` and `f2` are considered equal if neither - `f1 > f2` nor `f2 > f1` holds. + `f1 > f2` nor `f2 > f1` holds. Note than two NaN values are always + treated as unequal. - Two JSON null values are equal. + @note NaN values never compare equal to themselves or to other NaN values. + @param[in] lhs first JSON value to consider @param[in] rhs second JSON value to consider @return whether the values @a lhs and @a rhs are equal diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 3aabece5..cafb2067 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -5640,7 +5640,7 @@ class basic_json @ref push_back(const typename object_t::value_type&). Otherwise, @a init is converted to a JSON value and added using @ref push_back(basic_json&&). - @param init an initializer list + @param[in] init an initializer list @complexity Linear in the size of the initializer list @a init. @@ -6117,13 +6117,17 @@ class basic_json Compares two JSON values for equality according to the following rules: - Two JSON values are equal if (1) they are from the same type and (2) - their stored values are the same. + their stored values are the same according to their respective + `operator==`. - Integer and floating-point numbers are automatically converted before comparison. Floating-point numbers are compared indirectly: two floating-point numbers `f1` and `f2` are considered equal if neither - `f1 > f2` nor `f2 > f1` holds. + `f1 > f2` nor `f2 > f1` holds. Note than two NaN values are always + treated as unequal. - Two JSON null values are equal. + @note NaN values never compare equal to themselves or to other NaN values. + @param[in] lhs first JSON value to consider @param[in] rhs second JSON value to consider @return whether the values @a lhs and @a rhs are equal From b4dbebffcdb98c91130133adb6583a1b5dafe6a6 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 28 Mar 2017 23:28:54 +0200 Subject: [PATCH 125/530] :white_check_mark: added regression test for #464 --- test/src/unit-regression.cpp | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/test/src/unit-regression.cpp b/test/src/unit-regression.cpp index f7f4c350..a7b6e117 100644 --- a/test/src/unit-regression.cpp +++ b/test/src/unit-regression.cpp @@ -837,6 +837,14 @@ TEST_CASE("regression tests") CHECK(j["double_value"].is_number_float()); } + SECTION("issue #464 - VS2017 implicit to std::string conversion fix") + { + json v = "test"; + std::string test; + test = v; + CHECK(v == "test"); + } + SECTION("issue #465 - roundtrip error while parsing 1000000000000000010E5") { json j1 = json::parse("1000000000000000010E5"); From c2e80a72d71662b483b21bf1d648eedc43c720d4 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Wed, 29 Mar 2017 00:39:47 +0200 Subject: [PATCH 126/530] :hammer: deprecated j << istream / j >> ostream functions #367 The implementation is non-standard. Deprecation allows a simpler API in the future without removing any features. --- Makefile | 3 ++ src/json.hpp | 66 +++++++++++++++++++++++------------- src/json.hpp.re2c | 66 +++++++++++++++++++++++------------- test/src/unit-regression.cpp | 12 ++++--- test/src/unit-testsuites.cpp | 28 +++++++-------- test/src/unit-unicode.cpp | 4 +-- 6 files changed, 110 insertions(+), 69 deletions(-) diff --git a/Makefile b/Makefile index 20857022..ab403afb 100644 --- a/Makefile +++ b/Makefile @@ -49,6 +49,7 @@ doctest: # -Wno-documentation-unknown-command: code uses user-defined commands like @complexity # -Wno-exit-time-destructors: warning in Catch code # -Wno-keyword-macro: unit-tests use "#define private public" +# -Wno-deprecated-declarations: the library deprecated some functions # -Wno-weak-vtables: exception class is defined inline, but has virtual method # -Wno-range-loop-analysis: iterator_wrapper tests tests "for(const auto i...)" pedantic_clang: @@ -59,6 +60,7 @@ pedantic_clang: -Wno-documentation-unknown-command \ -Wno-exit-time-destructors \ -Wno-keyword-macro \ + -Wno-deprecated-declarations \ -Wno-weak-vtables \ -Wno-range-loop-analysis" @@ -66,6 +68,7 @@ pedantic_clang: pedantic_gcc: $(MAKE) json_unit CXX=g++ CXXFLAGS="\ -std=c++11 \ + -Wno-deprecated-declarations \ -Werror \ -Wall -Wpedantic -Wextra \ -Walloca \ diff --git a/src/json.hpp b/src/json.hpp index 37db5c45..3345cf2d 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -78,6 +78,15 @@ SOFTWARE. #pragma GCC diagnostic ignored "-Wdocumentation" #endif +// allow for portable deprecation warnings +#if defined(__clang__) || defined(__GNUC__) || defined(__GNUG__) + #define JSON_DEPRECATED __attribute__((deprecated)) +#elif defined(_MSC_VER) + #define JSON_DEPRECATED __declspec(deprecated) +#else + #define JSON_DEPRECATED +#endif + // allow to disable exceptions #if (defined(__cpp_exceptions) || defined(__EXCEPTIONS) || defined(_CPPUNWIND)) && not defined(JSON_NOEXCEPTION) #define JSON_THROW(exception) throw exception @@ -217,8 +226,8 @@ class parse_error : public exception const size_t byte; private: - parse_error(int id, size_t byte_, const char* what_arg) - : exception(id, what_arg), byte(byte_) + parse_error(int id_, size_t byte_, const char* what_arg) + : exception(id_, what_arg), byte(byte_) {} }; @@ -256,8 +265,8 @@ class invalid_iterator : public exception } private: - invalid_iterator(int id, const char* what_arg) - : exception(id, what_arg) + invalid_iterator(int id_, const char* what_arg) + : exception(id_, what_arg) {} }; @@ -295,8 +304,8 @@ class type_error : public exception } private: - type_error(int id, const char* what_arg) - : exception(id, what_arg) + type_error(int id_, const char* what_arg) + : exception(id_, what_arg) {} }; @@ -326,8 +335,8 @@ class out_of_range : public exception } private: - out_of_range(int id, const char* what_arg) - : exception(id, what_arg) + out_of_range(int id_, const char* what_arg) + : exception(id_, what_arg) {} }; @@ -352,8 +361,8 @@ class other_error : public exception } private: - other_error(int id, const char* what_arg) - : exception(id, what_arg) + other_error(int id_, const char* what_arg) + : exception(id_, what_arg) {} }; @@ -7083,8 +7092,12 @@ class basic_json /*! @brief serialize to stream - @copydoc operator<<(std::ostream&, const basic_json&) + @deprecated This stream operator is deprecated and will be removed in a + future version of the library. Please use + @ref std::ostream& operator<<(std::ostream&, const basic_json&) + instead; that is, replace calls like `j >> o;` with `o << j;`. */ + JSON_DEPRECATED friend std::ostream& operator>>(const basic_json& j, std::ostream& o) { return o << j; @@ -7357,6 +7370,20 @@ class basic_json return parse(std::begin(c), std::end(c), cb); } + /*! + @brief deserialize from stream + @deprecated This stream operator is deprecated and will be removed in a + future version of the library. Please use + @ref std::istream& operator>>(std::istream&, basic_json&) + instead; that is, replace calls like `j << i;` with `i >> j;`. + */ + JSON_DEPRECATED + friend std::istream& operator<<(basic_json& j, std::istream& i) + { + j = parser(i).parse(); + return i; + } + /*! @brief deserialize from stream @@ -7383,16 +7410,6 @@ class basic_json @since version 1.0.0 */ - friend std::istream& operator<<(basic_json& j, std::istream& i) - { - j = parser(i).parse(); - return i; - } - - /*! - @brief deserialize from stream - @copydoc operator<<(basic_json&, std::istream&) - */ friend std::istream& operator>>(std::istream& i, basic_json& j) { j = parser(i).parse(); @@ -8574,7 +8591,7 @@ class basic_json case 0x7f: // UTF-8 string (indefinite length) { std::string result; - while (check_length(v.size(), 1, idx), v[idx] != 0xff) + while (static_cast(check_length(v.size(), 1, idx)), v[idx] != 0xff) { string_t s = from_cbor_internal(v, idx); result += s; @@ -8670,7 +8687,7 @@ class basic_json case 0x9f: // array (indefinite length) { basic_json result = value_t::array; - while (check_length(v.size(), 1, idx), v[idx] != 0xff) + while (static_cast(check_length(v.size(), 1, idx)), v[idx] != 0xff) { result.push_back(from_cbor_internal(v, idx)); } @@ -8775,7 +8792,7 @@ class basic_json case 0xbf: // map (indefinite length) { basic_json result = value_t::object; - while (check_length(v.size(), 1, idx), v[idx] != 0xff) + while (static_cast(check_length(v.size(), 1, idx)), v[idx] != 0xff) { cbor_expect_string(v, idx); std::string key = from_cbor_internal(v, idx); @@ -13874,5 +13891,6 @@ inline nlohmann::json::json_pointer operator "" _json_pointer(const char* s, std #undef JSON_CATCH #undef JSON_THROW #undef JSON_TRY +#undef JSON_DEPRECATED #endif diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index cafb2067..60bb63c6 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -78,6 +78,15 @@ SOFTWARE. #pragma GCC diagnostic ignored "-Wdocumentation" #endif +// allow for portable deprecation warnings +#if defined(__clang__) || defined(__GNUC__) || defined(__GNUG__) + #define JSON_DEPRECATED __attribute__((deprecated)) +#elif defined(_MSC_VER) + #define JSON_DEPRECATED __declspec(deprecated) +#else + #define JSON_DEPRECATED +#endif + // allow to disable exceptions #if (defined(__cpp_exceptions) || defined(__EXCEPTIONS) || defined(_CPPUNWIND)) && not defined(JSON_NOEXCEPTION) #define JSON_THROW(exception) throw exception @@ -217,8 +226,8 @@ class parse_error : public exception const size_t byte; private: - parse_error(int id, size_t byte_, const char* what_arg) - : exception(id, what_arg), byte(byte_) + parse_error(int id_, size_t byte_, const char* what_arg) + : exception(id_, what_arg), byte(byte_) {} }; @@ -256,8 +265,8 @@ class invalid_iterator : public exception } private: - invalid_iterator(int id, const char* what_arg) - : exception(id, what_arg) + invalid_iterator(int id_, const char* what_arg) + : exception(id_, what_arg) {} }; @@ -295,8 +304,8 @@ class type_error : public exception } private: - type_error(int id, const char* what_arg) - : exception(id, what_arg) + type_error(int id_, const char* what_arg) + : exception(id_, what_arg) {} }; @@ -326,8 +335,8 @@ class out_of_range : public exception } private: - out_of_range(int id, const char* what_arg) - : exception(id, what_arg) + out_of_range(int id_, const char* what_arg) + : exception(id_, what_arg) {} }; @@ -352,8 +361,8 @@ class other_error : public exception } private: - other_error(int id, const char* what_arg) - : exception(id, what_arg) + other_error(int id_, const char* what_arg) + : exception(id_, what_arg) {} }; @@ -7083,8 +7092,12 @@ class basic_json /*! @brief serialize to stream - @copydoc operator<<(std::ostream&, const basic_json&) + @deprecated This stream operator is deprecated and will be removed in a + future version of the library. Please use + @ref std::ostream& operator<<(std::ostream&, const basic_json&) + instead; that is, replace calls like `j >> o;` with `o << j;`. */ + JSON_DEPRECATED friend std::ostream& operator>>(const basic_json& j, std::ostream& o) { return o << j; @@ -7357,6 +7370,20 @@ class basic_json return parse(std::begin(c), std::end(c), cb); } + /*! + @brief deserialize from stream + @deprecated This stream operator is deprecated and will be removed in a + future version of the library. Please use + @ref std::istream& operator>>(std::istream&, basic_json&) + instead; that is, replace calls like `j << i;` with `i >> j;`. + */ + JSON_DEPRECATED + friend std::istream& operator<<(basic_json& j, std::istream& i) + { + j = parser(i).parse(); + return i; + } + /*! @brief deserialize from stream @@ -7383,16 +7410,6 @@ class basic_json @since version 1.0.0 */ - friend std::istream& operator<<(basic_json& j, std::istream& i) - { - j = parser(i).parse(); - return i; - } - - /*! - @brief deserialize from stream - @copydoc operator<<(basic_json&, std::istream&) - */ friend std::istream& operator>>(std::istream& i, basic_json& j) { j = parser(i).parse(); @@ -8574,7 +8591,7 @@ class basic_json case 0x7f: // UTF-8 string (indefinite length) { std::string result; - while (check_length(v.size(), 1, idx), v[idx] != 0xff) + while (static_cast(check_length(v.size(), 1, idx)), v[idx] != 0xff) { string_t s = from_cbor_internal(v, idx); result += s; @@ -8670,7 +8687,7 @@ class basic_json case 0x9f: // array (indefinite length) { basic_json result = value_t::array; - while (check_length(v.size(), 1, idx), v[idx] != 0xff) + while (static_cast(check_length(v.size(), 1, idx)), v[idx] != 0xff) { result.push_back(from_cbor_internal(v, idx)); } @@ -8775,7 +8792,7 @@ class basic_json case 0xbf: // map (indefinite length) { basic_json result = value_t::object; - while (check_length(v.size(), 1, idx), v[idx] != 0xff) + while (static_cast(check_length(v.size(), 1, idx)), v[idx] != 0xff) { cbor_expect_string(v, idx); std::string key = from_cbor_internal(v, idx); @@ -12907,5 +12924,6 @@ inline nlohmann::json::json_pointer operator "" _json_pointer(const char* s, std #undef JSON_CATCH #undef JSON_THROW #undef JSON_TRY +#undef JSON_DEPRECATED #endif diff --git a/test/src/unit-regression.cpp b/test/src/unit-regression.cpp index a7b6e117..1e8f5c23 100644 --- a/test/src/unit-regression.cpp +++ b/test/src/unit-regression.cpp @@ -216,6 +216,8 @@ TEST_CASE("regression tests") { json a = {1, 2, 3}; json::reverse_iterator rit = ++a.rbegin(); + CHECK(*rit == json(2)); + CHECK(rit.value() == json(2)); } { json a = {1, 2, 3}; @@ -540,7 +542,7 @@ TEST_CASE("regression tests") CAPTURE(filename); json j; std::ifstream f(filename); - CHECK_NOTHROW(j << f); + CHECK_NOTHROW(f >> j); } } @@ -556,7 +558,7 @@ TEST_CASE("regression tests") CAPTURE(filename); json j; std::ifstream f(filename); - CHECK_NOTHROW(j << f); + CHECK_NOTHROW(f >> j); } } @@ -586,14 +588,14 @@ TEST_CASE("regression tests") std::stringstream ss; json j; ss << "123"; - CHECK_NOTHROW(j << ss); + CHECK_NOTHROW(ss >> j); // see https://github.com/nlohmann/json/issues/367#issuecomment-262841893: // ss is not at EOF; this yielded an error before the fix // (threw basic_string::append). No, it should just throw // a parse error because of the EOF. - CHECK_THROWS_AS(j << ss, json::parse_error); - CHECK_THROWS_WITH(j << ss, + CHECK_THROWS_AS(ss >> j, json::parse_error); + CHECK_THROWS_WITH(ss >> j, "[json.exception.parse_error.101] parse error at 1: parse error - unexpected end of input"); } diff --git a/test/src/unit-testsuites.cpp b/test/src/unit-testsuites.cpp index 8d6a8162..a4aa1956 100644 --- a/test/src/unit-testsuites.cpp +++ b/test/src/unit-testsuites.cpp @@ -79,7 +79,7 @@ TEST_CASE("compliance tests from json.org") CAPTURE(filename); json j; std::ifstream f(filename); - CHECK_THROWS_AS(j << f, json::parse_error); + CHECK_THROWS_AS(f >> j, json::parse_error); } } @@ -95,7 +95,7 @@ TEST_CASE("compliance tests from json.org") CAPTURE(filename); json j; std::ifstream f(filename); - CHECK_NOTHROW(j << f); + CHECK_NOTHROW(f >> j); } } } @@ -319,7 +319,7 @@ TEST_CASE("test suite from json-test-suite") // strings in a JSON array std::ifstream f("test/data/json_testsuite/sample.json"); json j; - CHECK_NOTHROW(j << f); + CHECK_NOTHROW(f >> j); // the array has 3 elements CHECK(j.size() == 3); @@ -334,35 +334,35 @@ TEST_CASE("json.org examples") { std::ifstream f("test/data/json.org/1.json"); json j; - CHECK_NOTHROW(j << f); + CHECK_NOTHROW(f >> j); } SECTION("2.json") { std::ifstream f("test/data/json.org/2.json"); json j; - CHECK_NOTHROW(j << f); + CHECK_NOTHROW(f >> j); } SECTION("3.json") { std::ifstream f("test/data/json.org/3.json"); json j; - CHECK_NOTHROW(j << f); + CHECK_NOTHROW(f >> j); } SECTION("4.json") { std::ifstream f("test/data/json.org/4.json"); json j; - CHECK_NOTHROW(j << f); + CHECK_NOTHROW(f >> j); } SECTION("5.json") { std::ifstream f("test/data/json.org/5.json"); json j; - CHECK_NOTHROW(j << f); + CHECK_NOTHROW(f >> j); } } @@ -545,7 +545,7 @@ TEST_CASE("nst's JSONTestSuite") CAPTURE(filename); std::ifstream f(filename); json j; - CHECK_NOTHROW(j << f); + CHECK_NOTHROW(f >> j); } } @@ -754,7 +754,7 @@ TEST_CASE("nst's JSONTestSuite") CAPTURE(filename); std::ifstream f(filename); json j; - CHECK_THROWS_AS(j << f, json::parse_error); + CHECK_THROWS_AS(f >> j, json::parse_error); } } @@ -777,7 +777,7 @@ TEST_CASE("nst's JSONTestSuite") CAPTURE(filename); std::ifstream f(filename); json j; - CHECK_NOTHROW(j << f); + CHECK_NOTHROW(f >> j); } } @@ -797,7 +797,7 @@ TEST_CASE("nst's JSONTestSuite") CAPTURE(filename); std::ifstream f(filename); json j; - CHECK_THROWS_AS(j << f, json::out_of_range); + CHECK_THROWS_AS(f >> j, json::out_of_range); } } @@ -824,7 +824,7 @@ TEST_CASE("nst's JSONTestSuite") CAPTURE(filename); std::ifstream f(filename); json j; - CHECK_THROWS_AS(j << f, json::parse_error); + CHECK_THROWS_AS(f >> j, json::parse_error); } } } @@ -851,7 +851,7 @@ TEST_CASE("Big List of Naughty Strings") { std::ifstream f("test/data/big-list-of-naughty-strings/blns.json"); json j; - CHECK_NOTHROW(j << f); + CHECK_NOTHROW(f >> j); } // check if parsed strings roundtrip diff --git a/test/src/unit-unicode.cpp b/test/src/unit-unicode.cpp index f7cf0ada..82324ca0 100644 --- a/test/src/unit-unicode.cpp +++ b/test/src/unit-unicode.cpp @@ -125,7 +125,7 @@ TEST_CASE("Unicode", "[hide]") // strings in a JSON array std::ifstream f("test/data/json_nlohmann_tests/all_unicode.json"); json j; - CHECK_NOTHROW(j << f); + CHECK_NOTHROW(f >> j); // the array has 1112064 + 1 elemnts (a terminating "null" value) // Note: 1112064 = 0x1FFFFF code points - 2048 invalid values between @@ -168,7 +168,7 @@ TEST_CASE("Unicode", "[hide]") // read a file with a UTF-8 BOM std::ifstream f("test/data/json_nlohmann_tests/bom.json"); json j; - CHECK_NOTHROW(j << f); + CHECK_NOTHROW(f >> j); } SECTION("error for incomplete/wrong BOM") From 60057a4728ac4a7a9826d14d3c15aba5213289e9 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Wed, 29 Mar 2017 07:54:26 +0200 Subject: [PATCH 127/530] :hammer: fixed bugs introduced by merging from develop --- src/json.hpp | 4 ++-- test/src/unit-regression.cpp | 2 +- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 903f3178..d2dbb7fe 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -7390,7 +7390,7 @@ class basic_json JSON_DEPRECATED friend std::istream& operator<<(basic_json& j, std::istream& i) { - j = parser(i).parse(); + j = parser(i).parse(true); return i; } @@ -7422,7 +7422,7 @@ class basic_json */ friend std::istream& operator>>(std::istream& i, basic_json& j) { - j = parser(i).parse(false); + j = parser(i).parse(true); return i; } diff --git a/test/src/unit-regression.cpp b/test/src/unit-regression.cpp index e8eae0f8..c221b1ab 100644 --- a/test/src/unit-regression.cpp +++ b/test/src/unit-regression.cpp @@ -596,7 +596,7 @@ TEST_CASE("regression tests") // a parse error because of the EOF. CHECK_THROWS_AS(ss >> j, json::parse_error); CHECK_THROWS_WITH(ss >> j, - "[json.exception.parse_error.101] parse error at 1: parse error - unexpected end of input"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected end of input"); } SECTION("issue #389 - Integer-overflow (OSS-Fuzz issue 267)") From 40aff4182f6e2b11bc8b0525aa965b7b261b3124 Mon Sep 17 00:00:00 2001 From: Lukas Barth Date: Thu, 30 Mar 2017 16:10:03 +0200 Subject: [PATCH 128/530] Pop for every push --- src/json.hpp | 3 +++ 1 file changed, 3 insertions(+) diff --git a/src/json.hpp b/src/json.hpp index 3345cf2d..90a48538 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -13886,6 +13886,9 @@ inline nlohmann::json::json_pointer operator "" _json_pointer(const char* s, std #if defined(__clang__) || defined(__GNUC__) || defined(__GNUG__) #pragma GCC diagnostic pop #endif +#if defined(__clang__) + #pragma GCC diagnostic pop +#endif // clean up #undef JSON_CATCH From 0b1b6307a52bc5347589f1c41c79f368ca7c1057 Mon Sep 17 00:00:00 2001 From: Lukas Barth Date: Thu, 30 Mar 2017 16:14:02 +0200 Subject: [PATCH 129/530] Also change the re2c file --- src/json.hpp.re2c | 3 +++ 1 file changed, 3 insertions(+) diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 60bb63c6..d88830b9 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -12919,6 +12919,9 @@ inline nlohmann::json::json_pointer operator "" _json_pointer(const char* s, std #if defined(__clang__) || defined(__GNUC__) || defined(__GNUG__) #pragma GCC diagnostic pop #endif +#if defined(__clang__) + #pragma GCC diagnostic pop +#endif // clean up #undef JSON_CATCH From d07596a0ea5e66e2677568d4c45b8d9147c9173b Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Thu, 30 Mar 2017 17:39:02 +0200 Subject: [PATCH 130/530] :memo: mentioned #540 and fixed #538 --- README.md | 8 +++++--- 1 file changed, 5 insertions(+), 3 deletions(-) diff --git a/README.md b/README.md index 803234d8..f8174fa8 100644 --- a/README.md +++ b/README.md @@ -517,9 +517,9 @@ namespace ns { } void from_json(const json& j, person& p) { - p.name = j["name"].get(); - p.address = j["address"].get(); - p.age = j["age"].get(); + p.name = j.at("name").get(); + p.address = j.at("address").get(); + p.age = j.at("age").get(); } } // namespace ns ``` @@ -531,6 +531,7 @@ Some important things: * Those methods **MUST** be in your type's namespace (which can be the global namespace), or the library will not be able to locate them (in this example, they are in namespace `ns`, where `person` is defined). * When using `get()`, `your_type` **MUST** be [DefaultConstructible](http://en.cppreference.com/w/cpp/concept/DefaultConstructible). (There is a way to bypass this requirement described later.) +* In function `from_json`, use function [`at()`](https://nlohmann.github.io/json/classnlohmann_1_1basic__json_a93403e803947b86f4da2d1fb3345cf2c.html#a93403e803947b86f4da2d1fb3345cf2c) to access the object values rather than `operator[]`. In case a key does not exists, `at` throws an exception that you can handle, whereas `operator[]` exhibits undefined behavior. #### How do I convert third-party types? @@ -833,6 +834,7 @@ I deeply appreciate the help of the following people. - [TedLyngmo](https://github.com/TedLyngmo) noted a typo in the README, removed unnecessary bit arithmetic, and fixed some `-Weffc++` warnings. - [Krzysztof Woś](https://github.com/krzysztofwos) made exceptions more visible. - [ftillier](https://github.com/ftillier) fixed a compiler warning. +- [tinloaf](https://github.com/tinloaf) made sure all pushed warnings are properly popped. Thanks a lot for helping out! Please [let me know](mailto:mail@nlohmann.me) if I forgot someone. From 4c821c9e9c842f957eb8100b642330639829b913 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Fri, 31 Mar 2017 17:11:51 +0200 Subject: [PATCH 131/530] :hammer: moved buffer size to interface --- src/json.hpp | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index d2dbb7fe..f3e20f06 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -10307,9 +10307,9 @@ class basic_json class cached_input_stream_adapter : public input_adapter { public: - cached_input_stream_adapter(std::istream& i) + cached_input_stream_adapter(std::istream& i, const size_t buffer_size) : is(i), start_position(is.tellg()), - buffer(1024 * 1024, std::char_traits::eof()) + buffer(buffer_size, std::char_traits::eof()) { // immediately abort if stream is erroneous if (JSON_UNLIKELY(i.fail())) @@ -10552,7 +10552,7 @@ class basic_json } explicit lexer(std::istream& i) - : ia(new cached_input_stream_adapter(i)), + : ia(new cached_input_stream_adapter(i, 1024 * 1024)), decimal_point_char(get_decimal_point()) {} From 50ee0a62f5eaf40d62c67115ebf58f959ab4dc60 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Fri, 31 Mar 2017 23:24:33 +0200 Subject: [PATCH 132/530] :hammer: replaced lookup-tables by switches --- src/json.hpp | 1314 ++++++++++++++++++++++---------- test/src/unit-class_parser.cpp | 2 +- 2 files changed, 921 insertions(+), 395 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index f3e20f06..51154ca1 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -10552,7 +10552,7 @@ class basic_json } explicit lexer(std::istream& i) - : ia(new cached_input_stream_adapter(i, 1024 * 1024)), + : ia(new cached_input_stream_adapter(i, 16384)), decimal_point_char(get_decimal_point()) {} @@ -10591,29 +10591,243 @@ class basic_json // must be called after \u was read; returns following xxxx as hex or -1 when error int get_codepoint() { - // a mapping to discover hex numbers - static int8_t ascii_to_hex[256] = {-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, -1, -1, -1, -1, -1, -1, -1, 10, 11, 12, 13, 14, 15, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, 10, 11, 12, 13, 14, 15, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1}; - + assert(current == 'u'); int codepoint = 0; - // check the next 4 bytes - for (size_t i = 0; i < 4; ++i) + switch (get()) { - const int8_t digit = ascii_to_hex[static_cast(get())]; - if (JSON_UNLIKELY(digit == -1)) - { + case '0': + break; + case '1': + codepoint += 0x1000; + break; + case '2': + codepoint += 0x2000; + break; + case '3': + codepoint += 0x3000; + break; + case '4': + codepoint += 0x4000; + break; + case '5': + codepoint += 0x5000; + break; + case '6': + codepoint += 0x6000; + break; + case '7': + codepoint += 0x7000; + break; + case '8': + codepoint += 0x8000; + break; + case '9': + codepoint += 0x9000; + break; + case 'A': + case 'a': + codepoint += 0xa000; + break; + case 'B': + case 'b': + codepoint += 0xb000; + break; + case 'C': + case 'c': + codepoint += 0xc000; + break; + case 'D': + case 'd': + codepoint += 0xd000; + break; + case 'E': + case 'e': + codepoint += 0xe000; + break; + case 'F': + case 'f': + codepoint += 0xf000; + break; + default: return -1; - } - else - { - codepoint += digit; - } + } - // except the last byte, result must be multiplied by 16 - if (i != 3) - { - codepoint <<= 4; - } + switch (get()) + { + case '0': + break; + case '1': + codepoint += 0x0100; + break; + case '2': + codepoint += 0x0200; + break; + case '3': + codepoint += 0x0300; + break; + case '4': + codepoint += 0x0400; + break; + case '5': + codepoint += 0x0500; + break; + case '6': + codepoint += 0x0600; + break; + case '7': + codepoint += 0x0700; + break; + case '8': + codepoint += 0x0800; + break; + case '9': + codepoint += 0x0900; + break; + case 'A': + case 'a': + codepoint += 0x0a00; + break; + case 'B': + case 'b': + codepoint += 0x0b00; + break; + case 'C': + case 'c': + codepoint += 0x0c00; + break; + case 'D': + case 'd': + codepoint += 0x0d00; + break; + case 'E': + case 'e': + codepoint += 0x0e00; + break; + case 'F': + case 'f': + codepoint += 0x0f00; + break; + default: + return -1; + } + + switch (get()) + { + case '0': + break; + case '1': + codepoint += 0x0010; + break; + case '2': + codepoint += 0x0020; + break; + case '3': + codepoint += 0x0030; + break; + case '4': + codepoint += 0x0040; + break; + case '5': + codepoint += 0x0050; + break; + case '6': + codepoint += 0x0060; + break; + case '7': + codepoint += 0x0070; + break; + case '8': + codepoint += 0x0080; + break; + case '9': + codepoint += 0x0090; + break; + case 'A': + case 'a': + codepoint += 0x00a0; + break; + case 'B': + case 'b': + codepoint += 0x00b0; + break; + case 'C': + case 'c': + codepoint += 0x00c0; + break; + case 'D': + case 'd': + codepoint += 0x00d0; + break; + case 'E': + case 'e': + codepoint += 0x00e0; + break; + case 'F': + case 'f': + codepoint += 0x00f0; + break; + default: + return -1; + } + + switch (get()) + { + case '0': + break; + case '1': + codepoint += 0x0001; + break; + case '2': + codepoint += 0x0002; + break; + case '3': + codepoint += 0x0003; + break; + case '4': + codepoint += 0x0004; + break; + case '5': + codepoint += 0x0005; + break; + case '6': + codepoint += 0x0006; + break; + case '7': + codepoint += 0x0007; + break; + case '8': + codepoint += 0x0008; + break; + case '9': + codepoint += 0x0009; + break; + case 'A': + case 'a': + codepoint += 0x000a; + break; + case 'B': + case 'b': + codepoint += 0x000b; + break; + case 'C': + case 'c': + codepoint += 0x000c; + break; + case 'D': + case 'd': + codepoint += 0x000d; + break; + case 'E': + case 'e': + codepoint += 0x000e; + break; + case 'F': + case 'f': + codepoint += 0x000f; + break; + default: + return -1; } return codepoint; @@ -10627,260 +10841,31 @@ class basic_json // we entered the function by reading an open quote assert(current == '\"'); - static unsigned char next[256] = {17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 17, 0, 0, 16, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 15, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 4, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 5, 2, 2, 6, 3, 3, 3, 7, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18}; - - // state variable - int state = -1; - - // whether the state is already set - bool state_set = false; - while (true) { // get next character get(); - // end of file while parsing string - if (JSON_UNLIKELY(current == std::char_traits::eof())) + switch (current) { - error_message = "invalid string: missing closing quote"; - return token_type::parse_error; - } - - // after coping with EOF, we only cope with bytes - //assert(0 <= current and current <= 255); - unsigned char ch = static_cast(current); - - // get next state - state = state_set ? state : next[ch]; - // reset variable - state_set = false; - - // 'add': 0, - // 'add_check1': 1, - // 'add_check2': 2, - // 'add_check3': 3, - // 'add_check_e0': 4, - // 'add_check_ed': 5, - // 'add_check_f0': 6, - // 'add_check_f4': 7, - // 'check1': 8, - // 'check2': 9, - // 'check3': 10, - // 'check_e0': 11, - // 'check_ed': 12, - // 'check_f0': 13, - // 'check_f4': 14, - // 'escape': 15, - // 'end': 16, - // 'error_invalid': 17, - // 'error_utf8': 18 - assert(0 <= state and state <= 18); - - switch (state) - { - // add - case 0: + // end of file while parsing string + case std::char_traits::eof(): { - add(current); - break; + error_message = "invalid string: missing closing quote"; + return token_type::parse_error; } - // add_check1 - case 1: + // closing quote + case '\"': { - add(current); - // next state is check1 - state = 8; - state_set = true; - break; + // terminate yytext + add('\0'); + --yylen; + return token_type::value_string; } - // add_check2 - case 2: - { - add(current); - // next state is check2 - state = 9; - state_set = true; - break; - } - - // add_check3 - case 3: - { - add(current); - // next state is check3 - state = 10; - state_set = true; - break; - } - - // add_check_e0 - case 4: - { - add(current); - // next state is check_e0 - state = 11; - state_set = true; - break; - } - - // add_check_ed - case 5: - { - add(current); - // next state is check_ed - state = 12; - state_set = true; - break; - } - - // add_check_f0 - case 6: - { - add(current); - // next state is check_f0 - state = 13; - state_set = true; - break; - } - - // add_check_f4 - case 7: - { - add(current); - // next state is check_f4 - state = 14; - state_set = true; - break; - } - - // check1 - case 8: - { - if (JSON_LIKELY(0x80 <= ch and ch <= 0xBF)) - { - add(current); - break; - } - else - { - error_message = "invalid string: not well-formed UTF-8 byte"; - return token_type::parse_error; - } - } - - // check2 - case 9: - { - if (JSON_LIKELY(0x80 <= ch and ch <= 0xBF)) - { - add(current); - // next state is check1 - state = 8; - state_set = true; - break; - } - else - { - error_message = "invalid string: not well-formed UTF-8 byte"; - return token_type::parse_error; - } - } - - // check3 - case 10: - { - if (JSON_LIKELY(0x80 <= ch and ch <= 0xBF)) - { - add(current); - // next state is check2 - state = 9; - state_set = true; - break; - } - else - { - error_message = "invalid string: not well-formed UTF-8 byte"; - return token_type::parse_error; - } - } - - // check_e0 - case 11: - { - if (JSON_LIKELY(0xA0 <= ch and ch <= 0xBF)) - { - add(current); - // next state is check1 - state = 8; - state_set = true; - break; - } - else - { - error_message = "invalid string: not well-formed UTF-8 byte"; - return token_type::parse_error; - } - } - - // check_ed - case 12: - { - if (JSON_LIKELY(0x80 <= ch and ch <= 0x9F)) - { - add(current); - // next state is check1 - state = 8; - state_set = true; - break; - } - else - { - error_message = "invalid string: not well-formed UTF-8 byte"; - return token_type::parse_error; - } - } - - // check_f0 - case 13: - { - if (JSON_LIKELY(0x90 <= ch and ch <= 0xBF)) - { - add(current); - // next state is check2 - state = 9; - state_set = true; - break; - } - else - { - error_message = "invalid string: not well-formed UTF-8 byte"; - return token_type::parse_error; - } - } - - // check_f4 - case 14: - { - if (JSON_LIKELY(0x80 <= ch and ch <= 0x8F)) - { - add(current); - // next state is check2 - state = 9; - state_set = true; - break; - } - else - { - error_message = "invalid string: not well-formed UTF-8 byte"; - return token_type::parse_error; - } - } - - // escape - case 15: + // escapes + case '\\': { switch (get()) { @@ -10935,7 +10920,7 @@ class basic_json // expect next \uxxxx entry if (JSON_LIKELY(get() == '\\' and get() == 'u')) { - int codepoint2 = get_codepoint(); + const int codepoint2 = get_codepoint(); if (JSON_UNLIKELY(codepoint2 == -1)) { @@ -11025,32 +11010,344 @@ class basic_json break; } - // end - case 16: - { - // terminate yytext - add('\0'); - --yylen; - return token_type::value_string; - } - - // error_invalid - case 17: + // invalid control characters + case '\x00': + case '\x01': + case '\x02': + case '\x03': + case '\x04': + case '\x05': + case '\x06': + case '\x07': + case '\x08': + case '\x09': + case '\x0a': + case '\x0b': + case '\x0c': + case '\x0d': + case '\x0e': + case '\x0f': + case '\x10': + case '\x11': + case '\x12': + case '\x13': + case '\x14': + case '\x15': + case '\x16': + case '\x17': + case '\x18': + case '\x19': + case '\x1a': + case '\x1b': + case '\x1c': + case '\x1d': + case '\x1e': + case '\x1f': { error_message = "invalid string: control characters (U+0000 through U+001f) must be escaped"; return token_type::parse_error; } - // error_utf8 - case 18: + // U+0020..U+007F (except U+0022 (quote) and U+005C (backspace)) + case '\x20': + case '\x21': + case '\x23': + case '\x24': + case '\x25': + case '\x26': + case '\x27': + case '\x28': + case '\x29': + case '\x2a': + case '\x2b': + case '\x2c': + case '\x2d': + case '\x2e': + case '\x2f': + case '\x30': + case '\x31': + case '\x32': + case '\x33': + case '\x34': + case '\x35': + case '\x36': + case '\x37': + case '\x38': + case '\x39': + case '\x3a': + case '\x3b': + case '\x3c': + case '\x3d': + case '\x3e': + case '\x3f': + case '\x40': + case '\x41': + case '\x42': + case '\x43': + case '\x44': + case '\x45': + case '\x46': + case '\x47': + case '\x48': + case '\x49': + case '\x4a': + case '\x4b': + case '\x4c': + case '\x4d': + case '\x4e': + case '\x4f': + case '\x50': + case '\x51': + case '\x52': + case '\x53': + case '\x54': + case '\x55': + case '\x56': + case '\x57': + case '\x58': + case '\x59': + case '\x5a': + case '\x5b': + case '\x5d': + case '\x5e': + case '\x5f': + case '\x60': + case '\x61': + case '\x62': + case '\x63': + case '\x64': + case '\x65': + case '\x66': + case '\x67': + case '\x68': + case '\x69': + case '\x6a': + case '\x6b': + case '\x6c': + case '\x6d': + case '\x6e': + case '\x6f': + case '\x70': + case '\x71': + case '\x72': + case '\x73': + case '\x74': + case '\x75': + case '\x76': + case '\x77': + case '\x78': + case '\x79': + case '\x7a': + case '\x7b': + case '\x7c': + case '\x7d': + case '\x7e': + case '\x7f': { + add(current); + break; + } + + // U+0080..U+07FF: bytes C2..DF 80..BF + case '\xc2': + case '\xc3': + case '\xc4': + case '\xc5': + case '\xc6': + case '\xc7': + case '\xc8': + case '\xc9': + case '\xca': + case '\xcb': + case '\xcc': + case '\xcd': + case '\xce': + case '\xcf': + case '\xd0': + case '\xd1': + case '\xd2': + case '\xd3': + case '\xd4': + case '\xd5': + case '\xd6': + case '\xd7': + case '\xd8': + case '\xd9': + case '\xda': + case '\xdb': + case '\xdc': + case '\xdd': + case '\xde': + case '\xdf': + { + add(current); + get(); + if (JSON_LIKELY('\x80' <= current and current <= '\xbf')) + { + add(current); + continue; + } + error_message = "invalid string: not well-formed UTF-8 byte"; return token_type::parse_error; } + // U+0800..U+0FFF: bytes E0 A0..BF 80..BF + case '\xe0': + { + add(current); + get(); + if (JSON_LIKELY('\xa0' <= current and current <= '\xbf')) + { + add(current); + get(); + if (JSON_LIKELY('\x80' <= current and current <= '\xbf')) + { + add(current); + continue; + } + } + + error_message = "invalid string: not well-formed UTF-8 byte"; + return token_type::parse_error; + } + + // U+1000..U+CFFF: bytes E1..EC 80..BF 80..BF + // U+E000..U+FFFF: bytes EE..EF 80..BF 80..BF + case '\xe1': + case '\xe2': + case '\xe3': + case '\xe4': + case '\xe5': + case '\xe6': + case '\xe7': + case '\xe8': + case '\xe9': + case '\xea': + case '\xeb': + case '\xec': + case '\xee': + case '\xef': + { + add(current); + get(); + if (JSON_LIKELY('\x80' <= current and current <= '\xbf')) + { + add(current); + get(); + if (JSON_LIKELY('\x80' <= current and current <= '\xbf')) + { + add(current); + continue; + } + } + + error_message = "invalid string: not well-formed UTF-8 byte"; + return token_type::parse_error; + } + + // U+D000..U+D7FF: bytes ED 80..9F 80..BF + case '\xed': + { + add(current); + get(); + if (JSON_LIKELY('\x80' <= current and current <= '\x9f')) + { + add(current); + get(); + if (JSON_LIKELY('\x80' <= current and current <= '\xbf')) + { + add(current); + continue; + } + } + + error_message = "invalid string: not well-formed UTF-8 byte"; + return token_type::parse_error; + } + + // U+10000..U+3FFFF F0 90..BF 80..BF 80..BF + case '\xf0': + { + add(current); + get(); + if (JSON_LIKELY('\x90' <= current and current <= '\xbf')) + { + add(current); + get(); + if (JSON_LIKELY('\x80' <= current and current <= '\xbf')) + { + add(current); + get(); + if (JSON_LIKELY('\x80' <= current and current <= '\xbf')) + { + add(current); + continue; + } + } + } + + error_message = "invalid string: not well-formed UTF-8 byte"; + return token_type::parse_error; + } + + // U+40000..U+FFFFF F1..F3 80..BF 80..BF 80..BF + case '\xf1': + case '\xf2': + case '\xf3': + { + add(current); + get(); + if (JSON_LIKELY('\x80' <= current and current <= '\xbf')) + { + add(current); + get(); + if (JSON_LIKELY('\x80' <= current and current <= '\xbf')) + { + add(current); + get(); + if (JSON_LIKELY('\x80' <= current and current <= '\xbf')) + { + add(current); + continue; + } + } + } + + error_message = "invalid string: not well-formed UTF-8 byte"; + return token_type::parse_error; + } + + // U+100000..U+10FFFF F4 80..8F 80..BF 80..BF + case '\xf4': + { + add(current); + get(); + if (JSON_LIKELY('\x80' <= current and current <= '\x8f')) + { + add(current); + get(); + if (JSON_LIKELY('\x80' <= current and current <= '\xbf')) + { + add(current); + get(); + if (JSON_LIKELY('\x80' <= current and current <= '\xbf')) + { + add(current); + continue; + } + } + } + + error_message = "invalid string: not well-formed UTF-8 byte"; + return token_type::parse_error; + } + + // remaining bytes (80..C1 and F5..FF) are not well-formed default: { - assert(false); // LCOV_EXCL_LINE + error_message = "invalid string: not well-formed UTF-8 byte"; + return token_type::parse_error; } } } @@ -11071,70 +11368,301 @@ class basic_json f = std::strtold(str, endptr); } + /*! + state | 0 | 1-9 | e E | + | - | . | anything + ---------|----------|----------|----------|---------|---------|----------|----------- + init | zero | any1 | [error] | [error] | minus | [error] | [error] + minus | zero | any1 | [error] | [error] | [error] | [error] | [error] + zero | done | done | exponent | done | done | decimal1 | done + any1 | any1 | any1 | exponent | done | done | decimal1 | done + decimal1 | decimal2 | [error] | [error] | [error] | [error] | [error] | [error] + decimal2 | decimal2 | decimal2 | exponent | done | done | done | done + exponent | any2 | any2 | [error] | sign | sign | [error] | [error] + sign | any2 | any2 | [error] | [error] | [error] | [error] | [error] + any2 | any2 | any2 | done | done | done | done | done + */ token_type scan_number() { - static unsigned char lookup[9][256] = - { - {10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 1, 10, 10, 3, 2, 2, 2, 2, 2, 2, 2, 2, 2, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10}, - {10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 3, 2, 2, 2, 2, 2, 2, 2, 2, 2, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10}, - {9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 4, 9, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 5, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 5, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9}, - {9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 4, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 5, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 5, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9}, - {10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10}, - {10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 8, 10, 8, 10, 10, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10}, - {9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 5, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 5, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9}, - {9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9}, - {10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10, 10} - }; - reset(); - bool has_sign = false; - bool has_exp = false; - bool has_point = false; + // the type of the parsed number; initially set to unsigned; will + // be changed if minus sign, decimal point or exponent is read + token_type number_type = token_type::value_unsigned; - int state = lookup[0][static_cast(current)]; - int old_state = 0; - - while (state != 9) + // state: we just found out we need to scan a number + switch (current) { - has_sign = has_sign or (state == 1); - has_point = has_point or (state == 4); - has_exp = has_exp or (state == 5); - - if (JSON_UNLIKELY(state == 10)) + case '-': { - // create error message based on previous state - switch (old_state) - { - case 0: - error_message = "invalid number; expected '-' or digit"; - break; - case 1: - error_message = "invalid number; expected digit after '-'"; - break; - case 4: - error_message = "invalid number; expected digit after '.'"; - break; - case 5: - error_message = "invalid number; expected '+', '-', or digit after exponent"; - break; - case 8: - error_message = "invalid number; expected digit after exponent sign"; - break; - default: - assert(false); // no error in the other states - break; - } - return token_type::parse_error; + add(current); + goto scan_number_minus; } - // add current character and fix decimal point - add((state == 4) ? decimal_point_char : current); - get(); - old_state = state; - state = lookup[state][static_cast(current)]; + case '0': + { + add(current); + goto scan_number_zero; + } + + case '1': + case '2': + case '3': + case '4': + case '5': + case '6': + case '7': + case '8': + case '9': + { + add(current); + goto scan_number_any1; + } + + default: + { + // all other characters are rejected outside scan_number() + assert(false); // LCOV_EXCL_LINE + } } +scan_number_minus: + // state: we just parsed a leading minus sign + number_type = token_type::value_integer; + switch (get()) + { + case '0': + { + add(current); + goto scan_number_zero; + } + + case '1': + case '2': + case '3': + case '4': + case '5': + case '6': + case '7': + case '8': + case '9': + { + add(current); + goto scan_number_any1; + } + + default: + { + error_message = "invalid number; expected digit after '-'"; + return token_type::parse_error; + } + } + +scan_number_zero: + // state: we just parse a zero (maybe with a leading minus sign) + switch (get()) + { + case '.': + { + add(decimal_point_char); + goto scan_number_decimal1; + } + + case 'e': + case 'E': + { + add(current); + goto scan_number_exponent; + } + + default: + { + goto scan_number_done; + } + } + +scan_number_any1: + // state: we just parsed a number 0-9 (maybe with a leading minus sign) + switch (get()) + { + case '0': + case '1': + case '2': + case '3': + case '4': + case '5': + case '6': + case '7': + case '8': + case '9': + { + add(current); + goto scan_number_any1; + } + + case '.': + { + add(decimal_point_char); + goto scan_number_decimal1; + } + + case 'e': + case 'E': + { + add(current); + goto scan_number_exponent; + } + + default: + { + goto scan_number_done; + } + } + +scan_number_decimal1: + // state: we just parsed a decimal point + number_type = token_type::value_float; + switch (get()) + { + case '0': + case '1': + case '2': + case '3': + case '4': + case '5': + case '6': + case '7': + case '8': + case '9': + { + add(current); + goto scan_number_decimal2; + } + + default: + { + error_message = "invalid number; expected digit after '.'"; + return token_type::parse_error; + } + } + +scan_number_decimal2: + // we just parsed at least one number after a decimal point + switch (get()) + { + case '0': + case '1': + case '2': + case '3': + case '4': + case '5': + case '6': + case '7': + case '8': + case '9': + { + add(current); + goto scan_number_decimal2; + } + + case 'e': + case 'E': + { + add(current); + goto scan_number_exponent; + } + + default: + { + goto scan_number_done; + } + } + +scan_number_exponent: + // we just parsed an exponent + number_type = token_type::value_float; + switch (get()) + { + case '+': + case '-': + { + add(current); + goto scan_number_sign; + } + + case '0': + case '1': + case '2': + case '3': + case '4': + case '5': + case '6': + case '7': + case '8': + case '9': + { + add(current); + goto scan_number_any2; + } + + default: + { + error_message = "invalid number; expected '+', '-', or digit after exponent"; + return token_type::parse_error; + } + } + +scan_number_sign: + // we just parsed an exponent sign + switch (get()) + { + case '0': + case '1': + case '2': + case '3': + case '4': + case '5': + case '6': + case '7': + case '8': + case '9': + { + add(current); + goto scan_number_any2; + } + + default: + { + error_message = "invalid number; expected digit after exponent sign"; + return token_type::parse_error; + } + } + +scan_number_any2: + // we just parsed a number after the exponent or exponent sign + switch (get()) + { + case '0': + case '1': + case '2': + case '3': + case '4': + case '5': + case '6': + case '7': + case '8': + case '9': + { + add(current); + goto scan_number_any2; + } + + default: + { + goto scan_number_done; + } + } + +scan_number_done: // unget the character after the number unget(); @@ -11143,30 +11671,42 @@ class basic_json --yylen; // try to parse integers first and fall back to floats - if (not has_exp and not has_point) + if (number_type == token_type::value_unsigned) { + char* endptr = nullptr; errno = 0; - if (has_sign) + const auto x = std::strtoull(yytext.data(), &endptr, 10); + + // we checked the number format before + assert(endptr == yytext.data() + yylen); + + if (errno == 0) { - char* endptr = nullptr; - const auto x = std::strtoll(yytext.data(), &endptr, 10); - value_integer = static_cast(x); - if (errno == 0 and endptr == yytext.data() + yylen and value_integer == x) - { - return token_type::value_integer; - } - } - else - { - char* endptr = nullptr; - const auto x = std::strtoull(yytext.data(), &endptr, 10); value_unsigned = static_cast(x); - if (errno == 0 and endptr == yytext.data() + yylen and value_unsigned == x) + if (value_unsigned == x) { return token_type::value_unsigned; } } } + else if (number_type == token_type::value_integer) + { + char* endptr = nullptr; + errno = 0; + const auto x = std::strtoll(yytext.data(), &endptr, 10); + + // we checked the number format before + assert(endptr == yytext.data() + yylen); + + if (errno == 0) + { + value_integer = static_cast(x); + if (value_integer == x) + { + return token_type::value_integer; + } + } + } strtof(value_float, yytext.data(), nullptr); return token_type::value_float; @@ -11223,17 +11763,9 @@ class basic_json int get() { ++chars_read; - - if (JSON_UNLIKELY(next_unget)) - { - next_unget = false; - } - else - { - current = ia->get_character(); - } - - return current; + return next_unget + ? (next_unget = false, current) + : (current = ia->get_character()); } /// unget a character to the input @@ -11317,7 +11849,6 @@ class basic_json } else { - // add character as is ss << c; } @@ -11441,8 +11972,7 @@ class basic_json public: /// a parser reading from a string literal parser(const char* buff, const parser_callback_t cb = nullptr) - : callback(cb), - m_lexer(buff, std::strlen(buff)) + : callback(cb), m_lexer(buff, std::strlen(buff)) {} /*! @@ -11528,22 +12058,11 @@ class basic_json return result; } - // no comma is expected here - unexpect(lexer::token_type::value_separator); - - // otherwise: parse key-value pairs - do + // parse values + while (true) { - // ugly, but could be fixed with loop reorganization - if (last_token == lexer::token_type::value_separator) - { - get_token(); - } - // store key expect(lexer::token_type::value_string); - // FIXME get_string returns const char*; maybe we can - // avoid this copy in the future const auto key = m_lexer.get_string(); bool keep_tag = false; @@ -11571,12 +12090,20 @@ class basic_json { result[key] = std::move(value); } - } - while (last_token == lexer::token_type::value_separator); - // closing } - expect(lexer::token_type::end_object); - get_token(); + // comma -> next value + if (last_token == lexer::token_type::value_separator) + { + get_token(); + continue; + } + + // closing } + expect(lexer::token_type::end_object); + get_token(); + break; + } + if (keep and callback and not callback(--depth, parse_event_t::object_end, result)) { result = basic_json(value_t::discarded); @@ -11609,30 +12136,29 @@ class basic_json return result; } - // no comma is expected here - unexpect(lexer::token_type::value_separator); - - // otherwise: parse values - do + // parse values + while (true) { - // ugly, but could be fixed with loop reorganization - if (last_token == lexer::token_type::value_separator) - { - get_token(); - } - // parse value auto value = parse_internal(keep); if (keep and not value.is_discarded()) { result.push_back(std::move(value)); } - } - while (last_token == lexer::token_type::value_separator); - // closing ] - expect(lexer::token_type::end_array); - get_token(); + // comma -> next value + if (last_token == lexer::token_type::value_separator) + { + get_token(); + continue; + } + + // closing ] + expect(lexer::token_type::end_array); + get_token(); + break; + } + if (keep and callback and not callback(--depth, parse_event_t::array_end, result)) { result = basic_json(value_t::discarded); @@ -11728,7 +12254,7 @@ class basic_json */ void expect(typename lexer::token_type t) const { - if (t != last_token) + if (JSON_UNLIKELY(t != last_token)) { std::string error_msg = "syntax error - "; if (last_token == lexer::token_type::parse_error) @@ -11750,7 +12276,7 @@ class basic_json */ void unexpect(typename lexer::token_type t) const { - if (t == last_token) + if (JSON_UNLIKELY(t == last_token)) { std::string error_msg = "syntax error - "; if (last_token == lexer::token_type::parse_error) diff --git a/test/src/unit-class_parser.cpp b/test/src/unit-class_parser.cpp index f36eb900..864b7be1 100644 --- a/test/src/unit-class_parser.cpp +++ b/test/src/unit-class_parser.cpp @@ -659,7 +659,7 @@ TEST_CASE("parser class") // test case to make sure no comma preceeds the first key CHECK_THROWS_AS(json::parser("{,\"key\": false}").parse(), json::parse_error); CHECK_THROWS_WITH(json::parser("{,\"key\": false}").parse(), - "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected ','"); + "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected ','; expected string literal"); // test case to make sure an object is properly closed CHECK_THROWS_AS(json::parser("[{\"key\": false true]").parse(), json::parse_error); CHECK_THROWS_WITH(json::parser("[{\"key\": false true]").parse(), From 3fa0610856c9eca53665c3cf2d0bc8b06b6bbb64 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Fri, 31 Mar 2017 23:57:24 +0200 Subject: [PATCH 133/530] :hammer: added minibench to ignore list --- .gitignore | 3 +++ 1 file changed, 3 insertions(+) diff --git a/.gitignore b/.gitignore index 69a81cbd..58ced04e 100644 --- a/.gitignore +++ b/.gitignore @@ -30,3 +30,6 @@ test/parse_afl_fuzzer test/parse_cbor_fuzzer test/parse_msgpack_fuzzer + +minibench + From c32d2e5b3c735c16eda5673c6cce5bf56effbf20 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sat, 1 Apr 2017 00:36:05 +0200 Subject: [PATCH 134/530] :hammer: removed unget() function --- src/json.hpp | 10 ++-------- 1 file changed, 2 insertions(+), 8 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index ae3dab6f..4198b140 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -11664,7 +11664,8 @@ scan_number_any2: scan_number_done: // unget the character after the number - unget(); + --chars_read; + next_unget = true; // terminate token add('\0'); @@ -11768,13 +11769,6 @@ scan_number_done: : (current = ia->get_character()); } - /// unget a character to the input - void unget() noexcept - { - --chars_read; - next_unget = true; - } - /// add a character to yytext void add(int c) { From 3a5cf9bd0a58aed6f26ed0802f938d0ed348f75e Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sat, 1 Apr 2017 08:34:58 +0200 Subject: [PATCH 135/530] :hammer: improved code coverage --- src/json.hpp | 10 ++++------ test/src/unit-class_lexer.cpp | 9 +++++++++ 2 files changed, 13 insertions(+), 6 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 4198b140..1c00e12a 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -10965,6 +10965,9 @@ class basic_json codepoint = codepoint1; } + // result of the above calculation yields a proper codepoint + assert(0x00 <= codepoint and codepoint <= 0x10FFFF); + // translate code point to bytes if (codepoint < 0x80) { @@ -10984,7 +10987,7 @@ class basic_json add(0x80 | ((codepoint >> 6) & 0x3F)); add(0x80 | (codepoint & 0x3F)); } - else if (codepoint <= 0x10ffff) + else { // 4-byte characters: 11110xxx 10xxxxxx 10xxxxxx 10xxxxxx add(0xF0 | (codepoint >> 18)); @@ -10992,11 +10995,6 @@ class basic_json add(0x80 | ((codepoint >> 6) & 0x3F)); add(0x80 | (codepoint & 0x3F)); } - else - { - error_message = "invalid string: code points above U+10FFFF are invalid"; - return token_type::parse_error; - } break; } diff --git a/test/src/unit-class_lexer.cpp b/test/src/unit-class_lexer.cpp index 468e1f52..2acea176 100644 --- a/test/src/unit-class_lexer.cpp +++ b/test/src/unit-class_lexer.cpp @@ -158,6 +158,15 @@ TEST_CASE("lexer class") } } + SECTION("very large string") + { + // strings larger than 1024 bytes yield a resize of the lexer's yytext buffer + std::string s("\""); + s += std::string(2048, 'x'); + s += "\""; + CHECK((json::lexer(s.c_str(), 2050).scan() == json::lexer::token_type::value_string)); + } + /* NOTE: to_unicode function has been removed SECTION("to_unicode") { From af9b21151ca704414ff82156e1a7eca98e3e62b1 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Th=C3=A9o=20DELRIEU?= Date: Sun, 2 Apr 2017 13:10:15 +0200 Subject: [PATCH 136/530] add enum class default conversions --- src/json.hpp | 25 ++++++++++--------------- src/json.hpp.re2c | 25 ++++++++++--------------- test/src/unit-conversions.cpp | 9 +++++++++ 3 files changed, 29 insertions(+), 30 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 90a48538..5df1bacc 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -454,12 +454,6 @@ using enable_if_t = typename std::enable_if::type; template using uncvref_t = typename std::remove_cv::type>::type; -// taken from http://stackoverflow.com/a/26936864/266378 -template -using is_unscoped_enum = - std::integral_constant::value and - std::is_enum::value>; - /* Implementation of two C++17 constructs: conjunction, negation. This is needed to avoid evaluating all the traits in a condition @@ -818,11 +812,12 @@ void to_json(BasicJsonType& j, CompatibleNumberIntegerType val) noexcept external_constructor::construct(j, static_cast(val)); } -template::value, int> = 0> -void to_json(BasicJsonType& j, UnscopedEnumType e) noexcept +template::value, int> = 0> +void to_json(BasicJsonType& j, EnumType e) noexcept { - external_constructor::construct(j, e); + using underlying_type = typename std::underlying_type::type; + external_constructor::construct(j, static_cast(e)); } template @@ -937,13 +932,13 @@ void from_json(const BasicJsonType& j, typename BasicJsonType::number_integer_t& get_arithmetic_value(j, val); } -template::value, int> = 0> -void from_json(const BasicJsonType& j, UnscopedEnumType& e) +template::value, int> = 0> +void from_json(const BasicJsonType& j, EnumType& e) { - typename std::underlying_type::type val; + typename std::underlying_type::type val; get_arithmetic_value(j, val); - e = static_cast(val); + e = static_cast(val); } template diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index d88830b9..da503495 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -454,12 +454,6 @@ using enable_if_t = typename std::enable_if::type; template using uncvref_t = typename std::remove_cv::type>::type; -// taken from http://stackoverflow.com/a/26936864/266378 -template -using is_unscoped_enum = - std::integral_constant::value and - std::is_enum::value>; - /* Implementation of two C++17 constructs: conjunction, negation. This is needed to avoid evaluating all the traits in a condition @@ -818,11 +812,12 @@ void to_json(BasicJsonType& j, CompatibleNumberIntegerType val) noexcept external_constructor::construct(j, static_cast(val)); } -template::value, int> = 0> -void to_json(BasicJsonType& j, UnscopedEnumType e) noexcept +template::value, int> = 0> +void to_json(BasicJsonType& j, EnumType e) noexcept { - external_constructor::construct(j, e); + using underlying_type = typename std::underlying_type::type; + external_constructor::construct(j, static_cast(e)); } template @@ -937,13 +932,13 @@ void from_json(const BasicJsonType& j, typename BasicJsonType::number_integer_t& get_arithmetic_value(j, val); } -template::value, int> = 0> -void from_json(const BasicJsonType& j, UnscopedEnumType& e) +template::value, int> = 0> +void from_json(const BasicJsonType& j, EnumType& e) { - typename std::underlying_type::type val; + typename std::underlying_type::type val; get_arithmetic_value(j, val); - e = static_cast(val); + e = static_cast(val); } template diff --git a/test/src/unit-conversions.cpp b/test/src/unit-conversions.cpp index 59969bf9..0940ec51 100644 --- a/test/src/unit-conversions.cpp +++ b/test/src/unit-conversions.cpp @@ -917,6 +917,15 @@ TEST_CASE("value conversion") } } + SECTION("get an enum") + { + enum c_enum { value_1, value_2 }; + enum class cpp_enum { value_1, value_2 }; + + CHECK(json(value_1).get() == value_1); + CHECK(json(cpp_enum::value_1).get() == cpp_enum::value_1); + } + SECTION("more involved conversions") { SECTION("object-like STL containers") From 89efe627fe58abd6e1c17d0cd89fea787a45ee01 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 2 Apr 2017 18:46:21 +0200 Subject: [PATCH 137/530] :hammer: a lot of restructuring - removed uncached input stream adapter; it was too slow anyway - implemented a class binary_read which parses CBOR based on input adapters - in the CBOR parser, numbers are created via memcpy to avoid undefined behavior --- Makefile | 6 +- src/json.hpp | 1311 ++++++++++++++++++++++++---------- test/src/unit-cbor.cpp | 32 +- test/src/unit-regression.cpp | 22 +- 4 files changed, 948 insertions(+), 423 deletions(-) diff --git a/Makefile b/Makefile index 179df3aa..138a85b6 100644 --- a/Makefile +++ b/Makefile @@ -47,7 +47,8 @@ doctest: # -Wno-keyword-macro: unit-tests use "#define private public" # -Wno-deprecated-declarations: the library deprecated some functions # -Wno-weak-vtables: exception class is defined inline, but has virtual method -# -Wno-range-loop-analysis: iterator_wrapper tests tests "for(const auto i...)" +# -Wno-range-loop-analysis: iterator_wrapper tests "for(const auto i...)" +# -Wno-float-equal: not all comparisons in the tests can be replaced by Approx pedantic_clang: $(MAKE) json_unit CXXFLAGS="\ -std=c++11 \ @@ -58,7 +59,8 @@ pedantic_clang: -Wno-keyword-macro \ -Wno-deprecated-declarations \ -Wno-weak-vtables \ - -Wno-range-loop-analysis" + -Wno-range-loop-analysis \ + -Wno-float-equal" # calling GCC with most warnings pedantic_gcc: diff --git a/src/json.hpp b/src/json.hpp index 1c00e12a..e96a53e3 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -38,7 +38,7 @@ SOFTWARE. #include // nullptr_t, ptrdiff_t, size_t #include // int64_t, uint64_t #include // abort, strtod, strtof, strtold, strtoul, strtoll, strtoull -#include // strlen +#include // memcpy, strlen #include // forward_list #include // function, hash, less #include // initializer_list @@ -9228,12 +9228,12 @@ class basic_json @since version 2.0.9, parameter @a start_index since 2.1.1 */ - static basic_json from_cbor(const std::vector& v, - const size_t start_index = 0) - { - size_t i = start_index; - return from_cbor_internal(v, i); - } + //static basic_json from_cbor(const std::vector& v, + // const size_t start_index = 0) + //{ + // size_t i = start_index; + // return from_cbor_internal(v, i); + //} /// @} @@ -10281,10 +10281,727 @@ class basic_json private: + //////////////////// + // input adapters // + //////////////////// + + /// abstract input adapter interface + class input_adapter + { + public: + virtual int get_character() = 0; + virtual std::string read(size_t offset, size_t length) = 0; + virtual ~input_adapter() {} + }; + + /// input adapter for cached stream input + class cached_input_stream_adapter : public input_adapter + { + public: + cached_input_stream_adapter(std::istream& i, const size_t buffer_size) + : is(i), start_position(is.tellg()), + buffer(buffer_size, std::char_traits::eof()) + { + // immediately abort if stream is erroneous + if (JSON_UNLIKELY(i.fail())) + { + JSON_THROW(parse_error::create(111, 0, "bad input stream")); + } + + // initial fill; unfilled buffer characters remain EOF + is.read(buffer.data(), static_cast(buffer.size())); + + // skip byte-order mark + assert(buffer.size() >= 3); + if (buffer[0] == '\xEF' and buffer[1] == '\xBB' and buffer[2] == '\xBF') + { + buffer_pos += 3; + processed_chars += 3; + } + } + + ~cached_input_stream_adapter() override + { + // clear stream flags + is.clear(); + // set stream after last processed char + is.seekg(start_position + static_cast(processed_chars - 1)); + } + + int get_character() override + { + // check if refilling is necessary + if (JSON_UNLIKELY(buffer_pos == buffer.size())) + { + // refill + is.read(reinterpret_cast(buffer.data()), static_cast(buffer.size())); + // set unfilled characters to EOF + std::fill_n(buffer.begin() + static_cast(is.gcount()), + buffer.size() - static_cast(is.gcount()), + std::char_traits::eof()); + // the buffer is ready + buffer_pos = 0; + } + + ++processed_chars; + const int res = buffer[buffer_pos++]; + return (res == std::char_traits::eof()) ? res : res & 0xFF; + } + + std::string read(size_t offset, size_t length) override + { + // create buffer + std::string result(length, '\0'); + + // save stream position + auto current_pos = is.tellg(); + // save stream flags + auto flags = is.rdstate(); + + // clear stream flags + is.clear(); + // set stream position + is.seekg(static_cast(offset)); + // read bytes + is.read(&result[0], static_cast(length)); + + // reset stream position + is.seekg(current_pos); + // reset stream flags + is.setstate(flags); + + return result; + } + + private: + /// the associated input stream + std::istream& is; + + /// chars returned via get_character() + size_t processed_chars = 0; + /// chars processed in the current buffer + size_t buffer_pos = 0; + + /// position of the stream when we started + const std::streampos start_position; + + /// internal buffer + std::vector buffer; + }; + + /// input adapter for buffer input + class input_buffer_adapter : public input_adapter + { + public: + input_buffer_adapter(const char* b, size_t l) + : input_adapter(), cursor(b), limit(b + l), start(b) + {} + + // delete because of pointer members + input_buffer_adapter(const input_buffer_adapter&) = delete; + input_buffer_adapter& operator=(input_buffer_adapter&) = delete; + + int get_character() override + { + if (JSON_LIKELY(cursor < limit)) + { + return *(cursor++) & 0xFF; + } + else + { + return std::char_traits::eof(); + } + } + + std::string read(size_t offset, size_t length) override + { + // avoid reading too many characters + const size_t max_length = static_cast(limit - start); + return std::string(start + offset, std::min({length, max_length})); + } + + private: + /// pointer to the current character + const char* cursor; + /// pointer past the last character + const char* limit; + /// pointer to the first character + const char* start; + }; + + //////////////////// + // binary formats // + //////////////////// + + private: + class binary_reader + { + public: + explicit binary_reader(std::istream& i) + : ia(new cached_input_stream_adapter(i, 16384)) + {} + + binary_reader(const char* buff, const size_t len) + : ia(new input_buffer_adapter(buff, len)) + {} + + ~binary_reader() + { + delete ia; + } + + // switch off unwanted functions (due to pointer members) + binary_reader(const binary_reader&) = delete; + binary_reader operator=(const binary_reader&) = delete; + + /*! + @param[in] get_char whether a new character should be retrieved from + the input (true, default) or whether the last + read character should be considered instead + */ + basic_json parse_cbor(const bool get_char = true) + { + switch (get_char ? get() : current) + { + // EOF + case std::char_traits::eof(): + { + JSON_THROW(parse_error::create(110, chars_read, "unexpected end of input")); + } + + // Integer 0x00..0x17 (0..23) + case 0x00: + case 0x01: + case 0x02: + case 0x03: + case 0x04: + case 0x05: + case 0x06: + case 0x07: + case 0x08: + case 0x09: + case 0x0a: + case 0x0b: + case 0x0c: + case 0x0d: + case 0x0e: + case 0x0f: + case 0x10: + case 0x11: + case 0x12: + case 0x13: + case 0x14: + case 0x15: + case 0x16: + case 0x17: + { + return static_cast(current); + } + + case 0x18: // Unsigned integer (one-byte uint8_t follows) + { + return get_number(); + } + + case 0x19: // Unsigned integer (two-byte uint16_t follows) + { + return get_number(); + } + + case 0x1a: // Unsigned integer (four-byte uint32_t follows) + { + return get_number(); + } + + case 0x1b: // Unsigned integer (eight-byte uint64_t follows) + { + return get_number(); + } + + // Negative integer -1-0x00..-1-0x17 (-1..-24) + case 0x20: + case 0x21: + case 0x22: + case 0x23: + case 0x24: + case 0x25: + case 0x26: + case 0x27: + case 0x28: + case 0x29: + case 0x2a: + case 0x2b: + case 0x2c: + case 0x2d: + case 0x2e: + case 0x2f: + case 0x30: + case 0x31: + case 0x32: + case 0x33: + case 0x34: + case 0x35: + case 0x36: + case 0x37: + { + return static_cast(0x20 - 1 - current); + } + + case 0x38: // Negative integer (one-byte uint8_t follows) + { + // must be uint8_t ! + return static_cast(-1) - get_number(); + } + + case 0x39: // Negative integer -1-n (two-byte uint16_t follows) + { + return static_cast(-1) - get_number(); + } + + case 0x3a: // Negative integer -1-n (four-byte uint32_t follows) + { + return static_cast(-1) - get_number(); + } + + case 0x3b: // Negative integer -1-n (eight-byte uint64_t follows) + { + return static_cast(-1) - static_cast(get_number()); + } + + // UTF-8 string (0x00..0x17 bytes follow) + case 0x60: + case 0x61: + case 0x62: + case 0x63: + case 0x64: + case 0x65: + case 0x66: + case 0x67: + case 0x68: + case 0x69: + case 0x6a: + case 0x6b: + case 0x6c: + case 0x6d: + case 0x6e: + case 0x6f: + case 0x70: + case 0x71: + case 0x72: + case 0x73: + case 0x74: + case 0x75: + case 0x76: + case 0x77: + case 0x78: // UTF-8 string (one-byte uint8_t for n follows) + case 0x79: // UTF-8 string (two-byte uint16_t for n follow) + case 0x7a: // UTF-8 string (four-byte uint32_t for n follow) + case 0x7b: // UTF-8 string (eight-byte uint64_t for n follow) + case 0x7f: // UTF-8 string (indefinite length) + { + return get_cbor_string(); + } + + // array (0x00..0x17 data items follow) + case 0x80: + case 0x81: + case 0x82: + case 0x83: + case 0x84: + case 0x85: + case 0x86: + case 0x87: + case 0x88: + case 0x89: + case 0x8a: + case 0x8b: + case 0x8c: + case 0x8d: + case 0x8e: + case 0x8f: + case 0x90: + case 0x91: + case 0x92: + case 0x93: + case 0x94: + case 0x95: + case 0x96: + case 0x97: + { + basic_json result = value_t::array; + const auto len = static_cast(current - 0x80); + for (size_t i = 0; i < len; ++i) + { + result.push_back(parse_cbor()); + } + return result; + } + + case 0x98: // array (one-byte uint8_t for n follows) + { + basic_json result = value_t::array; + const auto len = static_cast(get_number()); + for (size_t i = 0; i < len; ++i) + { + result.push_back(parse_cbor()); + } + return result; + } + + case 0x99: // array (two-byte uint16_t for n follow) + { + basic_json result = value_t::array; + const auto len = static_cast(get_number()); + for (size_t i = 0; i < len; ++i) + { + result.push_back(parse_cbor()); + } + return result; + } + + case 0x9a: // array (four-byte uint32_t for n follow) + { + basic_json result = value_t::array; + const auto len = static_cast(get_number()); + for (size_t i = 0; i < len; ++i) + { + result.push_back(parse_cbor()); + } + return result; + } + + case 0x9b: // array (eight-byte uint64_t for n follow) + { + basic_json result = value_t::array; + const auto len = static_cast(get_number()); + for (size_t i = 0; i < len; ++i) + { + result.push_back(parse_cbor()); + } + return result; + } + + case 0x9f: // array (indefinite length) + { + basic_json result = value_t::array; + while (get() != 0xff) + { + result.push_back(parse_cbor(false)); + } + return result; + } + + // map (0x00..0x17 pairs of data items follow) + case 0xa0: + case 0xa1: + case 0xa2: + case 0xa3: + case 0xa4: + case 0xa5: + case 0xa6: + case 0xa7: + case 0xa8: + case 0xa9: + case 0xaa: + case 0xab: + case 0xac: + case 0xad: + case 0xae: + case 0xaf: + case 0xb0: + case 0xb1: + case 0xb2: + case 0xb3: + case 0xb4: + case 0xb5: + case 0xb6: + case 0xb7: + { + basic_json result = value_t::object; + const auto len = static_cast(current - 0xa0); + for (size_t i = 0; i < len; ++i) + { + get(); + result[get_cbor_string()] = parse_cbor(); + } + return result; + } + + case 0xb8: // map (one-byte uint8_t for n follows) + { + basic_json result = value_t::object; + const auto len = static_cast(get_number()); + for (size_t i = 0; i < len; ++i) + { + get(); + result[get_cbor_string()] = parse_cbor(); + } + return result; + } + + case 0xb9: // map (two-byte uint16_t for n follow) + { + basic_json result = value_t::object; + const auto len = static_cast(get_number()); + for (size_t i = 0; i < len; ++i) + { + get(); + result[get_cbor_string()] = parse_cbor(); + } + return result; + } + + case 0xba: // map (four-byte uint32_t for n follow) + { + basic_json result = value_t::object; + const auto len = static_cast(get_number()); + for (size_t i = 0; i < len; ++i) + { + get(); + result[get_cbor_string()] = parse_cbor(); + } + return result; + } + + case 0xbb: // map (eight-byte uint64_t for n follow) + { + basic_json result = value_t::object; + const auto len = static_cast(get_number()); + for (size_t i = 0; i < len; ++i) + { + get(); + result[get_cbor_string()] = parse_cbor(); + } + return result; + } + + case 0xbf: // map (indefinite length) + { + basic_json result = value_t::object; + while (get() != 0xff) + { + result[get_cbor_string()] = parse_cbor(); + } + return result; + } + + case 0xf4: // false + { + return false; + } + + case 0xf5: // true + { + return true; + } + + case 0xf6: // null + { + return value_t::null; + } + + case 0xf9: // Half-Precision Float (two-byte IEEE 754) + { + const int byte1 = get(); + check_eof(); + const int byte2 = get(); + check_eof(); + + // code from RFC 7049, Appendix D, Figure 3: + // As half-precision floating-point numbers were only added to + // IEEE 754 in 2008, today's programming platforms often still + // only have limited support for them. It is very easy to + // include at least decoding support for them even without such + // support. An example of a small decoder for half-precision + // floating-point numbers in the C language is shown in Fig. 3. + const int half = (byte1 << 8) + byte2; + const int exp = (half >> 10) & 0x1f; + const int mant = half & 0x3ff; + double val; + if (exp == 0) + { + val = std::ldexp(mant, -24); + } + else if (exp != 31) + { + val = std::ldexp(mant + 1024, exp - 25); + } + else + { + val = mant == 0 + ? std::numeric_limits::infinity() + : std::numeric_limits::quiet_NaN(); + } + return (half & 0x8000) != 0 ? -val : val; + } + + case 0xfa: // Single-Precision Float (four-byte IEEE 754) + { + return get_number(); + } + + case 0xfb: // Double-Precision Float (eight-byte IEEE 754) + { + return get_number(); + } + + default: // anything else (0xFF is handled inside the other types) + { + std::stringstream ss; + ss << std::hex << current; + JSON_THROW(parse_error::create(112, chars_read, "error reading CBOR; last byte: 0x" + ss.str())); + } + } + } + + private: + int get() + { + ++chars_read; + return (current = ia->get_character()); + } + + // todo: check if this breaks with endianess + template + T get_number() + { + std::array vec; + for (size_t i = 0; i < sizeof(T); ++i) + { + get(); + check_eof(); + vec[sizeof(T) - i - 1] = static_cast(current); + } + + T result; + std::memcpy(&result, vec.data(), sizeof(T)); + return result; + } + + std::string get_string(const size_t len) + { + std::string result; + for (size_t i = 0; i < len; ++i) + { + get(); + check_eof(); + result.append(1, static_cast(current)); + } + return result; + } + + std::string get_cbor_string() + { + check_eof(); + + switch (current) + { + // UTF-8 string (0x00..0x17 bytes follow) + case 0x60: + case 0x61: + case 0x62: + case 0x63: + case 0x64: + case 0x65: + case 0x66: + case 0x67: + case 0x68: + case 0x69: + case 0x6a: + case 0x6b: + case 0x6c: + case 0x6d: + case 0x6e: + case 0x6f: + case 0x70: + case 0x71: + case 0x72: + case 0x73: + case 0x74: + case 0x75: + case 0x76: + case 0x77: + { + const auto len = static_cast(current - 0x60); + return get_string(len); + } + + case 0x78: // UTF-8 string (one-byte uint8_t for n follows) + { + const auto len = static_cast(get_number()); + return get_string(len); + } + + case 0x79: // UTF-8 string (two-byte uint16_t for n follow) + { + const auto len = static_cast(get_number()); + return get_string(len); + } + + case 0x7a: // UTF-8 string (four-byte uint32_t for n follow) + { + const auto len = static_cast(get_number()); + return get_string(len); + } + + case 0x7b: // UTF-8 string (eight-byte uint64_t for n follow) + { + const auto len = static_cast(get_number()); + return get_string(len); + } + + case 0x7f: // UTF-8 string (indefinite length) + { + std::string result; + while (get() != 0xff) + { + check_eof(); + result.append(1, static_cast(current)); + } + return result; + } + + default: + { + std::stringstream ss; + ss << std::hex << current; + JSON_THROW(parse_error::create(113, chars_read, "expected a CBOR string; last byte: 0x" + ss.str())); + } + } + } + + void check_eof() + { + if (JSON_UNLIKELY(current == std::char_traits::eof())) + { + JSON_THROW(parse_error::create(110, chars_read, "unexpected end of input")); + } + } + + private: + /// input adapter + input_adapter* ia = nullptr; + + /// the current character + int current = std::char_traits::eof(); + + /// the number of characters read + size_t chars_read = 0; + }; + + public: + static basic_json from_cbor(const std::vector& v, + const size_t start_index = 0) + { + binary_reader br(reinterpret_cast(v.data() + start_index), v.size() - start_index); + return br.parse_cbor(); + } + ////////////////////// // lexer and parser // ////////////////////// + private: /*! @brief lexical analysis @@ -10292,200 +11009,6 @@ class basic_json */ class lexer { - private: - - /// abstract input adapter interface - class input_adapter - { - public: - virtual int get_character() = 0; - virtual std::string read(size_t offset, size_t length) = 0; - virtual ~input_adapter() {} - }; - - /// input adapter for cached stream input - class cached_input_stream_adapter : public input_adapter - { - public: - cached_input_stream_adapter(std::istream& i, const size_t buffer_size) - : is(i), start_position(is.tellg()), - buffer(buffer_size, std::char_traits::eof()) - { - // immediately abort if stream is erroneous - if (JSON_UNLIKELY(i.fail())) - { - JSON_THROW(parse_error::create(111, 0, "bad input stream")); - } - - // initial fill; unfilled buffer characters remain EOF - is.read(buffer.data(), static_cast(buffer.size())); - - // skip byte-order mark - assert(buffer.size() >= 3); - if (buffer[0] == '\xEF' and buffer[1] == '\xBB' and buffer[2] == '\xBF') - { - buffer_pos += 3; - processed_chars += 3; - } - } - - ~cached_input_stream_adapter() override - { - // clear stream flags - is.clear(); - // set stream after last processed char - is.seekg(start_position + static_cast(processed_chars - 1)); - } - - int get_character() override - { - // check if refilling is necessary - if (JSON_UNLIKELY(buffer_pos == buffer.size())) - { - // refill - is.read(reinterpret_cast(buffer.data()), static_cast(buffer.size())); - // set unfilled characters to EOF - std::fill_n(buffer.begin() + static_cast(is.gcount()), - buffer.size() - static_cast(is.gcount()), - std::char_traits::eof()); - // the buffer is ready - buffer_pos = 0; - } - - ++processed_chars; - return buffer[buffer_pos++]; - } - - std::string read(size_t offset, size_t length) override - { - // create buffer - std::string result(length, '\0'); - - // save stream position - auto current_pos = is.tellg(); - // save stream flags - auto flags = is.rdstate(); - - // clear stream flags - is.clear(); - // set stream position - is.seekg(static_cast(offset)); - // read bytes - is.read(&result[0], static_cast(length)); - - // reset stream position - is.seekg(current_pos); - // reset stream flags - is.setstate(flags); - - return result; - } - - private: - /// the associated input stream - std::istream& is; - - /// chars returned via get_character() - size_t processed_chars = 0; - /// chars processed in the current buffer - size_t buffer_pos = 0; - - /// position of the stream when we started - const std::streampos start_position; - - /// internal buffer - std::vector buffer; - }; - - /// input adapter for uncached stream input - class input_stream_adapter : public input_adapter - { - public: - input_stream_adapter(std::istream& i) - : is(i) - { - // immediately abort if stream is erroneous - if (i.fail()) - { - JSON_THROW(parse_error::create(111, 0, "bad input stream")); - } - } - - int get_character() override - { - return is.get(); - } - - std::string read(size_t offset, size_t length) override - { - // create buffer - std::string result(length, '\0'); - - // save stream position - auto current_pos = is.tellg(); - // save stream flags - auto flags = is.rdstate(); - - // clear stream flags - is.clear(); - // set stream position - is.seekg(offset); - // read bytes - is.read(&result[0], length); - - // reset stream position - is.seekg(current_pos); - // reset stream flags - is.setstate(flags); - - return result; - } - - private: - /// the associated input stream - std::istream& is; - }; - - /// input adapter for buffer input - class input_buffer_adapter : public input_adapter - { - public: - input_buffer_adapter(const char* b, size_t l) - : input_adapter(), cursor(b), limit(b + l), start(b) - {} - - // delete because of pointer members - input_buffer_adapter(const input_buffer_adapter&) = delete; - input_buffer_adapter& operator=(input_buffer_adapter&) = delete; - - int get_character() override - { - if (JSON_LIKELY(cursor < limit)) - { - return *cursor++; - } - else - { - return std::char_traits::eof(); - } - } - - std::string read(size_t offset, size_t length) override - { - // avoid reading too many characters - const size_t max_length = static_cast(limit - start); - return std::string(start + offset, std::min({length, max_length})); - } - - private: - /// pointer to the current character - const char* cursor; - /// pointer past the last character - const char* limit; - /// pointer to the first character - const char* start; - }; - public: /// token types for the parser enum class token_type @@ -11009,178 +11532,178 @@ class basic_json } // invalid control characters - case '\x00': - case '\x01': - case '\x02': - case '\x03': - case '\x04': - case '\x05': - case '\x06': - case '\x07': - case '\x08': - case '\x09': - case '\x0a': - case '\x0b': - case '\x0c': - case '\x0d': - case '\x0e': - case '\x0f': - case '\x10': - case '\x11': - case '\x12': - case '\x13': - case '\x14': - case '\x15': - case '\x16': - case '\x17': - case '\x18': - case '\x19': - case '\x1a': - case '\x1b': - case '\x1c': - case '\x1d': - case '\x1e': - case '\x1f': + case 0x00: + case 0x01: + case 0x02: + case 0x03: + case 0x04: + case 0x05: + case 0x06: + case 0x07: + case 0x08: + case 0x09: + case 0x0a: + case 0x0b: + case 0x0c: + case 0x0d: + case 0x0e: + case 0x0f: + case 0x10: + case 0x11: + case 0x12: + case 0x13: + case 0x14: + case 0x15: + case 0x16: + case 0x17: + case 0x18: + case 0x19: + case 0x1a: + case 0x1b: + case 0x1c: + case 0x1d: + case 0x1e: + case 0x1f: { error_message = "invalid string: control characters (U+0000 through U+001f) must be escaped"; return token_type::parse_error; } // U+0020..U+007F (except U+0022 (quote) and U+005C (backspace)) - case '\x20': - case '\x21': - case '\x23': - case '\x24': - case '\x25': - case '\x26': - case '\x27': - case '\x28': - case '\x29': - case '\x2a': - case '\x2b': - case '\x2c': - case '\x2d': - case '\x2e': - case '\x2f': - case '\x30': - case '\x31': - case '\x32': - case '\x33': - case '\x34': - case '\x35': - case '\x36': - case '\x37': - case '\x38': - case '\x39': - case '\x3a': - case '\x3b': - case '\x3c': - case '\x3d': - case '\x3e': - case '\x3f': - case '\x40': - case '\x41': - case '\x42': - case '\x43': - case '\x44': - case '\x45': - case '\x46': - case '\x47': - case '\x48': - case '\x49': - case '\x4a': - case '\x4b': - case '\x4c': - case '\x4d': - case '\x4e': - case '\x4f': - case '\x50': - case '\x51': - case '\x52': - case '\x53': - case '\x54': - case '\x55': - case '\x56': - case '\x57': - case '\x58': - case '\x59': - case '\x5a': - case '\x5b': - case '\x5d': - case '\x5e': - case '\x5f': - case '\x60': - case '\x61': - case '\x62': - case '\x63': - case '\x64': - case '\x65': - case '\x66': - case '\x67': - case '\x68': - case '\x69': - case '\x6a': - case '\x6b': - case '\x6c': - case '\x6d': - case '\x6e': - case '\x6f': - case '\x70': - case '\x71': - case '\x72': - case '\x73': - case '\x74': - case '\x75': - case '\x76': - case '\x77': - case '\x78': - case '\x79': - case '\x7a': - case '\x7b': - case '\x7c': - case '\x7d': - case '\x7e': - case '\x7f': + case 0x20: + case 0x21: + case 0x23: + case 0x24: + case 0x25: + case 0x26: + case 0x27: + case 0x28: + case 0x29: + case 0x2a: + case 0x2b: + case 0x2c: + case 0x2d: + case 0x2e: + case 0x2f: + case 0x30: + case 0x31: + case 0x32: + case 0x33: + case 0x34: + case 0x35: + case 0x36: + case 0x37: + case 0x38: + case 0x39: + case 0x3a: + case 0x3b: + case 0x3c: + case 0x3d: + case 0x3e: + case 0x3f: + case 0x40: + case 0x41: + case 0x42: + case 0x43: + case 0x44: + case 0x45: + case 0x46: + case 0x47: + case 0x48: + case 0x49: + case 0x4a: + case 0x4b: + case 0x4c: + case 0x4d: + case 0x4e: + case 0x4f: + case 0x50: + case 0x51: + case 0x52: + case 0x53: + case 0x54: + case 0x55: + case 0x56: + case 0x57: + case 0x58: + case 0x59: + case 0x5a: + case 0x5b: + case 0x5d: + case 0x5e: + case 0x5f: + case 0x60: + case 0x61: + case 0x62: + case 0x63: + case 0x64: + case 0x65: + case 0x66: + case 0x67: + case 0x68: + case 0x69: + case 0x6a: + case 0x6b: + case 0x6c: + case 0x6d: + case 0x6e: + case 0x6f: + case 0x70: + case 0x71: + case 0x72: + case 0x73: + case 0x74: + case 0x75: + case 0x76: + case 0x77: + case 0x78: + case 0x79: + case 0x7a: + case 0x7b: + case 0x7c: + case 0x7d: + case 0x7e: + case 0x7f: { add(current); break; } // U+0080..U+07FF: bytes C2..DF 80..BF - case '\xc2': - case '\xc3': - case '\xc4': - case '\xc5': - case '\xc6': - case '\xc7': - case '\xc8': - case '\xc9': - case '\xca': - case '\xcb': - case '\xcc': - case '\xcd': - case '\xce': - case '\xcf': - case '\xd0': - case '\xd1': - case '\xd2': - case '\xd3': - case '\xd4': - case '\xd5': - case '\xd6': - case '\xd7': - case '\xd8': - case '\xd9': - case '\xda': - case '\xdb': - case '\xdc': - case '\xdd': - case '\xde': - case '\xdf': + case 0xc2: + case 0xc3: + case 0xc4: + case 0xc5: + case 0xc6: + case 0xc7: + case 0xc8: + case 0xc9: + case 0xca: + case 0xcb: + case 0xcc: + case 0xcd: + case 0xce: + case 0xcf: + case 0xd0: + case 0xd1: + case 0xd2: + case 0xd3: + case 0xd4: + case 0xd5: + case 0xd6: + case 0xd7: + case 0xd8: + case 0xd9: + case 0xda: + case 0xdb: + case 0xdc: + case 0xdd: + case 0xde: + case 0xdf: { add(current); get(); - if (JSON_LIKELY('\x80' <= current and current <= '\xbf')) + if (JSON_LIKELY(0x80 <= current and current <= 0xbf)) { add(current); continue; @@ -11191,15 +11714,15 @@ class basic_json } // U+0800..U+0FFF: bytes E0 A0..BF 80..BF - case '\xe0': + case 0xe0: { add(current); get(); - if (JSON_LIKELY('\xa0' <= current and current <= '\xbf')) + if (JSON_LIKELY(0xa0 <= current and current <= 0xbf)) { add(current); get(); - if (JSON_LIKELY('\x80' <= current and current <= '\xbf')) + if (JSON_LIKELY(0x80 <= current and current <= 0xbf)) { add(current); continue; @@ -11212,28 +11735,28 @@ class basic_json // U+1000..U+CFFF: bytes E1..EC 80..BF 80..BF // U+E000..U+FFFF: bytes EE..EF 80..BF 80..BF - case '\xe1': - case '\xe2': - case '\xe3': - case '\xe4': - case '\xe5': - case '\xe6': - case '\xe7': - case '\xe8': - case '\xe9': - case '\xea': - case '\xeb': - case '\xec': - case '\xee': - case '\xef': + case 0xe1: + case 0xe2: + case 0xe3: + case 0xe4: + case 0xe5: + case 0xe6: + case 0xe7: + case 0xe8: + case 0xe9: + case 0xea: + case 0xeb: + case 0xec: + case 0xee: + case 0xef: { add(current); get(); - if (JSON_LIKELY('\x80' <= current and current <= '\xbf')) + if (JSON_LIKELY(0x80 <= current and current <= 0xbf)) { add(current); get(); - if (JSON_LIKELY('\x80' <= current and current <= '\xbf')) + if (JSON_LIKELY(0x80 <= current and current <= 0xbf)) { add(current); continue; @@ -11245,15 +11768,15 @@ class basic_json } // U+D000..U+D7FF: bytes ED 80..9F 80..BF - case '\xed': + case 0xed: { add(current); get(); - if (JSON_LIKELY('\x80' <= current and current <= '\x9f')) + if (JSON_LIKELY(0x80 <= current and current <= 0x9f)) { add(current); get(); - if (JSON_LIKELY('\x80' <= current and current <= '\xbf')) + if (JSON_LIKELY(0x80 <= current and current <= 0xbf)) { add(current); continue; @@ -11265,19 +11788,19 @@ class basic_json } // U+10000..U+3FFFF F0 90..BF 80..BF 80..BF - case '\xf0': + case 0xf0: { add(current); get(); - if (JSON_LIKELY('\x90' <= current and current <= '\xbf')) + if (JSON_LIKELY(0x90 <= current and current <= 0xbf)) { add(current); get(); - if (JSON_LIKELY('\x80' <= current and current <= '\xbf')) + if (JSON_LIKELY(0x80 <= current and current <= 0xbf)) { add(current); get(); - if (JSON_LIKELY('\x80' <= current and current <= '\xbf')) + if (JSON_LIKELY(0x80 <= current and current <= 0xbf)) { add(current); continue; @@ -11290,21 +11813,21 @@ class basic_json } // U+40000..U+FFFFF F1..F3 80..BF 80..BF 80..BF - case '\xf1': - case '\xf2': - case '\xf3': + case 0xf1: + case 0xf2: + case 0xf3: { add(current); get(); - if (JSON_LIKELY('\x80' <= current and current <= '\xbf')) + if (JSON_LIKELY(0x80 <= current and current <= 0xbf)) { add(current); get(); - if (JSON_LIKELY('\x80' <= current and current <= '\xbf')) + if (JSON_LIKELY(0x80 <= current and current <= 0xbf)) { add(current); get(); - if (JSON_LIKELY('\x80' <= current and current <= '\xbf')) + if (JSON_LIKELY(0x80 <= current and current <= 0xbf)) { add(current); continue; @@ -11317,19 +11840,19 @@ class basic_json } // U+100000..U+10FFFF F4 80..8F 80..BF 80..BF - case '\xf4': + case 0xf4: { add(current); get(); - if (JSON_LIKELY('\x80' <= current and current <= '\x8f')) + if (JSON_LIKELY(0x80 <= current and current <= 0x8f)) { add(current); get(); - if (JSON_LIKELY('\x80' <= current and current <= '\xbf')) + if (JSON_LIKELY(0x80 <= current and current <= 0xbf)) { add(current); get(); - if (JSON_LIKELY('\x80' <= current and current <= '\xbf')) + if (JSON_LIKELY(0x80 <= current and current <= 0xbf)) { add(current); continue; diff --git a/test/src/unit-cbor.cpp b/test/src/unit-cbor.cpp index f1609f4f..390a1b52 100644 --- a/test/src/unit-cbor.cpp +++ b/test/src/unit-cbor.cpp @@ -1166,35 +1166,35 @@ TEST_CASE("CBOR") CHECK_THROWS_AS(json::from_cbor(std::vector({0x1b, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00})), json::parse_error); CHECK_THROWS_WITH(json::from_cbor(std::vector({0x18})), - "[json.exception.parse_error.110] parse error at 2: cannot read 1 bytes from vector"); + "[json.exception.parse_error.110] parse error at 2: unexpected end of input"); CHECK_THROWS_WITH(json::from_cbor(std::vector({0x19})), - "[json.exception.parse_error.110] parse error at 2: cannot read 2 bytes from vector"); + "[json.exception.parse_error.110] parse error at 2: unexpected end of input"); CHECK_THROWS_WITH(json::from_cbor(std::vector({0x19, 0x00})), - "[json.exception.parse_error.110] parse error at 2: cannot read 2 bytes from vector"); + "[json.exception.parse_error.110] parse error at 3: unexpected end of input"); CHECK_THROWS_WITH(json::from_cbor(std::vector({0x1a})), - "[json.exception.parse_error.110] parse error at 2: cannot read 4 bytes from vector"); + "[json.exception.parse_error.110] parse error at 2: unexpected end of input"); CHECK_THROWS_WITH(json::from_cbor(std::vector({0x1a, 0x00})), - "[json.exception.parse_error.110] parse error at 2: cannot read 4 bytes from vector"); + "[json.exception.parse_error.110] parse error at 3: unexpected end of input"); CHECK_THROWS_WITH(json::from_cbor(std::vector({0x1a, 0x00, 0x00})), - "[json.exception.parse_error.110] parse error at 2: cannot read 4 bytes from vector"); + "[json.exception.parse_error.110] parse error at 4: unexpected end of input"); CHECK_THROWS_WITH(json::from_cbor(std::vector({0x1a, 0x00, 0x00, 0x00})), - "[json.exception.parse_error.110] parse error at 2: cannot read 4 bytes from vector"); + "[json.exception.parse_error.110] parse error at 5: unexpected end of input"); CHECK_THROWS_WITH(json::from_cbor(std::vector({0x1b})), - "[json.exception.parse_error.110] parse error at 2: cannot read 8 bytes from vector"); + "[json.exception.parse_error.110] parse error at 2: unexpected end of input"); CHECK_THROWS_WITH(json::from_cbor(std::vector({0x1b, 0x00})), - "[json.exception.parse_error.110] parse error at 2: cannot read 8 bytes from vector"); + "[json.exception.parse_error.110] parse error at 3: unexpected end of input"); CHECK_THROWS_WITH(json::from_cbor(std::vector({0x1b, 0x00, 0x00})), - "[json.exception.parse_error.110] parse error at 2: cannot read 8 bytes from vector"); + "[json.exception.parse_error.110] parse error at 4: unexpected end of input"); CHECK_THROWS_WITH(json::from_cbor(std::vector({0x1b, 0x00, 0x00, 0x00})), - "[json.exception.parse_error.110] parse error at 2: cannot read 8 bytes from vector"); + "[json.exception.parse_error.110] parse error at 5: unexpected end of input"); CHECK_THROWS_WITH(json::from_cbor(std::vector({0x1b, 0x00, 0x00, 0x00, 0x00})), - "[json.exception.parse_error.110] parse error at 2: cannot read 8 bytes from vector"); + "[json.exception.parse_error.110] parse error at 6: unexpected end of input"); CHECK_THROWS_WITH(json::from_cbor(std::vector({0x1b, 0x00, 0x00, 0x00, 0x00, 0x00})), - "[json.exception.parse_error.110] parse error at 2: cannot read 8 bytes from vector"); + "[json.exception.parse_error.110] parse error at 7: unexpected end of input"); CHECK_THROWS_WITH(json::from_cbor(std::vector({0x1b, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00})), - "[json.exception.parse_error.110] parse error at 2: cannot read 8 bytes from vector"); + "[json.exception.parse_error.110] parse error at 8: unexpected end of input"); CHECK_THROWS_WITH(json::from_cbor(std::vector({0x1b, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00})), - "[json.exception.parse_error.110] parse error at 2: cannot read 8 bytes from vector"); + "[json.exception.parse_error.110] parse error at 9: unexpected end of input"); } SECTION("unsupported bytes") @@ -1756,7 +1756,7 @@ TEST_CASE("examples from RFC 7049 Appendix A") CHECK(json::parse("\"\\ud800\\udd51\"") == json::from_cbor(std::vector({0x64, 0xf0, 0x90, 0x85, 0x91}))); // indefinite length strings - CHECK(json::parse("\"streaming\"") == json::from_cbor(std::vector({0x7f, 0x65, 0x73, 0x74, 0x72, 0x65, 0x61, 0x64, 0x6d, 0x69, 0x6e, 0x67, 0xff}))); + CHECK(json::parse("\"streaming\"") == json::from_cbor(std::vector({0x7f, 0x73, 0x74, 0x72, 0x65, 0x61, 0x6d, 0x69, 0x6e, 0x67, 0xff}))); } SECTION("arrays") diff --git a/test/src/unit-regression.cpp b/test/src/unit-regression.cpp index c221b1ab..fb4f03d1 100644 --- a/test/src/unit-regression.cpp +++ b/test/src/unit-regression.cpp @@ -629,7 +629,7 @@ TEST_CASE("regression tests") std::vector vec {0x65, 0xf5, 0x0a, 0x48, 0x21}; CHECK_THROWS_AS(json::from_cbor(vec), json::parse_error); CHECK_THROWS_WITH(json::from_cbor(vec), - "[json.exception.parse_error.110] parse error at 2: cannot read 5 bytes from vector"); + "[json.exception.parse_error.110] parse error at 6: unexpected end of input"); } SECTION("issue #407 - Heap-buffer-overflow (OSS-Fuzz issue 343)") @@ -650,19 +650,19 @@ TEST_CASE("regression tests") std::vector vec3 {0xf9, 0x8f}; CHECK_THROWS_AS(json::from_cbor(vec3), json::parse_error); CHECK_THROWS_WITH(json::from_cbor(vec3), - "[json.exception.parse_error.110] parse error at 2: cannot read 2 bytes from vector"); + "[json.exception.parse_error.110] parse error at 3: unexpected end of input"); // related test case: incomplete Single-Precision Float (CBOR) std::vector vec4 {0xfa, 0x8f, 0x0a}; CHECK_THROWS_AS(json::from_cbor(vec4), json::parse_error); CHECK_THROWS_WITH(json::from_cbor(vec4), - "[json.exception.parse_error.110] parse error at 2: cannot read 4 bytes from vector"); + "[json.exception.parse_error.110] parse error at 4: unexpected end of input"); // related test case: incomplete Double-Precision Float (CBOR) std::vector vec5 {0xfb, 0x8f, 0x0a}; CHECK_THROWS_AS(json::from_cbor(vec5), json::parse_error); CHECK_THROWS_WITH(json::from_cbor(vec5), - "[json.exception.parse_error.110] parse error at 2: cannot read 8 bytes from vector"); + "[json.exception.parse_error.110] parse error at 4: unexpected end of input"); } SECTION("issue #408 - Heap-buffer-overflow (OSS-Fuzz issue 344)") @@ -705,7 +705,7 @@ TEST_CASE("regression tests") std::vector vec2; CHECK_THROWS_AS(json::from_cbor(vec2), json::parse_error); CHECK_THROWS_WITH(json::from_cbor(vec2), - "[json.exception.parse_error.110] parse error at 1: cannot read 1 bytes from vector"); + "[json.exception.parse_error.110] parse error at 1: unexpected end of input"); CHECK_THROWS_AS(json::from_msgpack(vec2), json::parse_error); CHECK_THROWS_WITH(json::from_msgpack(vec2), "[json.exception.parse_error.110] parse error at 1: cannot read 1 bytes from vector"); @@ -717,19 +717,19 @@ TEST_CASE("regression tests") std::vector vec1 {0x7f}; CHECK_THROWS_AS(json::from_cbor(vec1), json::parse_error); CHECK_THROWS_WITH(json::from_cbor(vec1), - "[json.exception.parse_error.110] parse error at 2: cannot read 1 bytes from vector"); + "[json.exception.parse_error.110] parse error at 2: unexpected end of input"); // related test case: empty array (indefinite length) std::vector vec2 {0x9f}; CHECK_THROWS_AS(json::from_cbor(vec2), json::parse_error); CHECK_THROWS_WITH(json::from_cbor(vec2), - "[json.exception.parse_error.110] parse error at 2: cannot read 1 bytes from vector"); + "[json.exception.parse_error.110] parse error at 2: unexpected end of input"); // related test case: empty map (indefinite length) std::vector vec3 {0xbf}; CHECK_THROWS_AS(json::from_cbor(vec3), json::parse_error); CHECK_THROWS_WITH(json::from_cbor(vec3), - "[json.exception.parse_error.110] parse error at 2: cannot read 1 bytes from vector"); + "[json.exception.parse_error.110] parse error at 2: unexpected end of input"); } SECTION("issue #412 - Heap-buffer-overflow (OSS-Fuzz issue 367)") @@ -763,19 +763,19 @@ TEST_CASE("regression tests") std::vector vec1 {0x7f, 0x61, 0x61}; CHECK_THROWS_AS(json::from_cbor(vec1), json::parse_error); CHECK_THROWS_WITH(json::from_cbor(vec1), - "[json.exception.parse_error.110] parse error at 4: cannot read 1 bytes from vector"); + "[json.exception.parse_error.110] parse error at 4: unexpected end of input"); // related test case: nonempty array (indefinite length) std::vector vec2 {0x9f, 0x01}; CHECK_THROWS_AS(json::from_cbor(vec2), json::parse_error); CHECK_THROWS_WITH(json::from_cbor(vec2), - "[json.exception.parse_error.110] parse error at 3: cannot read 1 bytes from vector"); + "[json.exception.parse_error.110] parse error at 3: unexpected end of input"); // related test case: nonempty map (indefinite length) std::vector vec3 {0xbf, 0x61, 0x61, 0x01}; CHECK_THROWS_AS(json::from_cbor(vec3), json::parse_error); CHECK_THROWS_WITH(json::from_cbor(vec3), - "[json.exception.parse_error.110] parse error at 5: cannot read 1 bytes from vector"); + "[json.exception.parse_error.110] parse error at 5: unexpected end of input"); } SECTION("issue #414 - compare with literal 0)") From ea667db446649a9beb5689e6e5b860831d06323e Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Mon, 3 Apr 2017 21:53:23 +0200 Subject: [PATCH 138/530] :hammer: fixed compilation errors TIL about C++: executing m[key()]=val(); for a std::map executes First key() then val() with Clang First val() then key() with GCC #wat --- src/json.hpp | 18 ++++++++++++------ 1 file changed, 12 insertions(+), 6 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index e96a53e3..9fadcae8 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -10722,7 +10722,8 @@ class basic_json for (size_t i = 0; i < len; ++i) { get(); - result[get_cbor_string()] = parse_cbor(); + auto key = get_cbor_string(); + result[key] = parse_cbor(); } return result; } @@ -10734,7 +10735,8 @@ class basic_json for (size_t i = 0; i < len; ++i) { get(); - result[get_cbor_string()] = parse_cbor(); + auto key = get_cbor_string(); + result[key] = parse_cbor(); } return result; } @@ -10746,7 +10748,8 @@ class basic_json for (size_t i = 0; i < len; ++i) { get(); - result[get_cbor_string()] = parse_cbor(); + auto key = get_cbor_string(); + result[key] = parse_cbor(); } return result; } @@ -10758,7 +10761,8 @@ class basic_json for (size_t i = 0; i < len; ++i) { get(); - result[get_cbor_string()] = parse_cbor(); + auto key = get_cbor_string(); + result[key] = parse_cbor(); } return result; } @@ -10770,7 +10774,8 @@ class basic_json for (size_t i = 0; i < len; ++i) { get(); - result[get_cbor_string()] = parse_cbor(); + auto key = get_cbor_string(); + result[key] = parse_cbor(); } return result; } @@ -10780,7 +10785,8 @@ class basic_json basic_json result = value_t::object; while (get() != 0xff) { - result[get_cbor_string()] = parse_cbor(); + auto key = get_cbor_string(); + result[key] = parse_cbor(); } return result; } From 782570d46fffc54ff1a98fc3045a7b9e509138b5 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Mon, 3 Apr 2017 22:58:46 +0200 Subject: [PATCH 139/530] :hammer: cleanup --- src/json.hpp | 683 +++++++-------------------------------------------- 1 file changed, 89 insertions(+), 594 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 9fadcae8..91a83731 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -8397,505 +8397,6 @@ class basic_json } } - /*! - @brief create a JSON value from a given CBOR vector - - @param[in] v CBOR serialization - @param[in] idx byte index to start reading from @a v - - @return deserialized JSON value - - @throw parse_error.110 if the given vector ends prematurely - @throw parse_error.112 if unsupported features from CBOR were - used in the given vector @a v or if the input is not valid CBOR - @throw parse_error.113 if a string was expected as map key, but not found - - @sa https://tools.ietf.org/html/rfc7049 - */ - static basic_json from_cbor_internal(const std::vector& v, size_t& idx) - { - // store and increment index - const size_t current_idx = idx++; - - // make sure reading 1 byte is safe - check_length(v.size(), 1, current_idx); - - switch (v[current_idx]) - { - // Integer 0x00..0x17 (0..23) - case 0x00: - case 0x01: - case 0x02: - case 0x03: - case 0x04: - case 0x05: - case 0x06: - case 0x07: - case 0x08: - case 0x09: - case 0x0a: - case 0x0b: - case 0x0c: - case 0x0d: - case 0x0e: - case 0x0f: - case 0x10: - case 0x11: - case 0x12: - case 0x13: - case 0x14: - case 0x15: - case 0x16: - case 0x17: - { - return v[current_idx]; - } - - case 0x18: // Unsigned integer (one-byte uint8_t follows) - { - idx += 1; // skip content byte - return get_from_vector(v, current_idx); - } - - case 0x19: // Unsigned integer (two-byte uint16_t follows) - { - idx += 2; // skip 2 content bytes - return get_from_vector(v, current_idx); - } - - case 0x1a: // Unsigned integer (four-byte uint32_t follows) - { - idx += 4; // skip 4 content bytes - return get_from_vector(v, current_idx); - } - - case 0x1b: // Unsigned integer (eight-byte uint64_t follows) - { - idx += 8; // skip 8 content bytes - return get_from_vector(v, current_idx); - } - - // Negative integer -1-0x00..-1-0x17 (-1..-24) - case 0x20: - case 0x21: - case 0x22: - case 0x23: - case 0x24: - case 0x25: - case 0x26: - case 0x27: - case 0x28: - case 0x29: - case 0x2a: - case 0x2b: - case 0x2c: - case 0x2d: - case 0x2e: - case 0x2f: - case 0x30: - case 0x31: - case 0x32: - case 0x33: - case 0x34: - case 0x35: - case 0x36: - case 0x37: - { - return static_cast(0x20 - 1 - v[current_idx]); - } - - case 0x38: // Negative integer (one-byte uint8_t follows) - { - idx += 1; // skip content byte - // must be uint8_t ! - return static_cast(-1) - get_from_vector(v, current_idx); - } - - case 0x39: // Negative integer -1-n (two-byte uint16_t follows) - { - idx += 2; // skip 2 content bytes - return static_cast(-1) - get_from_vector(v, current_idx); - } - - case 0x3a: // Negative integer -1-n (four-byte uint32_t follows) - { - idx += 4; // skip 4 content bytes - return static_cast(-1) - get_from_vector(v, current_idx); - } - - case 0x3b: // Negative integer -1-n (eight-byte uint64_t follows) - { - idx += 8; // skip 8 content bytes - return static_cast(-1) - static_cast(get_from_vector(v, current_idx)); - } - - // UTF-8 string (0x00..0x17 bytes follow) - case 0x60: - case 0x61: - case 0x62: - case 0x63: - case 0x64: - case 0x65: - case 0x66: - case 0x67: - case 0x68: - case 0x69: - case 0x6a: - case 0x6b: - case 0x6c: - case 0x6d: - case 0x6e: - case 0x6f: - case 0x70: - case 0x71: - case 0x72: - case 0x73: - case 0x74: - case 0x75: - case 0x76: - case 0x77: - { - const auto len = static_cast(v[current_idx] - 0x60); - const size_t offset = current_idx + 1; - idx += len; // skip content bytes - check_length(v.size(), len, offset); - return std::string(reinterpret_cast(v.data()) + offset, len); - } - - case 0x78: // UTF-8 string (one-byte uint8_t for n follows) - { - const auto len = static_cast(get_from_vector(v, current_idx)); - const size_t offset = current_idx + 2; - idx += len + 1; // skip size byte + content bytes - check_length(v.size(), len, offset); - return std::string(reinterpret_cast(v.data()) + offset, len); - } - - case 0x79: // UTF-8 string (two-byte uint16_t for n follow) - { - const auto len = static_cast(get_from_vector(v, current_idx)); - const size_t offset = current_idx + 3; - idx += len + 2; // skip 2 size bytes + content bytes - check_length(v.size(), len, offset); - return std::string(reinterpret_cast(v.data()) + offset, len); - } - - case 0x7a: // UTF-8 string (four-byte uint32_t for n follow) - { - const auto len = static_cast(get_from_vector(v, current_idx)); - const size_t offset = current_idx + 5; - idx += len + 4; // skip 4 size bytes + content bytes - check_length(v.size(), len, offset); - return std::string(reinterpret_cast(v.data()) + offset, len); - } - - case 0x7b: // UTF-8 string (eight-byte uint64_t for n follow) - { - const auto len = static_cast(get_from_vector(v, current_idx)); - const size_t offset = current_idx + 9; - idx += len + 8; // skip 8 size bytes + content bytes - check_length(v.size(), len, offset); - return std::string(reinterpret_cast(v.data()) + offset, len); - } - - case 0x7f: // UTF-8 string (indefinite length) - { - std::string result; - while (static_cast(check_length(v.size(), 1, idx)), v[idx] != 0xff) - { - string_t s = from_cbor_internal(v, idx); - result += s; - } - // skip break byte (0xFF) - idx += 1; - return result; - } - - // array (0x00..0x17 data items follow) - case 0x80: - case 0x81: - case 0x82: - case 0x83: - case 0x84: - case 0x85: - case 0x86: - case 0x87: - case 0x88: - case 0x89: - case 0x8a: - case 0x8b: - case 0x8c: - case 0x8d: - case 0x8e: - case 0x8f: - case 0x90: - case 0x91: - case 0x92: - case 0x93: - case 0x94: - case 0x95: - case 0x96: - case 0x97: - { - basic_json result = value_t::array; - const auto len = static_cast(v[current_idx] - 0x80); - for (size_t i = 0; i < len; ++i) - { - result.push_back(from_cbor_internal(v, idx)); - } - return result; - } - - case 0x98: // array (one-byte uint8_t for n follows) - { - basic_json result = value_t::array; - const auto len = static_cast(get_from_vector(v, current_idx)); - idx += 1; // skip 1 size byte - for (size_t i = 0; i < len; ++i) - { - result.push_back(from_cbor_internal(v, idx)); - } - return result; - } - - case 0x99: // array (two-byte uint16_t for n follow) - { - basic_json result = value_t::array; - const auto len = static_cast(get_from_vector(v, current_idx)); - idx += 2; // skip 4 size bytes - for (size_t i = 0; i < len; ++i) - { - result.push_back(from_cbor_internal(v, idx)); - } - return result; - } - - case 0x9a: // array (four-byte uint32_t for n follow) - { - basic_json result = value_t::array; - const auto len = static_cast(get_from_vector(v, current_idx)); - idx += 4; // skip 4 size bytes - for (size_t i = 0; i < len; ++i) - { - result.push_back(from_cbor_internal(v, idx)); - } - return result; - } - - case 0x9b: // array (eight-byte uint64_t for n follow) - { - basic_json result = value_t::array; - const auto len = static_cast(get_from_vector(v, current_idx)); - idx += 8; // skip 8 size bytes - for (size_t i = 0; i < len; ++i) - { - result.push_back(from_cbor_internal(v, idx)); - } - return result; - } - - case 0x9f: // array (indefinite length) - { - basic_json result = value_t::array; - while (static_cast(check_length(v.size(), 1, idx)), v[idx] != 0xff) - { - result.push_back(from_cbor_internal(v, idx)); - } - // skip break byte (0xFF) - idx += 1; - return result; - } - - // map (0x00..0x17 pairs of data items follow) - case 0xa0: - case 0xa1: - case 0xa2: - case 0xa3: - case 0xa4: - case 0xa5: - case 0xa6: - case 0xa7: - case 0xa8: - case 0xa9: - case 0xaa: - case 0xab: - case 0xac: - case 0xad: - case 0xae: - case 0xaf: - case 0xb0: - case 0xb1: - case 0xb2: - case 0xb3: - case 0xb4: - case 0xb5: - case 0xb6: - case 0xb7: - { - basic_json result = value_t::object; - const auto len = static_cast(v[current_idx] - 0xa0); - for (size_t i = 0; i < len; ++i) - { - cbor_expect_string(v, idx); - std::string key = from_cbor_internal(v, idx); - result[key] = from_cbor_internal(v, idx); - } - return result; - } - - case 0xb8: // map (one-byte uint8_t for n follows) - { - basic_json result = value_t::object; - const auto len = static_cast(get_from_vector(v, current_idx)); - idx += 1; // skip 1 size byte - for (size_t i = 0; i < len; ++i) - { - cbor_expect_string(v, idx); - std::string key = from_cbor_internal(v, idx); - result[key] = from_cbor_internal(v, idx); - } - return result; - } - - case 0xb9: // map (two-byte uint16_t for n follow) - { - basic_json result = value_t::object; - const auto len = static_cast(get_from_vector(v, current_idx)); - idx += 2; // skip 2 size bytes - for (size_t i = 0; i < len; ++i) - { - cbor_expect_string(v, idx); - std::string key = from_cbor_internal(v, idx); - result[key] = from_cbor_internal(v, idx); - } - return result; - } - - case 0xba: // map (four-byte uint32_t for n follow) - { - basic_json result = value_t::object; - const auto len = static_cast(get_from_vector(v, current_idx)); - idx += 4; // skip 4 size bytes - for (size_t i = 0; i < len; ++i) - { - cbor_expect_string(v, idx); - std::string key = from_cbor_internal(v, idx); - result[key] = from_cbor_internal(v, idx); - } - return result; - } - - case 0xbb: // map (eight-byte uint64_t for n follow) - { - basic_json result = value_t::object; - const auto len = static_cast(get_from_vector(v, current_idx)); - idx += 8; // skip 8 size bytes - for (size_t i = 0; i < len; ++i) - { - cbor_expect_string(v, idx); - std::string key = from_cbor_internal(v, idx); - result[key] = from_cbor_internal(v, idx); - } - return result; - } - - case 0xbf: // map (indefinite length) - { - basic_json result = value_t::object; - while (static_cast(check_length(v.size(), 1, idx)), v[idx] != 0xff) - { - cbor_expect_string(v, idx); - std::string key = from_cbor_internal(v, idx); - result[key] = from_cbor_internal(v, idx); - } - // skip break byte (0xFF) - idx += 1; - return result; - } - - case 0xf4: // false - { - return false; - } - - case 0xf5: // true - { - return true; - } - - case 0xf6: // null - { - return value_t::null; - } - - case 0xf9: // Half-Precision Float (two-byte IEEE 754) - { - idx += 2; // skip two content bytes - - // code from RFC 7049, Appendix D, Figure 3: - // As half-precision floating-point numbers were only added to - // IEEE 754 in 2008, today's programming platforms often still - // only have limited support for them. It is very easy to - // include at least decoding support for them even without such - // support. An example of a small decoder for half-precision - // floating-point numbers in the C language is shown in Fig. 3. - check_length(v.size(), 2, current_idx + 1); - const int half = (v[current_idx + 1] << 8) + v[current_idx + 2]; - const int exp = (half >> 10) & 0x1f; - const int mant = half & 0x3ff; - double val; - if (exp == 0) - { - val = std::ldexp(mant, -24); - } - else if (exp != 31) - { - val = std::ldexp(mant + 1024, exp - 25); - } - else - { - val = mant == 0 - ? std::numeric_limits::infinity() - : std::numeric_limits::quiet_NaN(); - } - return (half & 0x8000) != 0 ? -val : val; - } - - case 0xfa: // Single-Precision Float (four-byte IEEE 754) - { - // copy bytes in reverse order into the float variable - float res; - check_length(v.size(), sizeof(float), current_idx + 1); - for (size_t byte = 0; byte < sizeof(float); ++byte) - { - reinterpret_cast(&res)[sizeof(float) - byte - 1] = v[current_idx + 1 + byte]; - } - idx += sizeof(float); // skip content bytes - return res; - } - - case 0xfb: // Double-Precision Float (eight-byte IEEE 754) - { - // copy bytes in reverse order into the double variable - double res; - check_length(v.size(), sizeof(double), current_idx + 1); - for (size_t byte = 0; byte < sizeof(double); ++byte) - { - reinterpret_cast(&res)[sizeof(double) - byte - 1] = v[current_idx + 1 + byte]; - } - idx += sizeof(double); // skip content bytes - return res; - } - - default: // anything else (0xFF is handled inside the other types) - { - std::stringstream ss; - ss << std::hex << static_cast(v[current_idx]); - JSON_THROW(parse_error::create(112, current_idx + 1, "error reading CBOR; last byte: 0x" + ss.str())); - } - } - } - public: /*! @brief create a MessagePack serialization of a given JSON value @@ -9053,7 +8554,7 @@ class basic_json } /*! - @brief create a MessagePack serialization of a given JSON value + @brief create a CBOR serialization of a given JSON value Serializes a given JSON value @a j to a byte vector using the CBOR (Concise Binary Object Representation) serialization format. CBOR is a binary @@ -9141,100 +8642,6 @@ class basic_json return result; } - /*! - @brief create a JSON value from a byte vector in CBOR format - - Deserializes a given byte vector @a v to a JSON value using the CBOR - (Concise Binary Object Representation) serialization format. - - The library maps CBOR types to JSON value types as follows: - - CBOR type | JSON value type | first byte - ---------------------- | --------------- | ---------- - Integer | number_unsigned | 0x00..0x17 - Unsigned integer | number_unsigned | 0x18 - Unsigned integer | number_unsigned | 0x19 - Unsigned integer | number_unsigned | 0x1a - Unsigned integer | number_unsigned | 0x1b - Negative integer | number_integer | 0x20..0x37 - Negative integer | number_integer | 0x38 - Negative integer | number_integer | 0x39 - Negative integer | number_integer | 0x3a - Negative integer | number_integer | 0x3b - Negative integer | number_integer | 0x40..0x57 - UTF-8 string | string | 0x60..0x77 - UTF-8 string | string | 0x78 - UTF-8 string | string | 0x79 - UTF-8 string | string | 0x7a - UTF-8 string | string | 0x7b - UTF-8 string | string | 0x7f - array | array | 0x80..0x97 - array | array | 0x98 - array | array | 0x99 - array | array | 0x9a - array | array | 0x9b - array | array | 0x9f - map | object | 0xa0..0xb7 - map | object | 0xb8 - map | object | 0xb9 - map | object | 0xba - map | object | 0xbb - map | object | 0xbf - False | `false` | 0xf4 - True | `true` | 0xf5 - Nill | `null` | 0xf6 - Half-Precision Float | number_float | 0xf9 - Single-Precision Float | number_float | 0xfa - Double-Precision Float | number_float | 0xfb - - @warning The mapping is **incomplete** in the sense that not all CBOR - types can be converted to a JSON value. The following CBOR types - are not supported and will yield parse errors (parse_error.112): - - byte strings (0x40..0x5f) - - date/time (0xc0..0xc1) - - bignum (0xc2..0xc3) - - decimal fraction (0xc4) - - bigfloat (0xc5) - - tagged items (0xc6..0xd4, 0xd8..0xdb) - - expected conversions (0xd5..0xd7) - - simple values (0xe0..0xf3, 0xf8) - - undefined (0xf7) - - @warning CBOR allows map keys of any type, whereas JSON only allows - strings as keys in object values. Therefore, CBOR maps with keys - other than UTF-8 strings are rejected (parse_error.113). - - @note Any CBOR output created @ref to_cbor can be successfully parsed by - @ref from_cbor. - - @param[in] v a byte vector in CBOR format - @param[in] start_index the index to start reading from @a v (0 by default) - @return deserialized JSON value - - @throw parse_error.110 if the given vector ends prematurely - @throw parse_error.112 if unsupported features from CBOR were - used in the given vector @a v or if the input is not valid CBOR - @throw parse_error.113 if a string was expected as map key, but not found - - @complexity Linear in the size of the byte vector @a v. - - @liveexample{The example shows the deserialization of a byte vector in CBOR - format to a JSON value.,from_cbor} - - @sa http://cbor.io - @sa @ref to_cbor(const basic_json&) for the analogous serialization - @sa @ref from_msgpack(const std::vector&, const size_t) for the - related MessagePack format - - @since version 2.0.9, parameter @a start_index since 2.1.1 - */ - //static basic_json from_cbor(const std::vector& v, - // const size_t start_index = 0) - //{ - // size_t i = start_index; - // return from_cbor_internal(v, i); - //} - /// @} /////////////////////////// @@ -10996,6 +10403,94 @@ class basic_json }; public: + + /*! + @brief create a JSON value from a byte vector in CBOR format + + Deserializes a given byte vector @a v to a JSON value using the CBOR + (Concise Binary Object Representation) serialization format. + + The library maps CBOR types to JSON value types as follows: + + CBOR type | JSON value type | first byte + ---------------------- | --------------- | ---------- + Integer | number_unsigned | 0x00..0x17 + Unsigned integer | number_unsigned | 0x18 + Unsigned integer | number_unsigned | 0x19 + Unsigned integer | number_unsigned | 0x1a + Unsigned integer | number_unsigned | 0x1b + Negative integer | number_integer | 0x20..0x37 + Negative integer | number_integer | 0x38 + Negative integer | number_integer | 0x39 + Negative integer | number_integer | 0x3a + Negative integer | number_integer | 0x3b + Negative integer | number_integer | 0x40..0x57 + UTF-8 string | string | 0x60..0x77 + UTF-8 string | string | 0x78 + UTF-8 string | string | 0x79 + UTF-8 string | string | 0x7a + UTF-8 string | string | 0x7b + UTF-8 string | string | 0x7f + array | array | 0x80..0x97 + array | array | 0x98 + array | array | 0x99 + array | array | 0x9a + array | array | 0x9b + array | array | 0x9f + map | object | 0xa0..0xb7 + map | object | 0xb8 + map | object | 0xb9 + map | object | 0xba + map | object | 0xbb + map | object | 0xbf + False | `false` | 0xf4 + True | `true` | 0xf5 + Nill | `null` | 0xf6 + Half-Precision Float | number_float | 0xf9 + Single-Precision Float | number_float | 0xfa + Double-Precision Float | number_float | 0xfb + + @warning The mapping is **incomplete** in the sense that not all CBOR + types can be converted to a JSON value. The following CBOR types + are not supported and will yield parse errors (parse_error.112): + - byte strings (0x40..0x5f) + - date/time (0xc0..0xc1) + - bignum (0xc2..0xc3) + - decimal fraction (0xc4) + - bigfloat (0xc5) + - tagged items (0xc6..0xd4, 0xd8..0xdb) + - expected conversions (0xd5..0xd7) + - simple values (0xe0..0xf3, 0xf8) + - undefined (0xf7) + + @warning CBOR allows map keys of any type, whereas JSON only allows + strings as keys in object values. Therefore, CBOR maps with keys + other than UTF-8 strings are rejected (parse_error.113). + + @note Any CBOR output created @ref to_cbor can be successfully parsed by + @ref from_cbor. + + @param[in] v a byte vector in CBOR format + @param[in] start_index the index to start reading from @a v (0 by default) + @return deserialized JSON value + + @throw parse_error.110 if the given vector ends prematurely + @throw parse_error.112 if unsupported features from CBOR were + used in the given vector @a v or if the input is not valid CBOR + @throw parse_error.113 if a string was expected as map key, but not found + + @complexity Linear in the size of the byte vector @a v. + + @liveexample{The example shows the deserialization of a byte vector in CBOR + format to a JSON value.,from_cbor} + + @sa http://cbor.io + @sa @ref to_cbor(const basic_json&) for the analogous serialization + @sa @ref from_msgpack(const std::vector&, const size_t) for the + related MessagePack format + + @since version 2.0.9, parameter @a start_index since 2.1.1 + */ static basic_json from_cbor(const std::vector& v, const size_t start_index = 0) { From b15fc13dd194651a5acce92952ab3c09e0a39722 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 4 Apr 2017 16:59:19 +0200 Subject: [PATCH 140/530] :hammer: implemented MessagePack in binary_reader --- src/json.hpp | 1024 ++++++++++++++++++---------------- test/src/unit-cbor.cpp | 7 - test/src/unit-msgpack.cpp | 30 +- test/src/unit-regression.cpp | 8 +- 4 files changed, 567 insertions(+), 502 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index e6fbabdf..6853d8c0 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -7480,57 +7480,6 @@ class basic_json } } - /*! - @brief take sufficient bytes from a vector to fill an integer variable - - In the context of binary serialization formats, we need to read several - bytes from a byte vector and combine them to multi-byte integral data - types. - - @param[in] vec byte vector to read from - @param[in] current_index the position in the vector after which to read - - @return the next sizeof(T) bytes from @a vec, in reverse order as T - - @tparam T the integral return type - - @throw parse_error.110 if there are less than sizeof(T)+1 bytes in the - vector @a vec to read - - In the for loop, the bytes from the vector are copied in reverse order into - the return value. In the figures below, let sizeof(T)=4 and `i` be the loop - variable. - - Precondition: - - vec: | | | a | b | c | d | T: | | | | | - ^ ^ ^ ^ - current_index i ptr sizeof(T) - - Postcondition: - - vec: | | | a | b | c | d | T: | d | c | b | a | - ^ ^ ^ - | i ptr - current_index - - @sa Code adapted from . - */ - template - static T get_from_vector(const std::vector& vec, const size_t current_index) - { - // check if we can read sizeof(T) bytes starting the next index - check_length(vec.size(), sizeof(T), current_index + 1); - - T result; - auto* ptr = reinterpret_cast(&result); - for (size_t i = 0; i < sizeof(T); ++i) - { - *ptr++ = vec[current_index + sizeof(T) - i]; - } - return result; - } - /*! @brief create a MessagePack serialization of a given JSON value @@ -8041,357 +7990,6 @@ class basic_json } } - - /* - @brief checks if given lengths do not exceed the size of a given vector - - To secure the access to the byte vector during CBOR/MessagePack - deserialization, bytes are copied from the vector into buffers. This - function checks if the number of bytes to copy (@a len) does not exceed - the size @s size of the vector. Additionally, an @a offset is given from - where to start reading the bytes. - - This function checks whether reading the bytes is safe; that is, offset is - a valid index in the vector, offset+len - - @param[in] size size of the byte vector - @param[in] len number of bytes to read - @param[in] offset offset where to start reading - - vec: x x x x x X X X X X - ^ ^ ^ - 0 offset len - - @throws out_of_range if `len > v.size()` - */ - static void check_length(const size_t size, const size_t len, const size_t offset) - { - // simple case: requested length is greater than the vector's length - if (len > size or offset > size) - { - JSON_THROW(parse_error::create(110, offset + 1, "cannot read " + std::to_string(len) + " bytes from vector")); - } - - // second case: adding offset would result in overflow - if ((size > ((std::numeric_limits::max)() - offset))) - { - JSON_THROW(parse_error::create(110, offset + 1, "cannot read " + std::to_string(len) + " bytes from vector")); - } - - // last case: reading past the end of the vector - if (len + offset > size) - { - JSON_THROW(parse_error::create(110, offset + 1, "cannot read " + std::to_string(len) + " bytes from vector")); - } - } - - /*! - @brief check if the next byte belongs to a string - - While parsing a map, the keys must be strings. This function checks if the - current byte is one of the start bytes for a string in MessagePack: - - - 0xa0 - 0xbf: fixstr - - 0xd9: str 8 - - 0xda: str 16 - - 0xdb: str 32 - - @param[in] v MessagePack serialization - @param[in] idx byte index in @a v to check for a string - - @throw parse_error.113 if `v[idx]` does not belong to a string - */ - static void msgpack_expect_string(const std::vector& v, size_t idx) - { - check_length(v.size(), 1, idx); - - const auto byte = v[idx]; - if ((byte >= 0xa0 and byte <= 0xbf) or (byte >= 0xd9 and byte <= 0xdb)) - { - return; - } - - std::stringstream ss; - ss << std::hex << static_cast(v[idx]); - JSON_THROW(parse_error::create(113, idx + 1, "expected a MessagePack string; last byte: 0x" + ss.str())); - } - - /*! - @brief check if the next byte belongs to a string - - While parsing a map, the keys must be strings. This function checks if the - current byte is one of the start bytes for a string in CBOR: - - - 0x60 - 0x77: fixed length - - 0x78 - 0x7b: variable length - - 0x7f: indefinity length - - @param[in] v CBOR serialization - @param[in] idx byte index in @a v to check for a string - - @throw parse_error.113 if `v[idx]` does not belong to a string - */ - static void cbor_expect_string(const std::vector& v, size_t idx) - { - check_length(v.size(), 1, idx); - - const auto byte = v[idx]; - if ((byte >= 0x60 and byte <= 0x7b) or byte == 0x7f) - { - return; - } - - std::stringstream ss; - ss << std::hex << static_cast(v[idx]); - JSON_THROW(parse_error::create(113, idx + 1, "expected a CBOR string; last byte: 0x" + ss.str())); - } - - /*! - @brief create a JSON value from a given MessagePack vector - - @param[in] v MessagePack serialization - @param[in] idx byte index to start reading from @a v - - @return deserialized JSON value - - @throw parse_error.110 if the given vector ends prematurely - @throw parse_error.112 if unsupported features from MessagePack were - used in the given vector @a v or if the input is not valid MessagePack - @throw parse_error.113 if a string was expected as map key, but not found - - @sa https://github.com/msgpack/msgpack/blob/master/spec.md - */ - static basic_json from_msgpack_internal(const std::vector& v, size_t& idx) - { - // store and increment index - const size_t current_idx = idx++; - - // make sure reading 1 byte is safe - check_length(v.size(), 1, current_idx); - - if (v[current_idx] <= 0xbf) - { - if (v[current_idx] <= 0x7f) // positive fixint - { - return v[current_idx]; - } - if (v[current_idx] <= 0x8f) // fixmap - { - basic_json result = value_t::object; - const size_t len = v[current_idx] & 0x0f; - for (size_t i = 0; i < len; ++i) - { - msgpack_expect_string(v, idx); - std::string key = from_msgpack_internal(v, idx); - result[key] = from_msgpack_internal(v, idx); - } - return result; - } - else if (v[current_idx] <= 0x9f) // fixarray - { - basic_json result = value_t::array; - const size_t len = v[current_idx] & 0x0f; - for (size_t i = 0; i < len; ++i) - { - result.push_back(from_msgpack_internal(v, idx)); - } - return result; - } - else // fixstr - { - const size_t len = v[current_idx] & 0x1f; - const size_t offset = current_idx + 1; - idx += len; // skip content bytes - check_length(v.size(), len, offset); - return std::string(reinterpret_cast(v.data()) + offset, len); - } - } - else if (v[current_idx] >= 0xe0) // negative fixint - { - return static_cast(v[current_idx]); - } - else - { - switch (v[current_idx]) - { - case 0xc0: // nil - { - return value_t::null; - } - - case 0xc2: // false - { - return false; - } - - case 0xc3: // true - { - return true; - } - - case 0xca: // float 32 - { - // copy bytes in reverse order into the double variable - float res; - check_length(v.size(), sizeof(float), current_idx + 1); - for (size_t byte = 0; byte < sizeof(float); ++byte) - { - reinterpret_cast(&res)[sizeof(float) - byte - 1] = v[current_idx + 1 + byte]; - } - idx += sizeof(float); // skip content bytes - return res; - } - - case 0xcb: // float 64 - { - // copy bytes in reverse order into the double variable - double res; - check_length(v.size(), sizeof(double), current_idx + 1); - for (size_t byte = 0; byte < sizeof(double); ++byte) - { - reinterpret_cast(&res)[sizeof(double) - byte - 1] = v[current_idx + 1 + byte]; - } - idx += sizeof(double); // skip content bytes - return res; - } - - case 0xcc: // uint 8 - { - idx += 1; // skip content byte - return get_from_vector(v, current_idx); - } - - case 0xcd: // uint 16 - { - idx += 2; // skip 2 content bytes - return get_from_vector(v, current_idx); - } - - case 0xce: // uint 32 - { - idx += 4; // skip 4 content bytes - return get_from_vector(v, current_idx); - } - - case 0xcf: // uint 64 - { - idx += 8; // skip 8 content bytes - return get_from_vector(v, current_idx); - } - - case 0xd0: // int 8 - { - idx += 1; // skip content byte - return get_from_vector(v, current_idx); - } - - case 0xd1: // int 16 - { - idx += 2; // skip 2 content bytes - return get_from_vector(v, current_idx); - } - - case 0xd2: // int 32 - { - idx += 4; // skip 4 content bytes - return get_from_vector(v, current_idx); - } - - case 0xd3: // int 64 - { - idx += 8; // skip 8 content bytes - return get_from_vector(v, current_idx); - } - - case 0xd9: // str 8 - { - const auto len = static_cast(get_from_vector(v, current_idx)); - const size_t offset = current_idx + 2; - idx += len + 1; // skip size byte + content bytes - check_length(v.size(), len, offset); - return std::string(reinterpret_cast(v.data()) + offset, len); - } - - case 0xda: // str 16 - { - const auto len = static_cast(get_from_vector(v, current_idx)); - const size_t offset = current_idx + 3; - idx += len + 2; // skip 2 size bytes + content bytes - check_length(v.size(), len, offset); - return std::string(reinterpret_cast(v.data()) + offset, len); - } - - case 0xdb: // str 32 - { - const auto len = static_cast(get_from_vector(v, current_idx)); - const size_t offset = current_idx + 5; - idx += len + 4; // skip 4 size bytes + content bytes - check_length(v.size(), len, offset); - return std::string(reinterpret_cast(v.data()) + offset, len); - } - - case 0xdc: // array 16 - { - basic_json result = value_t::array; - const auto len = static_cast(get_from_vector(v, current_idx)); - idx += 2; // skip 2 size bytes - for (size_t i = 0; i < len; ++i) - { - result.push_back(from_msgpack_internal(v, idx)); - } - return result; - } - - case 0xdd: // array 32 - { - basic_json result = value_t::array; - const auto len = static_cast(get_from_vector(v, current_idx)); - idx += 4; // skip 4 size bytes - for (size_t i = 0; i < len; ++i) - { - result.push_back(from_msgpack_internal(v, idx)); - } - return result; - } - - case 0xde: // map 16 - { - basic_json result = value_t::object; - const auto len = static_cast(get_from_vector(v, current_idx)); - idx += 2; // skip 2 size bytes - for (size_t i = 0; i < len; ++i) - { - msgpack_expect_string(v, idx); - std::string key = from_msgpack_internal(v, idx); - result[key] = from_msgpack_internal(v, idx); - } - return result; - } - - case 0xdf: // map 32 - { - basic_json result = value_t::object; - const auto len = static_cast(get_from_vector(v, current_idx)); - idx += 4; // skip 4 size bytes - for (size_t i = 0; i < len; ++i) - { - msgpack_expect_string(v, idx); - std::string key = from_msgpack_internal(v, idx); - result[key] = from_msgpack_internal(v, idx); - } - return result; - } - - default: - { - std::stringstream ss; - ss << std::hex << static_cast(v[current_idx]); - JSON_THROW(parse_error::create(112, current_idx + 1, "error reading MessagePack; last byte: 0x" + ss.str())); - } - } - } - } - public: /*! @brief create a MessagePack serialization of a given JSON value @@ -8474,80 +8072,6 @@ class basic_json return result; } - /*! - @brief create a JSON value from a byte vector in MessagePack format - - Deserializes a given byte vector @a v to a JSON value using the MessagePack - serialization format. - - The library maps MessagePack types to JSON value types as follows: - - MessagePack type | JSON value type | first byte - ---------------- | --------------- | ---------- - positive fixint | number_unsigned | 0x00..0x7f - fixmap | object | 0x80..0x8f - fixarray | array | 0x90..0x9f - fixstr | string | 0xa0..0xbf - nil | `null` | 0xc0 - false | `false` | 0xc2 - true | `true` | 0xc3 - float 32 | number_float | 0xca - float 64 | number_float | 0xcb - uint 8 | number_unsigned | 0xcc - uint 16 | number_unsigned | 0xcd - uint 32 | number_unsigned | 0xce - uint 64 | number_unsigned | 0xcf - int 8 | number_integer | 0xd0 - int 16 | number_integer | 0xd1 - int 32 | number_integer | 0xd2 - int 64 | number_integer | 0xd3 - str 8 | string | 0xd9 - str 16 | string | 0xda - str 32 | string | 0xdb - array 16 | array | 0xdc - array 32 | array | 0xdd - map 16 | object | 0xde - map 32 | object | 0xdf - negative fixint | number_integer | 0xe0-0xff - - @warning The mapping is **incomplete** in the sense that not all - MessagePack types can be converted to a JSON value. The following - MessagePack types are not supported and will yield parse errors: - - bin 8 - bin 32 (0xc4..0xc6) - - ext 8 - ext 32 (0xc7..0xc9) - - fixext 1 - fixext 16 (0xd4..0xd8) - - @note Any MessagePack output created @ref to_msgpack can be successfully - parsed by @ref from_msgpack. - - @param[in] v a byte vector in MessagePack format - @param[in] start_index the index to start reading from @a v (0 by default) - @return deserialized JSON value - - @throw parse_error.110 if the given vector ends prematurely - @throw parse_error.112 if unsupported features from MessagePack were - used in the given vector @a v or if the input is not valid MessagePack - @throw parse_error.113 if a string was expected as map key, but not found - - @complexity Linear in the size of the byte vector @a v. - - @liveexample{The example shows the deserialization of a byte vector in - MessagePack format to a JSON value.,from_msgpack} - - @sa http://msgpack.org - @sa @ref to_msgpack(const basic_json&) for the analogous serialization - @sa @ref from_cbor(const std::vector&, const size_t) for the - related CBOR format - - @since version 2.0.9, parameter @a start_index since 2.1.1 - */ - static basic_json from_msgpack(const std::vector& v, - const size_t start_index = 0) - { - size_t i = start_index; - return from_msgpack_internal(v, i); - } - /*! @brief create a CBOR serialization of a given JSON value @@ -10262,6 +9786,408 @@ class basic_json } } + basic_json parse_msgpack() + { + switch (get()) + { + // EOF + case std::char_traits::eof(): + { + JSON_THROW(parse_error::create(110, chars_read, "unexpected end of input")); + } + + // positive fixint + case 0x00: + case 0x01: + case 0x02: + case 0x03: + case 0x04: + case 0x05: + case 0x06: + case 0x07: + case 0x08: + case 0x09: + case 0x0a: + case 0x0b: + case 0x0c: + case 0x0d: + case 0x0e: + case 0x0f: + case 0x10: + case 0x11: + case 0x12: + case 0x13: + case 0x14: + case 0x15: + case 0x16: + case 0x17: + case 0x18: + case 0x19: + case 0x1a: + case 0x1b: + case 0x1c: + case 0x1d: + case 0x1e: + case 0x1f: + case 0x20: + case 0x21: + case 0x22: + case 0x23: + case 0x24: + case 0x25: + case 0x26: + case 0x27: + case 0x28: + case 0x29: + case 0x2a: + case 0x2b: + case 0x2c: + case 0x2d: + case 0x2e: + case 0x2f: + case 0x30: + case 0x31: + case 0x32: + case 0x33: + case 0x34: + case 0x35: + case 0x36: + case 0x37: + case 0x38: + case 0x39: + case 0x3a: + case 0x3b: + case 0x3c: + case 0x3d: + case 0x3e: + case 0x3f: + case 0x40: + case 0x41: + case 0x42: + case 0x43: + case 0x44: + case 0x45: + case 0x46: + case 0x47: + case 0x48: + case 0x49: + case 0x4a: + case 0x4b: + case 0x4c: + case 0x4d: + case 0x4e: + case 0x4f: + case 0x50: + case 0x51: + case 0x52: + case 0x53: + case 0x54: + case 0x55: + case 0x56: + case 0x57: + case 0x58: + case 0x59: + case 0x5a: + case 0x5b: + case 0x5c: + case 0x5d: + case 0x5e: + case 0x5f: + case 0x60: + case 0x61: + case 0x62: + case 0x63: + case 0x64: + case 0x65: + case 0x66: + case 0x67: + case 0x68: + case 0x69: + case 0x6a: + case 0x6b: + case 0x6c: + case 0x6d: + case 0x6e: + case 0x6f: + case 0x70: + case 0x71: + case 0x72: + case 0x73: + case 0x74: + case 0x75: + case 0x76: + case 0x77: + case 0x78: + case 0x79: + case 0x7a: + case 0x7b: + case 0x7c: + case 0x7d: + case 0x7e: + case 0x7f: + { + return static_cast(current); + } + + // fixmap + case 0x80: + case 0x81: + case 0x82: + case 0x83: + case 0x84: + case 0x85: + case 0x86: + case 0x87: + case 0x88: + case 0x89: + case 0x8a: + case 0x8b: + case 0x8c: + case 0x8d: + case 0x8e: + case 0x8f: + { + basic_json result = value_t::object; + const auto len = static_cast(current & 0x0f); + for (size_t i = 0; i < len; ++i) + { + get(); + auto key = get_msgpack_string(); + result[key] = parse_msgpack(); + } + return result; + } + + // fixarray + case 0x90: + case 0x91: + case 0x92: + case 0x93: + case 0x94: + case 0x95: + case 0x96: + case 0x97: + case 0x98: + case 0x99: + case 0x9a: + case 0x9b: + case 0x9c: + case 0x9d: + case 0x9e: + case 0x9f: + { + basic_json result = value_t::array; + const auto len = static_cast(current & 0x0f); + for (size_t i = 0; i < len; ++i) + { + result.push_back(parse_msgpack()); + } + return result; + } + + // fixstr + case 0xa0: + case 0xa1: + case 0xa2: + case 0xa3: + case 0xa4: + case 0xa5: + case 0xa6: + case 0xa7: + case 0xa8: + case 0xa9: + case 0xaa: + case 0xab: + case 0xac: + case 0xad: + case 0xae: + case 0xaf: + case 0xb0: + case 0xb1: + case 0xb2: + case 0xb3: + case 0xb4: + case 0xb5: + case 0xb6: + case 0xb7: + case 0xb8: + case 0xb9: + case 0xba: + case 0xbb: + case 0xbc: + case 0xbd: + case 0xbe: + case 0xbf: + { + return get_msgpack_string(); + } + + case 0xc0: // nil + { + return value_t::null; + } + + case 0xc2: // false + { + return false; + } + + case 0xc3: // true + { + return true; + } + + case 0xca: // float 32 + { + return get_number(); + } + + case 0xcb: // float 64 + { + return get_number(); + } + + case 0xcc: // uint 8 + { + return get_number(); + } + + case 0xcd: // uint 16 + { + return get_number(); + } + + case 0xce: // uint 32 + { + return get_number(); + } + + case 0xcf: // uint 64 + { + return get_number(); + } + + case 0xd0: // int 8 + { + return get_number(); + } + + case 0xd1: // int 16 + { + return get_number(); + } + + case 0xd2: // int 32 + { + return get_number(); + } + + case 0xd3: // int 64 + { + return get_number(); + } + + case 0xd9: // str 8 + case 0xda: // str 16 + case 0xdb: // str 32 + { + return get_msgpack_string(); + } + + case 0xdc: // array 16 + { + basic_json result = value_t::array; + const auto len = static_cast(get_number()); + for (size_t i = 0; i < len; ++i) + { + result.push_back(parse_msgpack()); + } + return result; + } + + case 0xdd: // array 32 + { + basic_json result = value_t::array; + const auto len = static_cast(get_number()); + for (size_t i = 0; i < len; ++i) + { + result.push_back(parse_msgpack()); + } + return result; + } + + case 0xde: // map 16 + { + basic_json result = value_t::object; + const auto len = static_cast(get_number()); + for (size_t i = 0; i < len; ++i) + { + get(); + auto key = get_msgpack_string(); + result[key] = parse_msgpack(); + } + return result; + } + + case 0xdf: // map 32 + { + basic_json result = value_t::object; + const auto len = static_cast(get_number()); + for (size_t i = 0; i < len; ++i) + { + get(); + auto key = get_msgpack_string(); + result[key] = parse_msgpack(); + } + return result; + } + + // positive fixint + case 0xe0: + case 0xe1: + case 0xe2: + case 0xe3: + case 0xe4: + case 0xe5: + case 0xe6: + case 0xe7: + case 0xe8: + case 0xe9: + case 0xea: + case 0xeb: + case 0xec: + case 0xed: + case 0xee: + case 0xef: + case 0xf0: + case 0xf1: + case 0xf2: + case 0xf3: + case 0xf4: + case 0xf5: + case 0xf6: + case 0xf7: + case 0xf8: + case 0xf9: + case 0xfa: + case 0xfb: + case 0xfc: + case 0xfd: + case 0xfe: + case 0xff: + { + return static_cast(current); + } + + default: // anything else + { + std::stringstream ss; + ss << std::hex << current; + JSON_THROW(parse_error::create(112, chars_read, "error reading MessagePack; last byte: 0x" + ss.str())); + } + } + } + private: int get() { @@ -10378,6 +10304,77 @@ class basic_json } } + std::string get_msgpack_string() + { + check_eof(); + + switch (current) + { + // fixstr + case 0xa0: + case 0xa1: + case 0xa2: + case 0xa3: + case 0xa4: + case 0xa5: + case 0xa6: + case 0xa7: + case 0xa8: + case 0xa9: + case 0xaa: + case 0xab: + case 0xac: + case 0xad: + case 0xae: + case 0xaf: + case 0xb0: + case 0xb1: + case 0xb2: + case 0xb3: + case 0xb4: + case 0xb5: + case 0xb6: + case 0xb7: + case 0xb8: + case 0xb9: + case 0xba: + case 0xbb: + case 0xbc: + case 0xbd: + case 0xbe: + case 0xbf: + { + const auto len = static_cast(current & 0x1f); + return get_string(len); + } + + case 0xd9: // str 8 + { + const auto len = static_cast(get_number()); + return get_string(len); + } + + case 0xda: // str 16 + { + const auto len = static_cast(get_number()); + return get_string(len); + } + + case 0xdb: // str 32 + { + const auto len = static_cast(get_number()); + return get_string(len); + } + + default: + { + std::stringstream ss; + ss << std::hex << current; + JSON_THROW(parse_error::create(113, chars_read, "expected a MessagePack string; last byte: 0x" + ss.str())); + } + } + } + void check_eof() { if (JSON_UNLIKELY(current == std::char_traits::eof())) @@ -10493,6 +10490,81 @@ class basic_json return br.parse_cbor(); } + + /*! + @brief create a JSON value from a byte vector in MessagePack format + + Deserializes a given byte vector @a v to a JSON value using the MessagePack + serialization format. + + The library maps MessagePack types to JSON value types as follows: + + MessagePack type | JSON value type | first byte + ---------------- | --------------- | ---------- + positive fixint | number_unsigned | 0x00..0x7f + fixmap | object | 0x80..0x8f + fixarray | array | 0x90..0x9f + fixstr | string | 0xa0..0xbf + nil | `null` | 0xc0 + false | `false` | 0xc2 + true | `true` | 0xc3 + float 32 | number_float | 0xca + float 64 | number_float | 0xcb + uint 8 | number_unsigned | 0xcc + uint 16 | number_unsigned | 0xcd + uint 32 | number_unsigned | 0xce + uint 64 | number_unsigned | 0xcf + int 8 | number_integer | 0xd0 + int 16 | number_integer | 0xd1 + int 32 | number_integer | 0xd2 + int 64 | number_integer | 0xd3 + str 8 | string | 0xd9 + str 16 | string | 0xda + str 32 | string | 0xdb + array 16 | array | 0xdc + array 32 | array | 0xdd + map 16 | object | 0xde + map 32 | object | 0xdf + negative fixint | number_integer | 0xe0-0xff + + @warning The mapping is **incomplete** in the sense that not all + MessagePack types can be converted to a JSON value. The following + MessagePack types are not supported and will yield parse errors: + - bin 8 - bin 32 (0xc4..0xc6) + - ext 8 - ext 32 (0xc7..0xc9) + - fixext 1 - fixext 16 (0xd4..0xd8) + + @note Any MessagePack output created @ref to_msgpack can be successfully + parsed by @ref from_msgpack. + + @param[in] v a byte vector in MessagePack format + @param[in] start_index the index to start reading from @a v (0 by default) + @return deserialized JSON value + + @throw parse_error.110 if the given vector ends prematurely + @throw parse_error.112 if unsupported features from MessagePack were + used in the given vector @a v or if the input is not valid MessagePack + @throw parse_error.113 if a string was expected as map key, but not found + + @complexity Linear in the size of the byte vector @a v. + + @liveexample{The example shows the deserialization of a byte vector in + MessagePack format to a JSON value.,from_msgpack} + + @sa http://msgpack.org + @sa @ref to_msgpack(const basic_json&) for the analogous serialization + @sa @ref from_cbor(const std::vector&, const size_t) for the + related CBOR format + + @since version 2.0.9, parameter @a start_index since 2.1.1 + */ + static basic_json from_msgpack(const std::vector& v, + const size_t start_index = 0) + { + binary_reader br(reinterpret_cast(v.data() + start_index), v.size() - start_index); + return br.parse_msgpack(); + } + ////////////////////// // lexer and parser // ////////////////////// diff --git a/test/src/unit-cbor.cpp b/test/src/unit-cbor.cpp index 390a1b52..ae6f9a76 100644 --- a/test/src/unit-cbor.cpp +++ b/test/src/unit-cbor.cpp @@ -28,7 +28,6 @@ SOFTWARE. #include "catch.hpp" -#define private public #include "json.hpp" using nlohmann::json; @@ -1357,12 +1356,6 @@ TEST_CASE("CBOR regressions", "[!throws]") } } } - - SECTION("improve code coverage") - { - // exotic edge case - CHECK_THROWS_AS(json::check_length(0xffffffffffffffffull, 0xfffffffffffffff0ull, 0xff), json::parse_error); - } } TEST_CASE("CBOR roundtrips", "[hide]") diff --git a/test/src/unit-msgpack.cpp b/test/src/unit-msgpack.cpp index 5d157812..3f1d1643 100644 --- a/test/src/unit-msgpack.cpp +++ b/test/src/unit-msgpack.cpp @@ -1038,35 +1038,35 @@ TEST_CASE("MessagePack") CHECK_THROWS_AS(json::from_msgpack(std::vector({0xcf, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00})), json::parse_error); CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xcc})), - "[json.exception.parse_error.110] parse error at 2: cannot read 1 bytes from vector"); + "[json.exception.parse_error.110] parse error at 2: unexpected end of input"); CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xcd})), - "[json.exception.parse_error.110] parse error at 2: cannot read 2 bytes from vector"); + "[json.exception.parse_error.110] parse error at 2: unexpected end of input"); CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xcd, 0x00})), - "[json.exception.parse_error.110] parse error at 2: cannot read 2 bytes from vector"); + "[json.exception.parse_error.110] parse error at 3: unexpected end of input"); CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xce})), - "[json.exception.parse_error.110] parse error at 2: cannot read 4 bytes from vector"); + "[json.exception.parse_error.110] parse error at 2: unexpected end of input"); CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xce, 0x00})), - "[json.exception.parse_error.110] parse error at 2: cannot read 4 bytes from vector"); + "[json.exception.parse_error.110] parse error at 3: unexpected end of input"); CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xce, 0x00, 0x00})), - "[json.exception.parse_error.110] parse error at 2: cannot read 4 bytes from vector"); + "[json.exception.parse_error.110] parse error at 4: unexpected end of input"); CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xce, 0x00, 0x00, 0x00})), - "[json.exception.parse_error.110] parse error at 2: cannot read 4 bytes from vector"); + "[json.exception.parse_error.110] parse error at 5: unexpected end of input"); CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xcf})), - "[json.exception.parse_error.110] parse error at 2: cannot read 8 bytes from vector"); + "[json.exception.parse_error.110] parse error at 2: unexpected end of input"); CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xcf, 0x00})), - "[json.exception.parse_error.110] parse error at 2: cannot read 8 bytes from vector"); + "[json.exception.parse_error.110] parse error at 3: unexpected end of input"); CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xcf, 0x00, 0x00})), - "[json.exception.parse_error.110] parse error at 2: cannot read 8 bytes from vector"); + "[json.exception.parse_error.110] parse error at 4: unexpected end of input"); CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xcf, 0x00, 0x00, 0x00})), - "[json.exception.parse_error.110] parse error at 2: cannot read 8 bytes from vector"); + "[json.exception.parse_error.110] parse error at 5: unexpected end of input"); CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xcf, 0x00, 0x00, 0x00, 0x00})), - "[json.exception.parse_error.110] parse error at 2: cannot read 8 bytes from vector"); + "[json.exception.parse_error.110] parse error at 6: unexpected end of input"); CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xcf, 0x00, 0x00, 0x00, 0x00, 0x00})), - "[json.exception.parse_error.110] parse error at 2: cannot read 8 bytes from vector"); + "[json.exception.parse_error.110] parse error at 7: unexpected end of input"); CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xcf, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00})), - "[json.exception.parse_error.110] parse error at 2: cannot read 8 bytes from vector"); + "[json.exception.parse_error.110] parse error at 8: unexpected end of input"); CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xcf, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00})), - "[json.exception.parse_error.110] parse error at 2: cannot read 8 bytes from vector"); + "[json.exception.parse_error.110] parse error at 9: unexpected end of input"); } SECTION("unsupported bytes") diff --git a/test/src/unit-regression.cpp b/test/src/unit-regression.cpp index fb4f03d1..68ccda46 100644 --- a/test/src/unit-regression.cpp +++ b/test/src/unit-regression.cpp @@ -638,13 +638,13 @@ TEST_CASE("regression tests") std::vector vec1 {0xcb, 0x8f, 0x0a}; CHECK_THROWS_AS(json::from_msgpack(vec1), json::parse_error); CHECK_THROWS_WITH(json::from_msgpack(vec1), - "[json.exception.parse_error.110] parse error at 2: cannot read 8 bytes from vector"); + "[json.exception.parse_error.110] parse error at 4: unexpected end of input"); // related test case: incomplete float32 std::vector vec2 {0xca, 0x8f, 0x0a}; CHECK_THROWS_AS(json::from_msgpack(vec2), json::parse_error); CHECK_THROWS_WITH(json::from_msgpack(vec2), - "[json.exception.parse_error.110] parse error at 2: cannot read 4 bytes from vector"); + "[json.exception.parse_error.110] parse error at 4: unexpected end of input"); // related test case: incomplete Half-Precision Float (CBOR) std::vector vec3 {0xf9, 0x8f}; @@ -671,7 +671,7 @@ TEST_CASE("regression tests") std::vector vec1 {0x87}; CHECK_THROWS_AS(json::from_msgpack(vec1), json::parse_error); CHECK_THROWS_WITH(json::from_msgpack(vec1), - "[json.exception.parse_error.110] parse error at 2: cannot read 1 bytes from vector"); + "[json.exception.parse_error.110] parse error at 2: unexpected end of input"); // more test cases for MessagePack for (auto b : @@ -708,7 +708,7 @@ TEST_CASE("regression tests") "[json.exception.parse_error.110] parse error at 1: unexpected end of input"); CHECK_THROWS_AS(json::from_msgpack(vec2), json::parse_error); CHECK_THROWS_WITH(json::from_msgpack(vec2), - "[json.exception.parse_error.110] parse error at 1: cannot read 1 bytes from vector"); + "[json.exception.parse_error.110] parse error at 1: unexpected end of input"); } SECTION("issue #411 - Heap-buffer-overflow (OSS-Fuzz issue 366)") From c28bf823bce93b47afd05655b9eb535e7cfc05ec Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 4 Apr 2017 17:30:43 +0200 Subject: [PATCH 141/530] :hammer: added endianess check --- src/json.hpp | 42 +++++++++++++++++++++++++++++++----------- 1 file changed, 31 insertions(+), 11 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 6853d8c0..a26c69c5 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -9364,11 +9364,13 @@ class basic_json { public: explicit binary_reader(std::istream& i) - : ia(new cached_input_stream_adapter(i, 16384)) + : ia(new cached_input_stream_adapter(i, 16384)), + is_little_endian(little_endianess()) {} binary_reader(const char* buff, const size_t len) - : ia(new input_buffer_adapter(buff, len)) + : ia(new input_buffer_adapter(buff, len)), + is_little_endian(little_endianess()) {} ~binary_reader() @@ -9555,7 +9557,7 @@ class basic_json case 0x97: { basic_json result = value_t::array; - const auto len = static_cast(current - 0x80); + const auto len = static_cast(current & 0x1f); for (size_t i = 0; i < len; ++i) { result.push_back(parse_cbor()); @@ -9644,7 +9646,7 @@ class basic_json case 0xb7: { basic_json result = value_t::object; - const auto len = static_cast(current - 0xa0); + const auto len = static_cast(current & 0x1f); for (size_t i = 0; i < len; ++i) { get(); @@ -9780,7 +9782,7 @@ class basic_json default: // anything else (0xFF is handled inside the other types) { std::stringstream ss; - ss << std::hex << current; + ss << std::setw(2) << std::setfill('0') << std::hex << current; JSON_THROW(parse_error::create(112, chars_read, "error reading CBOR; last byte: 0x" + ss.str())); } } @@ -10182,20 +10184,26 @@ class basic_json default: // anything else { std::stringstream ss; - ss << std::hex << current; + ss << std::setw(2) << std::setfill('0') << std::hex << current; JSON_THROW(parse_error::create(112, chars_read, "error reading MessagePack; last byte: 0x" + ss.str())); } } } private: + // from http://stackoverflow.com/a/1001328/266378 + static bool little_endianess() + { + int num = 1; + return (*reinterpret_cast(&num) == 1); + } + int get() { ++chars_read; return (current = ia->get_character()); } - // todo: check if this breaks with endianess template T get_number() { @@ -10204,7 +10212,16 @@ class basic_json { get(); check_eof(); - vec[sizeof(T) - i - 1] = static_cast(current); + + // reverse byte order prior to conversion if necessary + if (is_little_endian) + { + vec[sizeof(T) - i - 1] = static_cast(current); + } + else + { + vec[i] = static_cast(current); + } } T result; @@ -10256,7 +10273,7 @@ class basic_json case 0x76: case 0x77: { - const auto len = static_cast(current - 0x60); + const auto len = static_cast(current & 0x1f); return get_string(len); } @@ -10298,7 +10315,7 @@ class basic_json default: { std::stringstream ss; - ss << std::hex << current; + ss << std::setw(2) << std::setfill('0') << std::hex << current; JSON_THROW(parse_error::create(113, chars_read, "expected a CBOR string; last byte: 0x" + ss.str())); } } @@ -10369,7 +10386,7 @@ class basic_json default: { std::stringstream ss; - ss << std::hex << current; + ss << std::setw(2) << std::setfill('0') << std::hex << current; JSON_THROW(parse_error::create(113, chars_read, "expected a MessagePack string; last byte: 0x" + ss.str())); } } @@ -10392,6 +10409,9 @@ class basic_json /// the number of characters read size_t chars_read = 0; + + /// whether we can assume little endianess + const bool is_little_endian = true; }; public: From 08fdfcca9a994fcf7022da250fa2c7328d3cf94b Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 4 Apr 2017 23:17:43 +0200 Subject: [PATCH 142/530] :hammer: implemented a binary writer --- src/json.hpp | 1443 ++++++++++++++++++++++++-------------------------- 1 file changed, 700 insertions(+), 743 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index a26c69c5..c04408b2 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -7423,746 +7423,6 @@ class basic_json /// @} - ////////////////////////////////////////// - // binary serialization/deserialization // - ////////////////////////////////////////// - - /// @name binary serialization/deserialization support - /// @{ - - private: - /*! - @note Some code in the switch cases has been copied, because otherwise - copilers would complain about implicit fallthrough and there is no - portable attribute to mute such warnings. - */ - template - static void add_to_vector(std::vector& vec, size_t bytes, const T number) - { - assert(bytes == 1 or bytes == 2 or bytes == 4 or bytes == 8); - - switch (bytes) - { - case 8: - { - vec.push_back(static_cast((static_cast(number) >> 070) & 0xff)); - vec.push_back(static_cast((static_cast(number) >> 060) & 0xff)); - vec.push_back(static_cast((static_cast(number) >> 050) & 0xff)); - vec.push_back(static_cast((static_cast(number) >> 040) & 0xff)); - vec.push_back(static_cast((number >> 030) & 0xff)); - vec.push_back(static_cast((number >> 020) & 0xff)); - vec.push_back(static_cast((number >> 010) & 0xff)); - vec.push_back(static_cast(number & 0xff)); - break; - } - - case 4: - { - vec.push_back(static_cast((number >> 030) & 0xff)); - vec.push_back(static_cast((number >> 020) & 0xff)); - vec.push_back(static_cast((number >> 010) & 0xff)); - vec.push_back(static_cast(number & 0xff)); - break; - } - - case 2: - { - vec.push_back(static_cast((number >> 010) & 0xff)); - vec.push_back(static_cast(number & 0xff)); - break; - } - - case 1: - { - vec.push_back(static_cast(number & 0xff)); - break; - } - } - } - - /*! - @brief create a MessagePack serialization of a given JSON value - - This is a straightforward implementation of the MessagePack specification. - - @param[in] j JSON value to serialize - @param[in,out] v byte vector to write the serialization to - - @sa https://github.com/msgpack/msgpack/blob/master/spec.md - */ - static void to_msgpack_internal(const basic_json& j, std::vector& v) - { - switch (j.type()) - { - case value_t::null: - { - // nil - v.push_back(0xc0); - break; - } - - case value_t::boolean: - { - // true and false - v.push_back(j.m_value.boolean ? 0xc3 : 0xc2); - break; - } - - case value_t::number_integer: - { - if (j.m_value.number_integer >= 0) - { - // MessagePack does not differentiate between positive - // signed integers and unsigned integers. Therefore, we - // used the code from the value_t::number_unsigned case - // here. - if (j.m_value.number_unsigned < 128) - { - // positive fixnum - add_to_vector(v, 1, j.m_value.number_unsigned); - } - else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) - { - // uint 8 - v.push_back(0xcc); - add_to_vector(v, 1, j.m_value.number_unsigned); - } - else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) - { - // uint 16 - v.push_back(0xcd); - add_to_vector(v, 2, j.m_value.number_unsigned); - } - else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) - { - // uint 32 - v.push_back(0xce); - add_to_vector(v, 4, j.m_value.number_unsigned); - } - else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) - { - // uint 64 - v.push_back(0xcf); - add_to_vector(v, 8, j.m_value.number_unsigned); - } - } - else - { - if (j.m_value.number_integer >= -32) - { - // negative fixnum - add_to_vector(v, 1, j.m_value.number_integer); - } - else if (j.m_value.number_integer >= (std::numeric_limits::min)() and j.m_value.number_integer <= (std::numeric_limits::max)()) - { - // int 8 - v.push_back(0xd0); - add_to_vector(v, 1, j.m_value.number_integer); - } - else if (j.m_value.number_integer >= (std::numeric_limits::min)() and j.m_value.number_integer <= (std::numeric_limits::max)()) - { - // int 16 - v.push_back(0xd1); - add_to_vector(v, 2, j.m_value.number_integer); - } - else if (j.m_value.number_integer >= (std::numeric_limits::min)() and j.m_value.number_integer <= (std::numeric_limits::max)()) - { - // int 32 - v.push_back(0xd2); - add_to_vector(v, 4, j.m_value.number_integer); - } - else if (j.m_value.number_integer >= (std::numeric_limits::min)() and j.m_value.number_integer <= (std::numeric_limits::max)()) - { - // int 64 - v.push_back(0xd3); - add_to_vector(v, 8, j.m_value.number_integer); - } - } - break; - } - - case value_t::number_unsigned: - { - if (j.m_value.number_unsigned < 128) - { - // positive fixnum - add_to_vector(v, 1, j.m_value.number_unsigned); - } - else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) - { - // uint 8 - v.push_back(0xcc); - add_to_vector(v, 1, j.m_value.number_unsigned); - } - else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) - { - // uint 16 - v.push_back(0xcd); - add_to_vector(v, 2, j.m_value.number_unsigned); - } - else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) - { - // uint 32 - v.push_back(0xce); - add_to_vector(v, 4, j.m_value.number_unsigned); - } - else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) - { - // uint 64 - v.push_back(0xcf); - add_to_vector(v, 8, j.m_value.number_unsigned); - } - break; - } - - case value_t::number_float: - { - // float 64 - v.push_back(0xcb); - const auto* helper = reinterpret_cast(&(j.m_value.number_float)); - for (size_t i = 0; i < 8; ++i) - { - v.push_back(helper[7 - i]); - } - break; - } - - case value_t::string: - { - const auto N = j.m_value.string->size(); - if (N <= 31) - { - // fixstr - v.push_back(static_cast(0xa0 | N)); - } - else if (N <= 255) - { - // str 8 - v.push_back(0xd9); - add_to_vector(v, 1, N); - } - else if (N <= 65535) - { - // str 16 - v.push_back(0xda); - add_to_vector(v, 2, N); - } - else if (N <= 4294967295) - { - // str 32 - v.push_back(0xdb); - add_to_vector(v, 4, N); - } - - // append string - std::copy(j.m_value.string->begin(), j.m_value.string->end(), - std::back_inserter(v)); - break; - } - - case value_t::array: - { - const auto N = j.m_value.array->size(); - if (N <= 15) - { - // fixarray - v.push_back(static_cast(0x90 | N)); - } - else if (N <= 0xffff) - { - // array 16 - v.push_back(0xdc); - add_to_vector(v, 2, N); - } - else if (N <= 0xffffffff) - { - // array 32 - v.push_back(0xdd); - add_to_vector(v, 4, N); - } - - // append each element - for (const auto& el : *j.m_value.array) - { - to_msgpack_internal(el, v); - } - break; - } - - case value_t::object: - { - const auto N = j.m_value.object->size(); - if (N <= 15) - { - // fixmap - v.push_back(static_cast(0x80 | (N & 0xf))); - } - else if (N <= 65535) - { - // map 16 - v.push_back(0xde); - add_to_vector(v, 2, N); - } - else if (N <= 4294967295) - { - // map 32 - v.push_back(0xdf); - add_to_vector(v, 4, N); - } - - // append each element - for (const auto& el : *j.m_value.object) - { - to_msgpack_internal(el.first, v); - to_msgpack_internal(el.second, v); - } - break; - } - - default: - { - break; - } - } - } - - /*! - @brief create a CBOR serialization of a given JSON value - - This is a straightforward implementation of the CBOR specification. - - @param[in] j JSON value to serialize - @param[in,out] v byte vector to write the serialization to - - @sa https://tools.ietf.org/html/rfc7049 - */ - static void to_cbor_internal(const basic_json& j, std::vector& v) - { - switch (j.type()) - { - case value_t::null: - { - v.push_back(0xf6); - break; - } - - case value_t::boolean: - { - v.push_back(j.m_value.boolean ? 0xf5 : 0xf4); - break; - } - - case value_t::number_integer: - { - if (j.m_value.number_integer >= 0) - { - // CBOR does not differentiate between positive signed - // integers and unsigned integers. Therefore, we used the - // code from the value_t::number_unsigned case here. - if (j.m_value.number_integer <= 0x17) - { - add_to_vector(v, 1, j.m_value.number_integer); - } - else if (j.m_value.number_integer <= (std::numeric_limits::max)()) - { - v.push_back(0x18); - // one-byte uint8_t - add_to_vector(v, 1, j.m_value.number_integer); - } - else if (j.m_value.number_integer <= (std::numeric_limits::max)()) - { - v.push_back(0x19); - // two-byte uint16_t - add_to_vector(v, 2, j.m_value.number_integer); - } - else if (j.m_value.number_integer <= (std::numeric_limits::max)()) - { - v.push_back(0x1a); - // four-byte uint32_t - add_to_vector(v, 4, j.m_value.number_integer); - } - else - { - v.push_back(0x1b); - // eight-byte uint64_t - add_to_vector(v, 8, j.m_value.number_integer); - } - } - else - { - // The conversions below encode the sign in the first - // byte, and the value is converted to a positive number. - const auto positive_number = -1 - j.m_value.number_integer; - if (j.m_value.number_integer >= -24) - { - v.push_back(static_cast(0x20 + positive_number)); - } - else if (positive_number <= (std::numeric_limits::max)()) - { - // int 8 - v.push_back(0x38); - add_to_vector(v, 1, positive_number); - } - else if (positive_number <= (std::numeric_limits::max)()) - { - // int 16 - v.push_back(0x39); - add_to_vector(v, 2, positive_number); - } - else if (positive_number <= (std::numeric_limits::max)()) - { - // int 32 - v.push_back(0x3a); - add_to_vector(v, 4, positive_number); - } - else - { - // int 64 - v.push_back(0x3b); - add_to_vector(v, 8, positive_number); - } - } - break; - } - - case value_t::number_unsigned: - { - if (j.m_value.number_unsigned <= 0x17) - { - v.push_back(static_cast(j.m_value.number_unsigned)); - } - else if (j.m_value.number_unsigned <= 0xff) - { - v.push_back(0x18); - // one-byte uint8_t - add_to_vector(v, 1, j.m_value.number_unsigned); - } - else if (j.m_value.number_unsigned <= 0xffff) - { - v.push_back(0x19); - // two-byte uint16_t - add_to_vector(v, 2, j.m_value.number_unsigned); - } - else if (j.m_value.number_unsigned <= 0xffffffff) - { - v.push_back(0x1a); - // four-byte uint32_t - add_to_vector(v, 4, j.m_value.number_unsigned); - } - else if (j.m_value.number_unsigned <= 0xffffffffffffffff) - { - v.push_back(0x1b); - // eight-byte uint64_t - add_to_vector(v, 8, j.m_value.number_unsigned); - } - break; - } - - case value_t::number_float: - { - // Double-Precision Float - v.push_back(0xfb); - const auto* helper = reinterpret_cast(&(j.m_value.number_float)); - for (size_t i = 0; i < 8; ++i) - { - v.push_back(helper[7 - i]); - } - break; - } - - case value_t::string: - { - const auto N = j.m_value.string->size(); - if (N <= 0x17) - { - v.push_back(static_cast(0x60 + N)); // 1 byte for string + size - } - else if (N <= 0xff) - { - v.push_back(0x78); // one-byte uint8_t for N - add_to_vector(v, 1, N); - } - else if (N <= 0xffff) - { - v.push_back(0x79); // two-byte uint16_t for N - add_to_vector(v, 2, N); - } - else if (N <= 0xffffffff) - { - v.push_back(0x7a); // four-byte uint32_t for N - add_to_vector(v, 4, N); - } - // LCOV_EXCL_START - else if (N <= 0xffffffffffffffff) - { - v.push_back(0x7b); // eight-byte uint64_t for N - add_to_vector(v, 8, N); - } - // LCOV_EXCL_STOP - - // append string - std::copy(j.m_value.string->begin(), j.m_value.string->end(), - std::back_inserter(v)); - break; - } - - case value_t::array: - { - const auto N = j.m_value.array->size(); - if (N <= 0x17) - { - v.push_back(static_cast(0x80 + N)); // 1 byte for array + size - } - else if (N <= 0xff) - { - v.push_back(0x98); // one-byte uint8_t for N - add_to_vector(v, 1, N); - } - else if (N <= 0xffff) - { - v.push_back(0x99); // two-byte uint16_t for N - add_to_vector(v, 2, N); - } - else if (N <= 0xffffffff) - { - v.push_back(0x9a); // four-byte uint32_t for N - add_to_vector(v, 4, N); - } - // LCOV_EXCL_START - else if (N <= 0xffffffffffffffff) - { - v.push_back(0x9b); // eight-byte uint64_t for N - add_to_vector(v, 8, N); - } - // LCOV_EXCL_STOP - - // append each element - for (const auto& el : *j.m_value.array) - { - to_cbor_internal(el, v); - } - break; - } - - case value_t::object: - { - const auto N = j.m_value.object->size(); - if (N <= 0x17) - { - v.push_back(static_cast(0xa0 + N)); // 1 byte for object + size - } - else if (N <= 0xff) - { - v.push_back(0xb8); - add_to_vector(v, 1, N); // one-byte uint8_t for N - } - else if (N <= 0xffff) - { - v.push_back(0xb9); - add_to_vector(v, 2, N); // two-byte uint16_t for N - } - else if (N <= 0xffffffff) - { - v.push_back(0xba); - add_to_vector(v, 4, N); // four-byte uint32_t for N - } - // LCOV_EXCL_START - else if (N <= 0xffffffffffffffff) - { - v.push_back(0xbb); - add_to_vector(v, 8, N); // eight-byte uint64_t for N - } - // LCOV_EXCL_STOP - - // append each element - for (const auto& el : *j.m_value.object) - { - to_cbor_internal(el.first, v); - to_cbor_internal(el.second, v); - } - break; - } - - default: - { - break; - } - } - } - - public: - /*! - @brief create a MessagePack serialization of a given JSON value - - Serializes a given JSON value @a j to a byte vector using the MessagePack - serialization format. MessagePack is a binary serialization format which - aims to be more compact than JSON itself, yet more efficient to parse. - - The library uses the following mapping from JSON values types to - MessagePack types according to the MessagePack specification: - - JSON value type | value/range | MessagePack type | first byte - --------------- | --------------------------------- | ---------------- | ---------- - null | `null` | nil | 0xc0 - boolean | `true` | true | 0xc3 - boolean | `false` | false | 0xc2 - number_integer | -9223372036854775808..-2147483649 | int64 | 0xd3 - number_integer | -2147483648..-32769 | int32 | 0xd2 - number_integer | -32768..-129 | int16 | 0xd1 - number_integer | -128..-33 | int8 | 0xd0 - number_integer | -32..-1 | negative fixint | 0xe0..0xff - number_integer | 0..127 | positive fixint | 0x00..0x7f - number_integer | 128..255 | uint 8 | 0xcc - number_integer | 256..65535 | uint 16 | 0xcd - number_integer | 65536..4294967295 | uint 32 | 0xce - number_integer | 4294967296..18446744073709551615 | uint 64 | 0xcf - number_unsigned | 0..127 | positive fixint | 0x00..0x7f - number_unsigned | 128..255 | uint 8 | 0xcc - number_unsigned | 256..65535 | uint 16 | 0xcd - number_unsigned | 65536..4294967295 | uint 32 | 0xce - number_unsigned | 4294967296..18446744073709551615 | uint 64 | 0xcf - number_float | *any value* | float 64 | 0xcb - string | *length*: 0..31 | fixstr | 0xa0..0xbf - string | *length*: 32..255 | str 8 | 0xd9 - string | *length*: 256..65535 | str 16 | 0xda - string | *length*: 65536..4294967295 | str 32 | 0xdb - array | *size*: 0..15 | fixarray | 0x90..0x9f - array | *size*: 16..65535 | array 16 | 0xdc - array | *size*: 65536..4294967295 | array 32 | 0xdd - object | *size*: 0..15 | fix map | 0x80..0x8f - object | *size*: 16..65535 | map 16 | 0xde - object | *size*: 65536..4294967295 | map 32 | 0xdf - - @note The mapping is **complete** in the sense that any JSON value type - can be converted to a MessagePack value. - - @note The following values can **not** be converted to a MessagePack value: - - strings with more than 4294967295 bytes - - arrays with more than 4294967295 elements - - objects with more than 4294967295 elements - - @note The following MessagePack types are not used in the conversion: - - bin 8 - bin 32 (0xc4..0xc6) - - ext 8 - ext 32 (0xc7..0xc9) - - float 32 (0xca) - - fixext 1 - fixext 16 (0xd4..0xd8) - - @note Any MessagePack output created @ref to_msgpack can be successfully - parsed by @ref from_msgpack. - - @param[in] j JSON value to serialize - @return MessagePack serialization as byte vector - - @complexity Linear in the size of the JSON value @a j. - - @liveexample{The example shows the serialization of a JSON value to a byte - vector in MessagePack format.,to_msgpack} - - @sa http://msgpack.org - @sa @ref from_msgpack(const std::vector&, const size_t) for the - analogous deserialization - @sa @ref to_cbor(const basic_json& for the related CBOR format - - @since version 2.0.9 - */ - static std::vector to_msgpack(const basic_json& j) - { - std::vector result; - to_msgpack_internal(j, result); - return result; - } - - /*! - @brief create a CBOR serialization of a given JSON value - - Serializes a given JSON value @a j to a byte vector using the CBOR (Concise - Binary Object Representation) serialization format. CBOR is a binary - serialization format which aims to be more compact than JSON itself, yet - more efficient to parse. - - The library uses the following mapping from JSON values types to - CBOR types according to the CBOR specification (RFC 7049): - - JSON value type | value/range | CBOR type | first byte - --------------- | ------------------------------------------ | ---------------------------------- | --------------- - null | `null` | Null | 0xf6 - boolean | `true` | True | 0xf5 - boolean | `false` | False | 0xf4 - number_integer | -9223372036854775808..-2147483649 | Negative integer (8 bytes follow) | 0x3b - number_integer | -2147483648..-32769 | Negative integer (4 bytes follow) | 0x3a - number_integer | -32768..-129 | Negative integer (2 bytes follow) | 0x39 - number_integer | -128..-25 | Negative integer (1 byte follow) | 0x38 - number_integer | -24..-1 | Negative integer | 0x20..0x37 - number_integer | 0..23 | Integer | 0x00..0x17 - number_integer | 24..255 | Unsigned integer (1 byte follow) | 0x18 - number_integer | 256..65535 | Unsigned integer (2 bytes follow) | 0x19 - number_integer | 65536..4294967295 | Unsigned integer (4 bytes follow) | 0x1a - number_integer | 4294967296..18446744073709551615 | Unsigned integer (8 bytes follow) | 0x1b - number_unsigned | 0..23 | Integer | 0x00..0x17 - number_unsigned | 24..255 | Unsigned integer (1 byte follow) | 0x18 - number_unsigned | 256..65535 | Unsigned integer (2 bytes follow) | 0x19 - number_unsigned | 65536..4294967295 | Unsigned integer (4 bytes follow) | 0x1a - number_unsigned | 4294967296..18446744073709551615 | Unsigned integer (8 bytes follow) | 0x1b - number_float | *any value* | Double-Precision Float | 0xfb - string | *length*: 0..23 | UTF-8 string | 0x60..0x77 - string | *length*: 23..255 | UTF-8 string (1 byte follow) | 0x78 - string | *length*: 256..65535 | UTF-8 string (2 bytes follow) | 0x79 - string | *length*: 65536..4294967295 | UTF-8 string (4 bytes follow) | 0x7a - string | *length*: 4294967296..18446744073709551615 | UTF-8 string (8 bytes follow) | 0x7b - array | *size*: 0..23 | array | 0x80..0x97 - array | *size*: 23..255 | array (1 byte follow) | 0x98 - array | *size*: 256..65535 | array (2 bytes follow) | 0x99 - array | *size*: 65536..4294967295 | array (4 bytes follow) | 0x9a - array | *size*: 4294967296..18446744073709551615 | array (8 bytes follow) | 0x9b - object | *size*: 0..23 | map | 0xa0..0xb7 - object | *size*: 23..255 | map (1 byte follow) | 0xb8 - object | *size*: 256..65535 | map (2 bytes follow) | 0xb9 - object | *size*: 65536..4294967295 | map (4 bytes follow) | 0xba - object | *size*: 4294967296..18446744073709551615 | map (8 bytes follow) | 0xbb - - @note The mapping is **complete** in the sense that any JSON value type - can be converted to a CBOR value. - - @note The following CBOR types are not used in the conversion: - - byte strings (0x40..0x5f) - - UTF-8 strings terminated by "break" (0x7f) - - arrays terminated by "break" (0x9f) - - maps terminated by "break" (0xbf) - - date/time (0xc0..0xc1) - - bignum (0xc2..0xc3) - - decimal fraction (0xc4) - - bigfloat (0xc5) - - tagged items (0xc6..0xd4, 0xd8..0xdb) - - expected conversions (0xd5..0xd7) - - simple values (0xe0..0xf3, 0xf8) - - undefined (0xf7) - - half and single-precision floats (0xf9-0xfa) - - break (0xff) - - @param[in] j JSON value to serialize - @return MessagePack serialization as byte vector - - @complexity Linear in the size of the JSON value @a j. - - @liveexample{The example shows the serialization of a JSON value to a byte - vector in CBOR format.,to_cbor} - - @sa http://cbor.io - @sa @ref from_cbor(const std::vector&, const size_t) for the - analogous deserialization - @sa @ref to_msgpack(const basic_json& for the related MessagePack format - - @since version 2.0.9 - */ - static std::vector to_cbor(const basic_json& j) - { - std::vector result; - to_cbor_internal(j, result); - return result; - } - - /// @} - /////////////////////////// // convenience functions // /////////////////////////// @@ -9355,9 +8615,12 @@ class basic_json const char* start; }; - //////////////////// - // binary formats // - //////////////////// + ////////////////////////////////////////// + // binary serialization/deserialization // + ////////////////////////////////////////// + + /// @name binary serialization/deserialization support + /// @{ private: class binary_reader @@ -10414,7 +9677,699 @@ class basic_json const bool is_little_endian = true; }; + class binary_writer + { + public: + binary_writer() + : is_little_endian(little_endianess()) + {} + + std::vector write_cbor(const basic_json& j) + { + write_cbor_internal(j); + return v; + } + + std::vector write_msgpack(const basic_json& j) + { + write_msgpack_internal(j); + return v; + } + + private: + void write_cbor_internal(const basic_json& j) + { + switch (j.type()) + { + case value_t::null: + { + v.push_back(0xf6); + break; + } + + case value_t::boolean: + { + v.push_back(j.m_value.boolean ? 0xf5 : 0xf4); + break; + } + + case value_t::number_integer: + { + if (j.m_value.number_integer >= 0) + { + // CBOR does not differentiate between positive signed + // integers and unsigned integers. Therefore, we used the + // code from the value_t::number_unsigned case here. + if (j.m_value.number_integer <= 0x17) + { + write_number(static_cast(j.m_value.number_integer)); + } + else if (j.m_value.number_integer <= (std::numeric_limits::max)()) + { + v.push_back(0x18); + write_number(static_cast(j.m_value.number_integer)); + } + else if (j.m_value.number_integer <= (std::numeric_limits::max)()) + { + v.push_back(0x19); + write_number(static_cast(j.m_value.number_integer)); + } + else if (j.m_value.number_integer <= (std::numeric_limits::max)()) + { + v.push_back(0x1a); + write_number(static_cast(j.m_value.number_integer)); + } + else + { + v.push_back(0x1b); + write_number(static_cast(j.m_value.number_integer)); + } + } + else + { + // The conversions below encode the sign in the first + // byte, and the value is converted to a positive number. + const auto positive_number = -1 - j.m_value.number_integer; + if (j.m_value.number_integer >= -24) + { + write_number(static_cast(0x20 + positive_number)); + } + else if (positive_number <= (std::numeric_limits::max)()) + { + v.push_back(0x38); + write_number(static_cast(positive_number)); + } + else if (positive_number <= (std::numeric_limits::max)()) + { + v.push_back(0x39); + write_number(static_cast(positive_number)); + } + else if (positive_number <= (std::numeric_limits::max)()) + { + v.push_back(0x3a); + write_number(static_cast(positive_number)); + } + else + { + v.push_back(0x3b); + write_number(static_cast(positive_number)); + } + } + break; + } + + case value_t::number_unsigned: + { + if (j.m_value.number_unsigned <= 0x17) + { + write_number(static_cast(j.m_value.number_unsigned)); + } + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + { + v.push_back(0x18); + write_number(static_cast(j.m_value.number_unsigned)); + } + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + { + v.push_back(0x19); + write_number(static_cast(j.m_value.number_unsigned)); + } + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + { + v.push_back(0x1a); + write_number(static_cast(j.m_value.number_unsigned)); + } + else + { + v.push_back(0x1b); + write_number(static_cast(j.m_value.number_unsigned)); + } + break; + } + + case value_t::number_float: + { + // Double-Precision Float + v.push_back(0xfb); + write_number(j.m_value.number_float); + break; + } + + case value_t::string: + { + const auto N = j.m_value.string->size(); + if (N <= 0x17) + { + write_number(static_cast(0x60 + N)); + } + else if (N <= 0xff) + { + v.push_back(0x78); + write_number(static_cast(N)); + } + else if (N <= 0xffff) + { + v.push_back(0x79); + write_number(static_cast(N)); + } + else if (N <= 0xffffffff) + { + v.push_back(0x7a); + write_number(static_cast(N)); + } + // LCOV_EXCL_START + else if (N <= 0xffffffffffffffff) + { + v.push_back(0x7b); + write_number(static_cast(N)); + } + // LCOV_EXCL_STOP + + // append string + std::copy(j.m_value.string->begin(), j.m_value.string->end(), + std::back_inserter(v)); + break; + } + + case value_t::array: + { + const auto N = j.m_value.array->size(); + if (N <= 0x17) + { + write_number(static_cast(0x80 + N)); + } + else if (N <= 0xff) + { + v.push_back(0x98); + write_number(static_cast(N)); + } + else if (N <= 0xffff) + { + v.push_back(0x99); + write_number(static_cast(N)); + } + else if (N <= 0xffffffff) + { + v.push_back(0x9a); + write_number(static_cast(N)); + } + // LCOV_EXCL_START + else if (N <= 0xffffffffffffffff) + { + v.push_back(0x9b); + write_number(static_cast(N)); + } + // LCOV_EXCL_STOP + + // append each element + for (const auto& el : *j.m_value.array) + { + write_cbor_internal(el); + } + break; + } + + case value_t::object: + { + const auto N = j.m_value.object->size(); + if (N <= 0x17) + { + write_number(static_cast(0xa0 + N)); + } + else if (N <= 0xff) + { + v.push_back(0xb8); + write_number(static_cast(N)); + } + else if (N <= 0xffff) + { + v.push_back(0xb9); + write_number(static_cast(N)); + } + else if (N <= 0xffffffff) + { + v.push_back(0xba); + write_number(static_cast(N)); + } + // LCOV_EXCL_START + else if (N <= 0xffffffffffffffff) + { + v.push_back(0xbb); + write_number(static_cast(N)); + } + // LCOV_EXCL_STOP + + // append each element + for (const auto& el : *j.m_value.object) + { + write_cbor_internal(el.first); + write_cbor_internal(el.second); + } + break; + } + + default: + { + break; + } + } + } + + void write_msgpack_internal(const basic_json& j) + { + switch (j.type()) + { + case value_t::null: + { + // nil + v.push_back(0xc0); + break; + } + + case value_t::boolean: + { + // true and false + v.push_back(j.m_value.boolean ? 0xc3 : 0xc2); + break; + } + + case value_t::number_integer: + { + if (j.m_value.number_integer >= 0) + { + // MessagePack does not differentiate between positive + // signed integers and unsigned integers. Therefore, we + // used the code from the value_t::number_unsigned case + // here. + if (j.m_value.number_unsigned < 128) + { + // positive fixnum + write_number(static_cast(j.m_value.number_integer)); + } + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + { + // uint 8 + v.push_back(0xcc); + write_number(static_cast(j.m_value.number_integer)); + } + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + { + // uint 16 + v.push_back(0xcd); + write_number(static_cast(j.m_value.number_integer)); + } + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + { + // uint 32 + v.push_back(0xce); + write_number(static_cast(j.m_value.number_integer)); + } + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + { + // uint 64 + v.push_back(0xcf); + write_number(static_cast(j.m_value.number_integer)); + } + } + else + { + if (j.m_value.number_integer >= -32) + { + // negative fixnum + write_number(static_cast(j.m_value.number_integer)); + } + else if (j.m_value.number_integer >= (std::numeric_limits::min)() and j.m_value.number_integer <= (std::numeric_limits::max)()) + { + // int 8 + v.push_back(0xd0); + write_number(static_cast(j.m_value.number_integer)); + } + else if (j.m_value.number_integer >= (std::numeric_limits::min)() and j.m_value.number_integer <= (std::numeric_limits::max)()) + { + // int 16 + v.push_back(0xd1); + write_number(static_cast(j.m_value.number_integer)); + } + else if (j.m_value.number_integer >= (std::numeric_limits::min)() and j.m_value.number_integer <= (std::numeric_limits::max)()) + { + // int 32 + v.push_back(0xd2); + write_number(static_cast(j.m_value.number_integer)); + } + else if (j.m_value.number_integer >= (std::numeric_limits::min)() and j.m_value.number_integer <= (std::numeric_limits::max)()) + { + // int 64 + v.push_back(0xd3); + write_number(static_cast(j.m_value.number_integer)); + } + } + break; + } + + case value_t::number_unsigned: + { + if (j.m_value.number_unsigned < 128) + { + // positive fixnum + write_number(static_cast(j.m_value.number_integer)); + } + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + { + // uint 8 + v.push_back(0xcc); + write_number(static_cast(j.m_value.number_integer)); + } + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + { + // uint 16 + v.push_back(0xcd); + write_number(static_cast(j.m_value.number_integer)); + } + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + { + // uint 32 + v.push_back(0xce); + write_number(static_cast(j.m_value.number_integer)); + } + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + { + // uint 64 + v.push_back(0xcf); + write_number(static_cast(j.m_value.number_integer)); + } + break; + } + + case value_t::number_float: + { + // float 64 + v.push_back(0xcb); + write_number(j.m_value.number_float); + break; + } + + case value_t::string: + { + const auto N = j.m_value.string->size(); + if (N <= 31) + { + // fixstr + write_number(static_cast(0xa0 | N)); + } + else if (N <= 255) + { + // str 8 + v.push_back(0xd9); + write_number(static_cast(N)); + } + else if (N <= 65535) + { + // str 16 + v.push_back(0xda); + write_number(static_cast(N)); + } + else if (N <= 4294967295) + { + // str 32 + v.push_back(0xdb); + write_number(static_cast(N)); + } + + // append string + std::copy(j.m_value.string->begin(), j.m_value.string->end(), + std::back_inserter(v)); + break; + } + + case value_t::array: + { + const auto N = j.m_value.array->size(); + if (N <= 15) + { + // fixarray + write_number(static_cast(0x90 | N)); + } + else if (N <= 0xffff) + { + // array 16 + v.push_back(0xdc); + write_number(static_cast(N)); + } + else if (N <= 0xffffffff) + { + // array 32 + v.push_back(0xdd); + write_number(static_cast(N)); + } + + // append each element + for (const auto& el : *j.m_value.array) + { + write_msgpack_internal(el); + } + break; + } + + case value_t::object: + { + const auto N = j.m_value.object->size(); + if (N <= 15) + { + // fixmap + write_number(static_cast(0x80 | (N & 0xf))); + } + else if (N <= 65535) + { + // map 16 + v.push_back(0xde); + write_number(static_cast(N)); + } + else if (N <= 4294967295) + { + // map 32 + v.push_back(0xdf); + write_number(static_cast(N)); + } + + // append each element + for (const auto& el : *j.m_value.object) + { + write_msgpack_internal(el.first); + write_msgpack_internal(el.second); + } + break; + } + + default: + { + break; + } + } + } + + template + void write_number(T n) + { + std::array vec; + std::memcpy(vec.data(), &n, sizeof(T)); + + for (size_t i = 0; i < sizeof(T); ++i) + { + // reverse byte order prior to conversion if necessary + if (is_little_endian) + { + v.push_back(vec[sizeof(T) - i - 1]); + } + else + { + v.push_back(vec[i]); + } + } + } + + // from http://stackoverflow.com/a/1001328/266378 + static bool little_endianess() + { + int num = 1; + return (*reinterpret_cast(&num) == 1); + } + + private: + /// whether we can assume little endianess + const bool is_little_endian = true; + + /// the vector that is used as output + std::vector v; + }; + public: + /*! + @brief create a CBOR serialization of a given JSON value + + Serializes a given JSON value @a j to a byte vector using the CBOR (Concise + Binary Object Representation) serialization format. CBOR is a binary + serialization format which aims to be more compact than JSON itself, yet + more efficient to parse. + + The library uses the following mapping from JSON values types to + CBOR types according to the CBOR specification (RFC 7049): + + JSON value type | value/range | CBOR type | first byte + --------------- | ------------------------------------------ | ---------------------------------- | --------------- + null | `null` | Null | 0xf6 + boolean | `true` | True | 0xf5 + boolean | `false` | False | 0xf4 + number_integer | -9223372036854775808..-2147483649 | Negative integer (8 bytes follow) | 0x3b + number_integer | -2147483648..-32769 | Negative integer (4 bytes follow) | 0x3a + number_integer | -32768..-129 | Negative integer (2 bytes follow) | 0x39 + number_integer | -128..-25 | Negative integer (1 byte follow) | 0x38 + number_integer | -24..-1 | Negative integer | 0x20..0x37 + number_integer | 0..23 | Integer | 0x00..0x17 + number_integer | 24..255 | Unsigned integer (1 byte follow) | 0x18 + number_integer | 256..65535 | Unsigned integer (2 bytes follow) | 0x19 + number_integer | 65536..4294967295 | Unsigned integer (4 bytes follow) | 0x1a + number_integer | 4294967296..18446744073709551615 | Unsigned integer (8 bytes follow) | 0x1b + number_unsigned | 0..23 | Integer | 0x00..0x17 + number_unsigned | 24..255 | Unsigned integer (1 byte follow) | 0x18 + number_unsigned | 256..65535 | Unsigned integer (2 bytes follow) | 0x19 + number_unsigned | 65536..4294967295 | Unsigned integer (4 bytes follow) | 0x1a + number_unsigned | 4294967296..18446744073709551615 | Unsigned integer (8 bytes follow) | 0x1b + number_float | *any value* | Double-Precision Float | 0xfb + string | *length*: 0..23 | UTF-8 string | 0x60..0x77 + string | *length*: 23..255 | UTF-8 string (1 byte follow) | 0x78 + string | *length*: 256..65535 | UTF-8 string (2 bytes follow) | 0x79 + string | *length*: 65536..4294967295 | UTF-8 string (4 bytes follow) | 0x7a + string | *length*: 4294967296..18446744073709551615 | UTF-8 string (8 bytes follow) | 0x7b + array | *size*: 0..23 | array | 0x80..0x97 + array | *size*: 23..255 | array (1 byte follow) | 0x98 + array | *size*: 256..65535 | array (2 bytes follow) | 0x99 + array | *size*: 65536..4294967295 | array (4 bytes follow) | 0x9a + array | *size*: 4294967296..18446744073709551615 | array (8 bytes follow) | 0x9b + object | *size*: 0..23 | map | 0xa0..0xb7 + object | *size*: 23..255 | map (1 byte follow) | 0xb8 + object | *size*: 256..65535 | map (2 bytes follow) | 0xb9 + object | *size*: 65536..4294967295 | map (4 bytes follow) | 0xba + object | *size*: 4294967296..18446744073709551615 | map (8 bytes follow) | 0xbb + + @note The mapping is **complete** in the sense that any JSON value type + can be converted to a CBOR value. + + @note The following CBOR types are not used in the conversion: + - byte strings (0x40..0x5f) + - UTF-8 strings terminated by "break" (0x7f) + - arrays terminated by "break" (0x9f) + - maps terminated by "break" (0xbf) + - date/time (0xc0..0xc1) + - bignum (0xc2..0xc3) + - decimal fraction (0xc4) + - bigfloat (0xc5) + - tagged items (0xc6..0xd4, 0xd8..0xdb) + - expected conversions (0xd5..0xd7) + - simple values (0xe0..0xf3, 0xf8) + - undefined (0xf7) + - half and single-precision floats (0xf9-0xfa) + - break (0xff) + + @param[in] j JSON value to serialize + @return MessagePack serialization as byte vector + + @complexity Linear in the size of the JSON value @a j. + + @liveexample{The example shows the serialization of a JSON value to a byte + vector in CBOR format.,to_cbor} + + @sa http://cbor.io + @sa @ref from_cbor(const std::vector&, const size_t) for the + analogous deserialization + @sa @ref to_msgpack(const basic_json& for the related MessagePack format + + @since version 2.0.9 + */ + static std::vector to_cbor(const basic_json& j) + { + binary_writer bw; + return bw.write_cbor(j); + } + + /*! + @brief create a MessagePack serialization of a given JSON value + + Serializes a given JSON value @a j to a byte vector using the MessagePack + serialization format. MessagePack is a binary serialization format which + aims to be more compact than JSON itself, yet more efficient to parse. + + The library uses the following mapping from JSON values types to + MessagePack types according to the MessagePack specification: + + JSON value type | value/range | MessagePack type | first byte + --------------- | --------------------------------- | ---------------- | ---------- + null | `null` | nil | 0xc0 + boolean | `true` | true | 0xc3 + boolean | `false` | false | 0xc2 + number_integer | -9223372036854775808..-2147483649 | int64 | 0xd3 + number_integer | -2147483648..-32769 | int32 | 0xd2 + number_integer | -32768..-129 | int16 | 0xd1 + number_integer | -128..-33 | int8 | 0xd0 + number_integer | -32..-1 | negative fixint | 0xe0..0xff + number_integer | 0..127 | positive fixint | 0x00..0x7f + number_integer | 128..255 | uint 8 | 0xcc + number_integer | 256..65535 | uint 16 | 0xcd + number_integer | 65536..4294967295 | uint 32 | 0xce + number_integer | 4294967296..18446744073709551615 | uint 64 | 0xcf + number_unsigned | 0..127 | positive fixint | 0x00..0x7f + number_unsigned | 128..255 | uint 8 | 0xcc + number_unsigned | 256..65535 | uint 16 | 0xcd + number_unsigned | 65536..4294967295 | uint 32 | 0xce + number_unsigned | 4294967296..18446744073709551615 | uint 64 | 0xcf + number_float | *any value* | float 64 | 0xcb + string | *length*: 0..31 | fixstr | 0xa0..0xbf + string | *length*: 32..255 | str 8 | 0xd9 + string | *length*: 256..65535 | str 16 | 0xda + string | *length*: 65536..4294967295 | str 32 | 0xdb + array | *size*: 0..15 | fixarray | 0x90..0x9f + array | *size*: 16..65535 | array 16 | 0xdc + array | *size*: 65536..4294967295 | array 32 | 0xdd + object | *size*: 0..15 | fix map | 0x80..0x8f + object | *size*: 16..65535 | map 16 | 0xde + object | *size*: 65536..4294967295 | map 32 | 0xdf + + @note The mapping is **complete** in the sense that any JSON value type + can be converted to a MessagePack value. + + @note The following values can **not** be converted to a MessagePack value: + - strings with more than 4294967295 bytes + - arrays with more than 4294967295 elements + - objects with more than 4294967295 elements + + @note The following MessagePack types are not used in the conversion: + - bin 8 - bin 32 (0xc4..0xc6) + - ext 8 - ext 32 (0xc7..0xc9) + - float 32 (0xca) + - fixext 1 - fixext 16 (0xd4..0xd8) + + @note Any MessagePack output created @ref to_msgpack can be successfully + parsed by @ref from_msgpack. + + @param[in] j JSON value to serialize + @return MessagePack serialization as byte vector + + @complexity Linear in the size of the JSON value @a j. + + @liveexample{The example shows the serialization of a JSON value to a byte + vector in MessagePack format.,to_msgpack} + + @sa http://msgpack.org + @sa @ref from_msgpack(const std::vector&, const size_t) for the + analogous deserialization + @sa @ref to_cbor(const basic_json& for the related CBOR format + + @since version 2.0.9 + */ + static std::vector to_msgpack(const basic_json& j) + { + binary_writer bw; + return bw.write_msgpack(j); + } /*! @brief create a JSON value from a byte vector in CBOR format @@ -10585,6 +10540,8 @@ class basic_json return br.parse_msgpack(); } + /// @} + ////////////////////// // lexer and parser // ////////////////////// From b992acc2e78578b2106a427d23013cba75841997 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Wed, 5 Apr 2017 20:39:27 +0200 Subject: [PATCH 143/530] :hammer: fixed a compiler warning --- src/json.hpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/json.hpp b/src/json.hpp index c04408b2..ff3fe826 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -10199,7 +10199,7 @@ class basic_json const bool is_little_endian = true; /// the vector that is used as output - std::vector v; + std::vector v {}; }; public: From ff72f3886321eef8e61f9e1b64ccb19032e18549 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Thu, 6 Apr 2017 19:54:08 +0200 Subject: [PATCH 144/530] :hammer: fixed another warning MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Do not store eof() in a char buffer… --- src/json.hpp | 39 +++++++++++++++++++++++++-------------- 1 file changed, 25 insertions(+), 14 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index ff3fe826..97addceb 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -8485,8 +8485,7 @@ class basic_json { public: cached_input_stream_adapter(std::istream& i, const size_t buffer_size) - : is(i), start_position(is.tellg()), - buffer(buffer_size, std::char_traits::eof()) + : is(i), start_position(is.tellg()), buffer(buffer_size, '\0') { // immediately abort if stream is erroneous if (JSON_UNLIKELY(i.fail())) @@ -8494,12 +8493,13 @@ class basic_json JSON_THROW(parse_error::create(111, 0, "bad input stream")); } - // initial fill; unfilled buffer characters remain EOF + // initial fill is.read(buffer.data(), static_cast(buffer.size())); + // store number of bytes in the buffer + fill_size = static_cast(is.gcount()); // skip byte-order mark - assert(buffer.size() >= 3); - if (buffer[0] == '\xEF' and buffer[1] == '\xBB' and buffer[2] == '\xBF') + if (fill_size >= 3 and buffer[0] == '\xEF' and buffer[1] == '\xBB' and buffer[2] == '\xBF') { buffer_pos += 3; processed_chars += 3; @@ -8516,22 +8516,28 @@ class basic_json int get_character() override { - // check if refilling is necessary - if (JSON_UNLIKELY(buffer_pos == buffer.size())) + // check if refilling is necessary and possible + if (buffer_pos == fill_size and not eof) { // refill - is.read(reinterpret_cast(buffer.data()), static_cast(buffer.size())); - // set unfilled characters to EOF - std::fill_n(buffer.begin() + static_cast(is.gcount()), - buffer.size() - static_cast(is.gcount()), - std::char_traits::eof()); + is.read(buffer.data(), static_cast(buffer.size())); + // store number of bytes in the buffer + fill_size = static_cast(is.gcount()); + + // remember that filling did not yield new input + if (fill_size == 0) + { + eof = true; + } + // the buffer is ready buffer_pos = 0; } ++processed_chars; - const int res = buffer[buffer_pos++]; - return (res == std::char_traits::eof()) ? res : res & 0xFF; + return eof + ? std::char_traits::eof() + : buffer[buffer_pos++] & 0xFF; } std::string read(size_t offset, size_t length) override @@ -8568,6 +8574,11 @@ class basic_json /// chars processed in the current buffer size_t buffer_pos = 0; + /// whether stream reached eof + bool eof = false; + /// how many chars have been copied to the buffer by last (re)fill + size_t fill_size = 0; + /// position of the stream when we started const std::streampos start_position; From 90273e930c135a3f893ef2e303267f76b9b246b5 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Fri, 7 Apr 2017 15:44:41 +0200 Subject: [PATCH 145/530] :bug: fix for #512 We totally forgot to implement the comparison operators other than == and != for scalar types. Consequently, comparing a JSON value with a scalar type led to compile errors. --- src/json.hpp | 88 ++++++++++++++++++++++++++++++++++++ src/json.hpp.re2c | 88 ++++++++++++++++++++++++++++++++++++ test/src/unit-regression.cpp | 35 ++++++++++++++ 3 files changed, 211 insertions(+) diff --git a/src/json.hpp b/src/json.hpp index 5df1bacc..98659054 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -6385,6 +6385,28 @@ class basic_json return operator<(lhs_type, rhs_type); } + /*! + @brief comparison: less than + @copydoc operator<(const_reference, const_reference) + */ + template::value, int>::type = 0> + friend bool operator<(const_reference lhs, const ScalarType rhs) noexcept + { + return (lhs < basic_json(rhs)); + } + + /*! + @brief comparison: less than + @copydoc operator<(const_reference, const_reference) + */ + template::value, int>::type = 0> + friend bool operator<(const ScalarType lhs, const_reference rhs) noexcept + { + return (basic_json(lhs) < rhs); + } + /*! @brief comparison: less than or equal @@ -6407,6 +6429,28 @@ class basic_json return not (rhs < lhs); } + /*! + @brief comparison: less than or equal + @copydoc operator<=(const_reference, const_reference) + */ + template::value, int>::type = 0> + friend bool operator<=(const_reference lhs, const ScalarType rhs) noexcept + { + return (lhs <= basic_json(rhs)); + } + + /*! + @brief comparison: less than or equal + @copydoc operator<=(const_reference, const_reference) + */ + template::value, int>::type = 0> + friend bool operator<=(const ScalarType lhs, const_reference rhs) noexcept + { + return (basic_json(lhs) <= rhs); + } + /*! @brief comparison: greater than @@ -6429,6 +6473,28 @@ class basic_json return not (lhs <= rhs); } + /*! + @brief comparison: greater than + @copydoc operator>(const_reference, const_reference) + */ + template::value, int>::type = 0> + friend bool operator>(const_reference lhs, const ScalarType rhs) noexcept + { + return (lhs > basic_json(rhs)); + } + + /*! + @brief comparison: greater than + @copydoc operator>(const_reference, const_reference) + */ + template::value, int>::type = 0> + friend bool operator>(const ScalarType lhs, const_reference rhs) noexcept + { + return (basic_json(lhs) > rhs); + } + /*! @brief comparison: greater than or equal @@ -6451,6 +6517,28 @@ class basic_json return not (lhs < rhs); } + /*! + @brief comparison: greater than or equal + @copydoc operator>=(const_reference, const_reference) + */ + template::value, int>::type = 0> + friend bool operator>=(const_reference lhs, const ScalarType rhs) noexcept + { + return (lhs >= basic_json(rhs)); + } + + /*! + @brief comparison: greater than or equal + @copydoc operator>=(const_reference, const_reference) + */ + template::value, int>::type = 0> + friend bool operator>=(const ScalarType lhs, const_reference rhs) noexcept + { + return (basic_json(lhs) >= rhs); + } + /// @} diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index da503495..5e90f611 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -6385,6 +6385,28 @@ class basic_json return operator<(lhs_type, rhs_type); } + /*! + @brief comparison: less than + @copydoc operator<(const_reference, const_reference) + */ + template::value, int>::type = 0> + friend bool operator<(const_reference lhs, const ScalarType rhs) noexcept + { + return (lhs < basic_json(rhs)); + } + + /*! + @brief comparison: less than + @copydoc operator<(const_reference, const_reference) + */ + template::value, int>::type = 0> + friend bool operator<(const ScalarType lhs, const_reference rhs) noexcept + { + return (basic_json(lhs) < rhs); + } + /*! @brief comparison: less than or equal @@ -6407,6 +6429,28 @@ class basic_json return not (rhs < lhs); } + /*! + @brief comparison: less than or equal + @copydoc operator<=(const_reference, const_reference) + */ + template::value, int>::type = 0> + friend bool operator<=(const_reference lhs, const ScalarType rhs) noexcept + { + return (lhs <= basic_json(rhs)); + } + + /*! + @brief comparison: less than or equal + @copydoc operator<=(const_reference, const_reference) + */ + template::value, int>::type = 0> + friend bool operator<=(const ScalarType lhs, const_reference rhs) noexcept + { + return (basic_json(lhs) <= rhs); + } + /*! @brief comparison: greater than @@ -6429,6 +6473,28 @@ class basic_json return not (lhs <= rhs); } + /*! + @brief comparison: greater than + @copydoc operator>(const_reference, const_reference) + */ + template::value, int>::type = 0> + friend bool operator>(const_reference lhs, const ScalarType rhs) noexcept + { + return (lhs > basic_json(rhs)); + } + + /*! + @brief comparison: greater than + @copydoc operator>(const_reference, const_reference) + */ + template::value, int>::type = 0> + friend bool operator>(const ScalarType lhs, const_reference rhs) noexcept + { + return (basic_json(lhs) > rhs); + } + /*! @brief comparison: greater than or equal @@ -6451,6 +6517,28 @@ class basic_json return not (lhs < rhs); } + /*! + @brief comparison: greater than or equal + @copydoc operator>=(const_reference, const_reference) + */ + template::value, int>::type = 0> + friend bool operator>=(const_reference lhs, const ScalarType rhs) noexcept + { + return (lhs >= basic_json(rhs)); + } + + /*! + @brief comparison: greater than or equal + @copydoc operator>=(const_reference, const_reference) + */ + template::value, int>::type = 0> + friend bool operator>=(const ScalarType lhs, const_reference rhs) noexcept + { + return (basic_json(lhs) >= rhs); + } + /// @} diff --git a/test/src/unit-regression.cpp b/test/src/unit-regression.cpp index 1e8f5c23..13c88483 100644 --- a/test/src/unit-regression.cpp +++ b/test/src/unit-regression.cpp @@ -973,4 +973,39 @@ TEST_CASE("regression tests") // check if serializations match CHECK(json::to_cbor(j2) == vec2); } + + SECTION("issue #512 - use of overloaded operator '<=' is ambiguous") + { + json j; + j["a"] = 5; + + // json op scalar + CHECK(j["a"] == 5); + CHECK(j["a"] != 4); + + CHECK(j["a"] <= 7); + CHECK(j["a"] < 7); + CHECK(j["a"] >= 3); + CHECK(j["a"] > 3); + + + CHECK(not(j["a"] <= 4)); + CHECK(not(j["a"] < 4)); + CHECK(not(j["a"] >= 6)); + CHECK(not(j["a"] > 6)); + + // scalar op json + CHECK(5 == j["a"]); + CHECK(4 != j["a"]); + + CHECK(7 >= j["a"]); + CHECK(7 > j["a"]); + CHECK(3 <= j["a"]); + CHECK(3 < j["a"]); + + CHECK(not(4 >= j["a"])); + CHECK(not(4 > j["a"])); + CHECK(not(6 <= j["a"])); + CHECK(not(6 < j["a"])); + } } From 97a25de93866fc92a32e44c1856217dd6327f212 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Fri, 7 Apr 2017 18:29:09 +0200 Subject: [PATCH 146/530] :sparkles: proposal for #428 This implementation forwards the iterators to std::map::insert. --- doc/examples/insert__range_object.cpp | 20 +++++++++++ doc/examples/insert__range_object.output | 3 ++ src/json.hpp | 46 ++++++++++++++++++++++++ src/json.hpp.re2c | 46 ++++++++++++++++++++++++ test/src/unit-modifiers.cpp | 36 ++++++++++++++++++- 5 files changed, 150 insertions(+), 1 deletion(-) create mode 100644 doc/examples/insert__range_object.cpp create mode 100644 doc/examples/insert__range_object.output diff --git a/doc/examples/insert__range_object.cpp b/doc/examples/insert__range_object.cpp new file mode 100644 index 00000000..4cd58938 --- /dev/null +++ b/doc/examples/insert__range_object.cpp @@ -0,0 +1,20 @@ +#include + +using json = nlohmann::json; + +int main() +{ + // create two JSON objects + json j1 = {{"one", "eins"}, {"two", "zwei"}}; + json j2 = {{"eleven", "elf"}, {"seventeen", "siebzehn"}}; + + // output objects + std::cout << j1 << '\n'; + std::cout << j2 << '\n'; + + // insert range from j2 to j1 + j1.insert(j2.begin(), j2.end()); + + // output result of insert call + std::cout << j1 << '\n'; +} diff --git a/doc/examples/insert__range_object.output b/doc/examples/insert__range_object.output new file mode 100644 index 00000000..a5985158 --- /dev/null +++ b/doc/examples/insert__range_object.output @@ -0,0 +1,3 @@ +{"one":"eins","two":"zwei"} +{"eleven":"elf","seventeen":"siebzehn"} +{"eleven":"elf","one":"eins","seventeen":"siebzehn","two":"zwei"} diff --git a/src/json.hpp b/src/json.hpp index 98659054..182a634c 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -5978,6 +5978,52 @@ class basic_json return result; } + /*! + @brief inserts elements + + Inserts elements from range `[first, last)`. + + @param[in] first begin of the range of elements to insert + @param[in] last end of the range of elements to insert + + @throw type_error.309 if called on JSON values other than objects; example: + `"cannot use insert() with string"` + @throw invalid_iterator.202 if iterator @a first or @a last does does not + point to an object; example: `"iterators first and last must point to + objects"` + @throw invalid_iterator.210 if @a first and @a last do not belong to the + same JSON value; example: `"iterators do not fit"` + + @complexity Logarithmic: `O(N*log(size() + N))`, where `N` is the number + of elements to insert. + + @liveexample{The example shows how `insert()` is used.,insert__range_object} + + @since version 3.0.0 + */ + void insert(const_iterator first, const_iterator last) + { + // insert only works for objects + if (not is_object()) + { + JSON_THROW(type_error::create(309, "cannot use insert() with " + type_name())); + } + + // check if range iterators belong to the same JSON object + if (first.m_object != last.m_object) + { + JSON_THROW(invalid_iterator::create(210, "iterators do not fit")); + } + + // passed iterators must belong to objects + if (not first.m_object->is_object() or not first.m_object->is_object()) + { + JSON_THROW(invalid_iterator::create(202, "iterators first and last must point to objects")); + } + + m_value.object->insert(first.m_it.object_iterator, last.m_it.object_iterator); + } + /*! @brief exchanges the values diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 5e90f611..02866b22 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -5978,6 +5978,52 @@ class basic_json return result; } + /*! + @brief inserts elements + + Inserts elements from range `[first, last)`. + + @param[in] first begin of the range of elements to insert + @param[in] last end of the range of elements to insert + + @throw type_error.309 if called on JSON values other than objects; example: + `"cannot use insert() with string"` + @throw invalid_iterator.202 if iterator @a first or @a last does does not + point to an object; example: `"iterators first and last must point to + objects"` + @throw invalid_iterator.210 if @a first and @a last do not belong to the + same JSON value; example: `"iterators do not fit"` + + @complexity Logarithmic: `O(N*log(size() + N))`, where `N` is the number + of elements to insert. + + @liveexample{The example shows how `insert()` is used.,insert__range_object} + + @since version 3.0.0 + */ + void insert(const_iterator first, const_iterator last) + { + // insert only works for objects + if (not is_object()) + { + JSON_THROW(type_error::create(309, "cannot use insert() with " + type_name())); + } + + // check if range iterators belong to the same JSON object + if (first.m_object != last.m_object) + { + JSON_THROW(invalid_iterator::create(210, "iterators do not fit")); + } + + // passed iterators must belong to objects + if (not first.m_object->is_object() or not first.m_object->is_object()) + { + JSON_THROW(invalid_iterator::create(202, "iterators first and last must point to objects")); + } + + m_value.object->insert(first.m_it.object_iterator, last.m_it.object_iterator); + } + /*! @brief exchanges the values diff --git a/test/src/unit-modifiers.cpp b/test/src/unit-modifiers.cpp index 0c332111..b860beee 100644 --- a/test/src/unit-modifiers.cpp +++ b/test/src/unit-modifiers.cpp @@ -594,7 +594,7 @@ TEST_CASE("modifiers") } } - SECTION("range") + SECTION("range for array") { json j_other_array = {"first", "second"}; @@ -631,6 +631,40 @@ TEST_CASE("modifiers") } } + SECTION("range for object") + { + json j_object1 = {{"one", "eins"}, {"two", "zwei"}}; + json j_object2 = {{"eleven", "elf"}, {"seventeen", "siebzehn"}}; + + SECTION("proper usage") + { + j_object1.insert(j_object2.begin(), j_object2.end()); + CHECK(j_object1.size() == 4); + } + + SECTION("empty range") + { + j_object1.insert(j_object2.begin(), j_object2.begin()); + CHECK(j_object1.size() == 2); + } + + SECTION("invalid iterators") + { + json j_other_array2 = {"first", "second"}; + + CHECK_THROWS_AS(j_array.insert(j_object2.begin(), j_object2.end()), json::type_error); + CHECK_THROWS_AS(j_object1.insert(j_object1.begin(), j_object2.end()), json::invalid_iterator); + CHECK_THROWS_AS(j_object1.insert(j_array.begin(), j_array.end()), json::invalid_iterator); + + CHECK_THROWS_WITH(j_array.insert(j_object2.begin(), j_object2.end()), + "[json.exception.type_error.309] cannot use insert() with array"); + CHECK_THROWS_WITH(j_object1.insert(j_object1.begin(), j_object2.end()), + "[json.exception.invalid_iterator.210] iterators do not fit"); + CHECK_THROWS_WITH(j_object1.insert(j_array.begin(), j_array.end()), + "[json.exception.invalid_iterator.202] iterators first and last must point to objects"); + } + } + SECTION("initializer list at position") { SECTION("insert before begin()") From d4e49873b74f54edd9446ef2b85f4f82063530c1 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sat, 8 Apr 2017 21:56:42 +0200 Subject: [PATCH 147/530] :construction_worker: experimenting with configurations --- appveyor.yml | 21 +++++++++++++++------ 1 file changed, 15 insertions(+), 6 deletions(-) diff --git a/appveyor.yml b/appveyor.yml index fc410e45..0d85453e 100644 --- a/appveyor.yml +++ b/appveyor.yml @@ -1,12 +1,21 @@ version: '{build}' + +configuration: + - Debug + - Release + os: -- Visual Studio 2015 -- Visual Studio 2017 + - Visual Studio 2015 + - Visual Studio 2017 + init: [] + install: [] + build_script: -- IF "%APPVEYOR_BUILD_WORKER_IMAGE%" == "Visual Studio 2015" ( SET GEN="Visual Studio 14 2015") ELSE (SET GEN="Visual Studio 15 2017") -- cmake . -G%GEN% -- cmake --build . --config Release + - IF "%APPVEYOR_BUILD_WORKER_IMAGE%" == "Visual Studio 2015" ( SET GEN="Visual Studio 14 2015") ELSE (SET GEN="Visual Studio 15 2017") + - cmake . -G%GEN% + - cmake --build . --config Release + test_script: -- ctest -C Release -V + - ctest -C Release -V From 6f99d5b2e978ee6ef448fe39a426ba12c101f8f9 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sat, 8 Apr 2017 23:39:17 +0200 Subject: [PATCH 148/530] :hammer: fixed test case One test case for CBOR and MessagePack assumed little endianess. --- test/src/unit-cbor.cpp | 7 +------ test/src/unit-msgpack.cpp | 7 +------ 2 files changed, 2 insertions(+), 12 deletions(-) diff --git a/test/src/unit-cbor.cpp b/test/src/unit-cbor.cpp index ae6f9a76..debb3732 100644 --- a/test/src/unit-cbor.cpp +++ b/test/src/unit-cbor.cpp @@ -727,14 +727,9 @@ TEST_CASE("CBOR") const auto result = json::to_cbor(j); CHECK(result == expected); - // restore value (reverse array for endianess) - double restored; - std::reverse(expected.begin(), expected.end()); - memcpy(&restored, expected.data(), sizeof(double)); - CHECK(restored == v); - // roundtrip CHECK(json::from_cbor(result) == j); + CHECK(json::from_cbor(result) == v); } } diff --git a/test/src/unit-msgpack.cpp b/test/src/unit-msgpack.cpp index 3f1d1643..a580913d 100644 --- a/test/src/unit-msgpack.cpp +++ b/test/src/unit-msgpack.cpp @@ -676,14 +676,9 @@ TEST_CASE("MessagePack") const auto result = json::to_msgpack(j); CHECK(result == expected); - // restore value (reverse array for endianess) - double restored; - std::reverse(expected.begin(), expected.end()); - memcpy(&restored, expected.data(), sizeof(double)); - CHECK(restored == v); - // roundtrip CHECK(json::from_msgpack(result) == j); + CHECK(json::from_msgpack(result) == v); } } } From 9d9c5216d766f1b4d2e3ac4031064e27951ddcdb Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 9 Apr 2017 10:59:45 +0200 Subject: [PATCH 149/530] :construction_worker: additional flags for MSVC #464 MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit To test a fix for issue #464 (not yet implemented), we first need to have an MSVC build with “/permissive- /std:c++latest /utf-8”. --- appveyor.yml | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/appveyor.yml b/appveyor.yml index 0d85453e..cabfd143 100644 --- a/appveyor.yml +++ b/appveyor.yml @@ -8,13 +8,17 @@ os: - Visual Studio 2015 - Visual Studio 2017 +environment: + - additional_flags: "${CMAKE_CXXFLAGS}" + - additional_flags: "${CMAKE_CXXFLAGS} /permissive- /std:c++latest /utf-8" + init: [] install: [] build_script: - IF "%APPVEYOR_BUILD_WORKER_IMAGE%" == "Visual Studio 2015" ( SET GEN="Visual Studio 14 2015") ELSE (SET GEN="Visual Studio 15 2017") - - cmake . -G%GEN% + - cmake . -G%GEN% -DCMAKE_CXX_FLAGS=%additional_flags% - cmake --build . --config Release test_script: From d3d87674c8fad6af27d018b74c547d3a37249195 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 9 Apr 2017 11:00:39 +0200 Subject: [PATCH 150/530] :construction_worker: fixed AppVeyor syntax --- appveyor.yml | 1 + 1 file changed, 1 insertion(+) diff --git a/appveyor.yml b/appveyor.yml index cabfd143..409f5230 100644 --- a/appveyor.yml +++ b/appveyor.yml @@ -9,6 +9,7 @@ os: - Visual Studio 2017 environment: + matrix: - additional_flags: "${CMAKE_CXXFLAGS}" - additional_flags: "${CMAKE_CXXFLAGS} /permissive- /std:c++latest /utf-8" From 180d249f6c78e7699798ef2a818bedd4f2dd11be Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 9 Apr 2017 11:38:42 +0200 Subject: [PATCH 151/530] :construction_worker: another try --- appveyor.yml | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/appveyor.yml b/appveyor.yml index 409f5230..f1122d59 100644 --- a/appveyor.yml +++ b/appveyor.yml @@ -10,8 +10,8 @@ os: environment: matrix: - - additional_flags: "${CMAKE_CXXFLAGS}" - - additional_flags: "${CMAKE_CXXFLAGS} /permissive- /std:c++latest /utf-8" + - additional_flags: "" + - additional_flags: "/permissive- /std:c++latest /utf-8" init: [] @@ -19,7 +19,7 @@ install: [] build_script: - IF "%APPVEYOR_BUILD_WORKER_IMAGE%" == "Visual Studio 2015" ( SET GEN="Visual Studio 14 2015") ELSE (SET GEN="Visual Studio 15 2017") - - cmake . -G%GEN% -DCMAKE_CXX_FLAGS=%additional_flags% + - cmake . -G%GEN% -DCMAKE_CXX_FLAGS="${CMAKE_CXXFLAGS} %additional_flags%" - cmake --build . --config Release test_script: From a9f409458099a2bcb72f9d2d476b37e26483f25a Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 9 Apr 2017 11:47:31 +0200 Subject: [PATCH 152/530] :construction_worker: another try --- appveyor.yml | 6 +----- 1 file changed, 1 insertion(+), 5 deletions(-) diff --git a/appveyor.yml b/appveyor.yml index f1122d59..1832f28c 100644 --- a/appveyor.yml +++ b/appveyor.yml @@ -1,9 +1,5 @@ version: '{build}' -configuration: - - Debug - - Release - os: - Visual Studio 2015 - Visual Studio 2017 @@ -19,7 +15,7 @@ install: [] build_script: - IF "%APPVEYOR_BUILD_WORKER_IMAGE%" == "Visual Studio 2015" ( SET GEN="Visual Studio 14 2015") ELSE (SET GEN="Visual Studio 15 2017") - - cmake . -G%GEN% -DCMAKE_CXX_FLAGS="${CMAKE_CXXFLAGS} %additional_flags%" + - cmake . -G%GEN% -DCMAKE_CXX_FLAGS="%additional_flags%" - cmake --build . --config Release test_script: From c40c8ac92e9a34ddb417478ac7c52c5716aa401b Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 9 Apr 2017 12:00:30 +0200 Subject: [PATCH 153/530] :construction_worker: excluding MSVC 2015 with "/permissive- /std:c++latest /utf-8" --- appveyor.yml | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/appveyor.yml b/appveyor.yml index 1832f28c..4f423e2f 100644 --- a/appveyor.yml +++ b/appveyor.yml @@ -9,6 +9,11 @@ environment: - additional_flags: "" - additional_flags: "/permissive- /std:c++latest /utf-8" +matrix: + exclude: + - additional_flags: "/permissive- /std:c++latest /utf-8" + os: Visual Studio 2015 + init: [] install: [] From 628be157a27f71b4f045fc687d723f9e1bbdfc51 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 9 Apr 2017 13:52:04 +0200 Subject: [PATCH 154/530] :bug: added fix for #464 --- src/json.hpp | 3 +++ src/json.hpp.re2c | 3 +++ 2 files changed, 6 insertions(+) diff --git a/src/json.hpp b/src/json.hpp index 98659054..bdbd351b 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -3729,6 +3729,9 @@ class basic_json not std::is_same::value #ifndef _MSC_VER // fix for issue #167 operator<< ambiguity under VS2015 and not std::is_same>::value +#endif +#if defined(_MSC_VER) && defined(_HAS_CXX17) // fix for issue #464 + and not std::is_same::value #endif , int >::type = 0 > operator ValueType() const diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 5e90f611..6710233c 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -3729,6 +3729,9 @@ class basic_json not std::is_same::value #ifndef _MSC_VER // fix for issue #167 operator<< ambiguity under VS2015 and not std::is_same>::value +#endif +#if defined(_MSC_VER) && defined(_HAS_CXX17) // fix for issue #464 + and not std::is_same::value #endif , int >::type = 0 > operator ValueType() const From e48114bbd6625e6199baa10cbfe783bb95c32604 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 9 Apr 2017 14:02:04 +0200 Subject: [PATCH 155/530] :hammer: check for value of _HAS_CXX17 --- src/json.hpp | 2 +- src/json.hpp.re2c | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index bdbd351b..3b8925a3 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -3730,7 +3730,7 @@ class basic_json #ifndef _MSC_VER // fix for issue #167 operator<< ambiguity under VS2015 and not std::is_same>::value #endif -#if defined(_MSC_VER) && defined(_HAS_CXX17) // fix for issue #464 +#if defined(_MSC_VER) && defined(_HAS_CXX17) && _HAS_CXX17 == 1 // fix for issue #464 and not std::is_same::value #endif , int >::type = 0 > diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 6710233c..eca19ccf 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -3730,7 +3730,7 @@ class basic_json #ifndef _MSC_VER // fix for issue #167 operator<< ambiguity under VS2015 and not std::is_same>::value #endif -#if defined(_MSC_VER) && defined(_HAS_CXX17) // fix for issue #464 +#if defined(_MSC_VER) && defined(_HAS_CXX17) && _HAS_CXX17 == 1 // fix for issue #464 and not std::is_same::value #endif , int >::type = 0 > From 186a9fd44d529648ad83e318e15729cadcfccd0b Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 9 Apr 2017 19:28:15 +0200 Subject: [PATCH 156/530] :hammer: simplified interface for parser, lexer, and binary_reader These classes are now constructed with an interface adapter. This moves complexity from various places into the interface adapter class, or to some factories which now implement the different flavors of input. Furthermore, input adapters are kept in std::shared_ptr to avoid the need of manual deletion. --- src/json.hpp | 183 +++++------ test/src/unit-class_lexer.cpp | 69 +++-- test/src/unit-class_parser.cpp | 543 +++++++++++++++++---------------- 3 files changed, 409 insertions(+), 386 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index fbc996ad..20c5b2bc 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -7284,7 +7284,7 @@ class basic_json static basic_json parse(const CharT s, const parser_callback_t cb = nullptr) { - return parser(reinterpret_cast(s), cb).parse(true); + return parser(input_adapter::create(s), cb).parse(true); } /*! @@ -7319,7 +7319,7 @@ class basic_json static basic_json parse(std::istream& i, const parser_callback_t cb = nullptr) { - return parser(i, cb).parse(true); + return parser(input_adapter::create(i), cb).parse(true); } /*! @@ -7328,7 +7328,7 @@ class basic_json static basic_json parse(std::istream&& i, const parser_callback_t cb = nullptr) { - return parser(i, cb).parse(true); + return parser(input_adapter::create(i), cb).parse(true); } /*! @@ -7383,27 +7383,7 @@ class basic_json static basic_json parse(IteratorType first, IteratorType last, const parser_callback_t cb = nullptr) { - // assertion to check that the iterator range is indeed contiguous, - // see http://stackoverflow.com/a/35008842/266378 for more discussion - assert(std::accumulate(first, last, std::pair(true, 0), - [&first](std::pair res, decltype(*first) val) - { - res.first &= (val == *(std::next(std::addressof(*first), res.second++))); - return res; - }).first); - - // assertion to check that each element is 1 byte long - static_assert(sizeof(typename std::iterator_traits::value_type) == 1, - "each element in the iterator range must have the size of 1 byte"); - - // if iterator range is empty, create a parser with an empty string - // to generate "unexpected EOF" error message - if (std::distance(first, last) <= 0) - { - return parser("").parse(true); - } - - return parser(first, last, cb).parse(true); + return parser(input_adapter::create(first, last), cb).parse(true); } /*! @@ -7473,7 +7453,7 @@ class basic_json JSON_DEPRECATED friend std::istream& operator<<(basic_json& j, std::istream& i) { - j = parser(i).parse(true); + j = parser(input_adapter::create(i)).parse(true); return i; } @@ -7505,7 +7485,7 @@ class basic_json */ friend std::istream& operator>>(std::istream& i, basic_json& j) { - j = parser(i).parse(true); + j = parser(input_adapter::create(i)).parse(true); return i; } @@ -8566,6 +8546,84 @@ class basic_json virtual int get_character() = 0; virtual std::string read(size_t offset, size_t length) = 0; virtual ~input_adapter() {} + + // native support + + /// input adapter for input stream + static std::shared_ptr create(std::istream& i, const size_t buffer_size = 16384) + { + return std::shared_ptr(new cached_input_stream_adapter(i, buffer_size)); + } + + /// input adapter for input stream + static std::shared_ptr create(std::istream&& i, const size_t buffer_size = 16384) + { + return std::shared_ptr(new cached_input_stream_adapter(i, buffer_size)); + } + + /// input adapter for buffer + static std::shared_ptr create(const char* b, size_t l) + { + return std::shared_ptr(new input_buffer_adapter(b, l)); + } + + // derived support + + /// input adapter for string literal + template::value and + std::is_integral::type>::value and + sizeof(typename std::remove_pointer::type) == 1, int>::type = 0> + static std::shared_ptr create(CharT b) + { + return create(reinterpret_cast(b), + std::strlen(reinterpret_cast(b))); + } + + /// input adapter for iterator range with contiguous storage + template::iterator_category, std::random_access_iterator_tag>::value + , int>::type + = 0> + static std::shared_ptr create(IteratorType first, IteratorType last) + { + // assertion to check that the iterator range is indeed contiguous, + // see http://stackoverflow.com/a/35008842/266378 for more discussion + assert(std::accumulate(first, last, std::pair(true, 0), + [&first](std::pair res, decltype(*first) val) + { + res.first &= (val == *(std::next(std::addressof(*first), res.second++))); + return res; + }).first); + + // assertion to check that each element is 1 byte long + static_assert(sizeof(typename std::iterator_traits::value_type) == 1, + "each element in the iterator range must have the size of 1 byte"); + + return create(reinterpret_cast(&(*first)), + static_cast(std::distance(first, last))); + } + + /// input adapter for array + template + static std::shared_ptr create(T (&array)[N]) + { + // delegate the call to the iterator-range overload + return create(std::begin(array), std::end(array)); + } + + /// input adapter for contiguous container + template::value and + std::is_base_of< + std::random_access_iterator_tag, + typename std::iterator_traits()))>::iterator_category>::value + , int>::type = 0> + static std::shared_ptr create(const ContiguousContainer& c) + { + // delegate the call to the iterator-range overload + return create(std::begin(c), std::end(c)); + } }; /// input adapter for cached stream input @@ -8725,25 +8783,10 @@ class basic_json class binary_reader { public: - explicit binary_reader(std::istream& i) - : ia(new cached_input_stream_adapter(i, 16384)), - is_little_endian(little_endianess()) + explicit binary_reader(std::shared_ptr a) + : ia(a), is_little_endian(little_endianess()) {} - binary_reader(const char* buff, const size_t len) - : ia(new input_buffer_adapter(buff, len)), - is_little_endian(little_endianess()) - {} - - ~binary_reader() - { - delete ia; - } - - // switch off unwanted functions (due to pointer members) - binary_reader(const binary_reader&) = delete; - binary_reader operator=(const binary_reader&) = delete; - /*! @param[in] get_char whether a new character should be retrieved from the input (true, default) or whether the last @@ -9764,7 +9807,7 @@ class basic_json private: /// input adapter - input_adapter* ia = nullptr; + std::shared_ptr ia = nullptr; /// the current character int current = std::char_traits::eof(); @@ -10560,7 +10603,7 @@ class basic_json static basic_json from_cbor(const std::vector& v, const size_t start_index = 0) { - binary_reader br(reinterpret_cast(v.data() + start_index), v.size() - start_index); + binary_reader br(input_adapter::create(v.begin() + static_cast(start_index), v.end())); return br.parse_cbor(); } @@ -10635,7 +10678,7 @@ class basic_json static basic_json from_msgpack(const std::vector& v, const size_t start_index = 0) { - binary_reader br(reinterpret_cast(v.data() + start_index), v.size() - start_index); + binary_reader br(input_adapter::create(v.begin() + static_cast(start_index), v.end())); return br.parse_msgpack(); } @@ -10718,26 +10761,10 @@ class basic_json } } - explicit lexer(std::istream& i) - : ia(new cached_input_stream_adapter(i, 16384)), - decimal_point_char(get_decimal_point()) + explicit lexer(std::shared_ptr a) + : ia(a), decimal_point_char(get_decimal_point()) {} - lexer(const char* buff, const size_t len) - : ia(new input_buffer_adapter(buff, len)), - decimal_point_char(get_decimal_point()) - {} - - ~lexer() - { - delete ia; - } - - // switch off unwanted functions (due to pointer members) - lexer() = delete; - lexer(const lexer&) = delete; - lexer operator=(const lexer&) = delete; - private: ///////////////////// // locales @@ -12091,7 +12118,7 @@ scan_number_done: private: /// input adapter - input_adapter* ia = nullptr; + std::shared_ptr ia = nullptr; /// the current character int current = std::char_traits::eof(); @@ -12129,28 +12156,10 @@ scan_number_done: class parser { public: - /// a parser reading from a string literal - parser(const char* buff, const parser_callback_t cb = nullptr) - : callback(cb), m_lexer(buff, std::strlen(buff)) - {} - - /*! - @brief a parser reading from an input stream - @throw parse_error.111 if input stream is in a bad state - */ - parser(std::istream& is, const parser_callback_t cb = nullptr) - : callback(cb), m_lexer(is) - {} - - /// a parser reading from an iterator range with contiguous storage - template::iterator_category, std::random_access_iterator_tag>::value - , int>::type - = 0> - parser(IteratorType first, IteratorType last, const parser_callback_t cb = nullptr) - : callback(cb), - m_lexer(reinterpret_cast(&(*first)), - static_cast(std::distance(first, last))) + /// a parser reading from an input adapter + explicit parser(std::shared_ptr ia, + const parser_callback_t cb = nullptr) + : callback(cb), m_lexer(ia) {} /*! diff --git a/test/src/unit-class_lexer.cpp b/test/src/unit-class_lexer.cpp index 2acea176..071337b5 100644 --- a/test/src/unit-class_lexer.cpp +++ b/test/src/unit-class_lexer.cpp @@ -32,56 +32,63 @@ SOFTWARE. #include "json.hpp" using nlohmann::json; +// shortcut to scan a string literal +json::lexer::token_type scan_string(const char* s); +json::lexer::token_type scan_string(const char* s) +{ + return json::lexer(json::input_adapter::create(s)).scan(); +} + TEST_CASE("lexer class") { SECTION("scan") { SECTION("structural characters") { - CHECK((json::lexer("[", 1).scan() == json::lexer::token_type::begin_array)); - CHECK((json::lexer("]", 1).scan() == json::lexer::token_type::end_array)); - CHECK((json::lexer("{", 1).scan() == json::lexer::token_type::begin_object)); - CHECK((json::lexer("}", 1).scan() == json::lexer::token_type::end_object)); - CHECK((json::lexer(",", 1).scan() == json::lexer::token_type::value_separator)); - CHECK((json::lexer(":", 1).scan() == json::lexer::token_type::name_separator)); + CHECK((scan_string("[") == json::lexer::token_type::begin_array)); + CHECK((scan_string("]") == json::lexer::token_type::end_array)); + CHECK((scan_string("{") == json::lexer::token_type::begin_object)); + CHECK((scan_string("}") == json::lexer::token_type::end_object)); + CHECK((scan_string(",") == json::lexer::token_type::value_separator)); + CHECK((scan_string(":") == json::lexer::token_type::name_separator)); } SECTION("literal names") { - CHECK((json::lexer("null", 4).scan() == json::lexer::token_type::literal_null)); - CHECK((json::lexer("true", 4).scan() == json::lexer::token_type::literal_true)); - CHECK((json::lexer("false", 5).scan() == json::lexer::token_type::literal_false)); + CHECK((scan_string("null") == json::lexer::token_type::literal_null)); + CHECK((scan_string("true") == json::lexer::token_type::literal_true)); + CHECK((scan_string("false") == json::lexer::token_type::literal_false)); } SECTION("numbers") { - CHECK((json::lexer("0", 1).scan() == json::lexer::token_type::value_unsigned)); - CHECK((json::lexer("1", 1).scan() == json::lexer::token_type::value_unsigned)); - CHECK((json::lexer("2", 1).scan() == json::lexer::token_type::value_unsigned)); - CHECK((json::lexer("3", 1).scan() == json::lexer::token_type::value_unsigned)); - CHECK((json::lexer("4", 1).scan() == json::lexer::token_type::value_unsigned)); - CHECK((json::lexer("5", 1).scan() == json::lexer::token_type::value_unsigned)); - CHECK((json::lexer("6", 1).scan() == json::lexer::token_type::value_unsigned)); - CHECK((json::lexer("7", 1).scan() == json::lexer::token_type::value_unsigned)); - CHECK((json::lexer("8", 1).scan() == json::lexer::token_type::value_unsigned)); - CHECK((json::lexer("9", 1).scan() == json::lexer::token_type::value_unsigned)); + CHECK((scan_string("0") == json::lexer::token_type::value_unsigned)); + CHECK((scan_string("1") == json::lexer::token_type::value_unsigned)); + CHECK((scan_string("2") == json::lexer::token_type::value_unsigned)); + CHECK((scan_string("3") == json::lexer::token_type::value_unsigned)); + CHECK((scan_string("4") == json::lexer::token_type::value_unsigned)); + CHECK((scan_string("5") == json::lexer::token_type::value_unsigned)); + CHECK((scan_string("6") == json::lexer::token_type::value_unsigned)); + CHECK((scan_string("7") == json::lexer::token_type::value_unsigned)); + CHECK((scan_string("8") == json::lexer::token_type::value_unsigned)); + CHECK((scan_string("9") == json::lexer::token_type::value_unsigned)); - CHECK((json::lexer("-0", 2).scan() == json::lexer::token_type::value_integer)); - CHECK((json::lexer("-1", 2).scan() == json::lexer::token_type::value_integer)); + CHECK((scan_string("-0") == json::lexer::token_type::value_integer)); + CHECK((scan_string("-1") == json::lexer::token_type::value_integer)); - CHECK((json::lexer("1.1", 3).scan() == json::lexer::token_type::value_float)); - CHECK((json::lexer("-1.1", 4).scan() == json::lexer::token_type::value_float)); - CHECK((json::lexer("1E10", 4).scan() == json::lexer::token_type::value_float)); + CHECK((scan_string("1.1") == json::lexer::token_type::value_float)); + CHECK((scan_string("-1.1") == json::lexer::token_type::value_float)); + CHECK((scan_string("1E10") == json::lexer::token_type::value_float)); } SECTION("whitespace") { // result is end_of_input, because not token is following - CHECK((json::lexer(" ", 1).scan() == json::lexer::token_type::end_of_input)); - CHECK((json::lexer("\t", 1).scan() == json::lexer::token_type::end_of_input)); - CHECK((json::lexer("\n", 1).scan() == json::lexer::token_type::end_of_input)); - CHECK((json::lexer("\r", 1).scan() == json::lexer::token_type::end_of_input)); - CHECK((json::lexer(" \t\n\r\n\t ", 7).scan() == json::lexer::token_type::end_of_input)); + CHECK((scan_string(" ") == json::lexer::token_type::end_of_input)); + CHECK((scan_string("\t") == json::lexer::token_type::end_of_input)); + CHECK((scan_string("\n") == json::lexer::token_type::end_of_input)); + CHECK((scan_string("\r") == json::lexer::token_type::end_of_input)); + CHECK((scan_string(" \t\n\r\n\t ") == json::lexer::token_type::end_of_input)); } } @@ -112,7 +119,7 @@ TEST_CASE("lexer class") // create string from the ASCII code const auto s = std::string(1, static_cast(c)); // store scan() result - const auto res = json::lexer(s.c_str(), 1).scan(); + const auto res = scan_string(s.c_str()); switch (c) { @@ -164,7 +171,7 @@ TEST_CASE("lexer class") std::string s("\""); s += std::string(2048, 'x'); s += "\""; - CHECK((json::lexer(s.c_str(), 2050).scan() == json::lexer::token_type::value_string)); + CHECK((scan_string(s.c_str()) == json::lexer::token_type::value_string)); } /* NOTE: to_unicode function has been removed diff --git a/test/src/unit-class_parser.cpp b/test/src/unit-class_parser.cpp index 864b7be1..e0fffac4 100644 --- a/test/src/unit-class_parser.cpp +++ b/test/src/unit-class_parser.cpp @@ -34,36 +34,43 @@ using nlohmann::json; #include +// shortcut to parse a string literal +json::parser parse_string(const char* s); +json::parser parse_string(const char* s) +{ + return json::parser(json::input_adapter::create(s)); +} + TEST_CASE("parser class") { SECTION("parse") { SECTION("null") { - CHECK(json::parser("null").parse() == json(nullptr)); + CHECK(parse_string("null").parse() == json(nullptr)); } SECTION("true") { - CHECK(json::parser("true").parse() == json(true)); + CHECK(parse_string("true").parse() == json(true)); } SECTION("false") { - CHECK(json::parser("false").parse() == json(false)); + CHECK(parse_string("false").parse() == json(false)); } SECTION("array") { SECTION("empty array") { - CHECK(json::parser("[]").parse() == json(json::value_t::array)); - CHECK(json::parser("[ ]").parse() == json(json::value_t::array)); + CHECK(parse_string("[]").parse() == json(json::value_t::array)); + CHECK(parse_string("[ ]").parse() == json(json::value_t::array)); } SECTION("nonempty array") { - CHECK(json::parser("[true, false, null]").parse() == json({true, false, nullptr})); + CHECK(parse_string("[true, false, null]").parse() == json({true, false, nullptr})); } } @@ -71,113 +78,113 @@ TEST_CASE("parser class") { SECTION("empty object") { - CHECK(json::parser("{}").parse() == json(json::value_t::object)); - CHECK(json::parser("{ }").parse() == json(json::value_t::object)); + CHECK(parse_string("{}").parse() == json(json::value_t::object)); + CHECK(parse_string("{ }").parse() == json(json::value_t::object)); } SECTION("nonempty object") { - CHECK(json::parser("{\"\": true, \"one\": 1, \"two\": null}").parse() == json({{"", true}, {"one", 1}, {"two", nullptr}})); + CHECK(parse_string("{\"\": true, \"one\": 1, \"two\": null}").parse() == json({{"", true}, {"one", 1}, {"two", nullptr}})); } } SECTION("string") { // empty string - CHECK(json::parser("\"\"").parse() == json(json::value_t::string)); + CHECK(parse_string("\"\"").parse() == json(json::value_t::string)); SECTION("errors") { // error: tab in string - CHECK_THROWS_AS(json::parser("\"\t\"").parse(), json::parse_error); - CHECK_THROWS_WITH(json::parser("\"\t\"").parse(), + CHECK_THROWS_AS(parse_string("\"\t\"").parse(), json::parse_error); + CHECK_THROWS_WITH(parse_string("\"\t\"").parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: control characters (U+0000 through U+001f) must be escaped; last read '\"'"); // error: newline in string - CHECK_THROWS_AS(json::parser("\"\n\"").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("\"\r\"").parse(), json::parse_error); - CHECK_THROWS_WITH(json::parser("\"\n\"").parse(), + CHECK_THROWS_AS(parse_string("\"\n\"").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("\"\r\"").parse(), json::parse_error); + CHECK_THROWS_WITH(parse_string("\"\n\"").parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: control characters (U+0000 through U+001f) must be escaped; last read '\"'"); - CHECK_THROWS_WITH(json::parser("\"\r\"").parse(), + CHECK_THROWS_WITH(parse_string("\"\r\"").parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: control characters (U+0000 through U+001f) must be escaped; last read '\"'"); // error: backspace in string - CHECK_THROWS_AS(json::parser("\"\b\"").parse(), json::parse_error); - CHECK_THROWS_WITH(json::parser("\"\b\"").parse(), + CHECK_THROWS_AS(parse_string("\"\b\"").parse(), json::parse_error); + CHECK_THROWS_WITH(parse_string("\"\b\"").parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: control characters (U+0000 through U+001f) must be escaped; last read '\"'"); // improve code coverage - CHECK_THROWS_AS(json::parser("\uFF01").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("[-4:1,]").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("\uFF01").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("[-4:1,]").parse(), json::parse_error); // unescaped control characters - CHECK_THROWS_AS(json::parser("\"\x00\"").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("\"\x01\"").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("\"\x02\"").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("\"\x03\"").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("\"\x04\"").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("\"\x05\"").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("\"\x06\"").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("\"\x07\"").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("\"\x08\"").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("\"\x09\"").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("\"\x0a\"").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("\"\x0b\"").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("\"\x0c\"").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("\"\x0d\"").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("\"\x0e\"").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("\"\x0f\"").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("\"\x10\"").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("\"\x11\"").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("\"\x12\"").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("\"\x13\"").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("\"\x14\"").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("\"\x15\"").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("\"\x16\"").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("\"\x17\"").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("\"\x18\"").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("\"\x19\"").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("\"\x1a\"").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("\"\x1b\"").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("\"\x1c\"").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("\"\x1d\"").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("\"\x1e\"").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("\"\x1f\"").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("\"\x00\"").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("\"\x01\"").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("\"\x02\"").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("\"\x03\"").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("\"\x04\"").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("\"\x05\"").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("\"\x06\"").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("\"\x07\"").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("\"\x08\"").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("\"\x09\"").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("\"\x0a\"").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("\"\x0b\"").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("\"\x0c\"").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("\"\x0d\"").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("\"\x0e\"").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("\"\x0f\"").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("\"\x10\"").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("\"\x11\"").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("\"\x12\"").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("\"\x13\"").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("\"\x14\"").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("\"\x15\"").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("\"\x16\"").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("\"\x17\"").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("\"\x18\"").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("\"\x19\"").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("\"\x1a\"").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("\"\x1b\"").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("\"\x1c\"").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("\"\x1d\"").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("\"\x1e\"").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("\"\x1f\"").parse(), json::parse_error); } SECTION("escaped") { // quotation mark "\"" auto r1 = R"("\"")"_json; - CHECK(json::parser("\"\\\"\"").parse() == r1); + CHECK(parse_string("\"\\\"\"").parse() == r1); // reverse solidus "\\" auto r2 = R"("\\")"_json; - CHECK(json::parser("\"\\\\\"").parse() == r2); + CHECK(parse_string("\"\\\\\"").parse() == r2); // solidus - CHECK(json::parser("\"\\/\"").parse() == R"("/")"_json); + CHECK(parse_string("\"\\/\"").parse() == R"("/")"_json); // backspace - CHECK(json::parser("\"\\b\"").parse() == json("\b")); + CHECK(parse_string("\"\\b\"").parse() == json("\b")); // formfeed - CHECK(json::parser("\"\\f\"").parse() == json("\f")); + CHECK(parse_string("\"\\f\"").parse() == json("\f")); // newline - CHECK(json::parser("\"\\n\"").parse() == json("\n")); + CHECK(parse_string("\"\\n\"").parse() == json("\n")); // carriage return - CHECK(json::parser("\"\\r\"").parse() == json("\r")); + CHECK(parse_string("\"\\r\"").parse() == json("\r")); // horizontal tab - CHECK(json::parser("\"\\t\"").parse() == json("\t")); + CHECK(parse_string("\"\\t\"").parse() == json("\t")); - CHECK(json::parser("\"\\u0001\"").parse().get() == "\x01"); - CHECK(json::parser("\"\\u000a\"").parse().get() == "\n"); - CHECK(json::parser("\"\\u00b0\"").parse().get() == "°"); - CHECK(json::parser("\"\\u0c00\"").parse().get() == "ఀ"); - CHECK(json::parser("\"\\ud000\"").parse().get() == "퀀"); - CHECK(json::parser("\"\\u000E\"").parse().get() == "\x0E"); - CHECK(json::parser("\"\\u00F0\"").parse().get() == "ð"); - CHECK(json::parser("\"\\u0100\"").parse().get() == "Ā"); - CHECK(json::parser("\"\\u2000\"").parse().get() == " "); - CHECK(json::parser("\"\\uFFFF\"").parse().get() == "￿"); - CHECK(json::parser("\"\\u20AC\"").parse().get() == "€"); - CHECK(json::parser("\"€\"").parse().get() == "€"); - CHECK(json::parser("\"🎈\"").parse().get() == "🎈"); + CHECK(parse_string("\"\\u0001\"").parse().get() == "\x01"); + CHECK(parse_string("\"\\u000a\"").parse().get() == "\n"); + CHECK(parse_string("\"\\u00b0\"").parse().get() == "°"); + CHECK(parse_string("\"\\u0c00\"").parse().get() == "ఀ"); + CHECK(parse_string("\"\\ud000\"").parse().get() == "퀀"); + CHECK(parse_string("\"\\u000E\"").parse().get() == "\x0E"); + CHECK(parse_string("\"\\u00F0\"").parse().get() == "ð"); + CHECK(parse_string("\"\\u0100\"").parse().get() == "Ā"); + CHECK(parse_string("\"\\u2000\"").parse().get() == " "); + CHECK(parse_string("\"\\uFFFF\"").parse().get() == "￿"); + CHECK(parse_string("\"\\u20AC\"").parse().get() == "€"); + CHECK(parse_string("\"€\"").parse().get() == "€"); + CHECK(parse_string("\"🎈\"").parse().get() == "🎈"); - CHECK(json::parse("\"\\ud80c\\udc60\"").get() == u8"\U00013060"); - CHECK(json::parse("\"\\ud83c\\udf1e\"").get() == "🌞"); + CHECK(parse_string("\"\\ud80c\\udc60\"").parse().get() == u8"\U00013060"); + CHECK(parse_string("\"\\ud83c\\udf1e\"").parse().get() == "🌞"); } } @@ -187,40 +194,40 @@ TEST_CASE("parser class") { SECTION("without exponent") { - CHECK(json::parser("-128").parse() == json(-128)); - CHECK(json::parser("-0").parse() == json(-0)); - CHECK(json::parser("0").parse() == json(0)); - CHECK(json::parser("128").parse() == json(128)); + CHECK(parse_string("-128").parse() == json(-128)); + CHECK(parse_string("-0").parse() == json(-0)); + CHECK(parse_string("0").parse() == json(0)); + CHECK(parse_string("128").parse() == json(128)); } SECTION("with exponent") { - CHECK(json::parser("0e1").parse() == json(0e1)); - CHECK(json::parser("0E1").parse() == json(0e1)); + CHECK(parse_string("0e1").parse() == json(0e1)); + CHECK(parse_string("0E1").parse() == json(0e1)); - CHECK(json::parser("10000E-4").parse() == json(10000e-4)); - CHECK(json::parser("10000E-3").parse() == json(10000e-3)); - CHECK(json::parser("10000E-2").parse() == json(10000e-2)); - CHECK(json::parser("10000E-1").parse() == json(10000e-1)); - CHECK(json::parser("10000E0").parse() == json(10000e0)); - CHECK(json::parser("10000E1").parse() == json(10000e1)); - CHECK(json::parser("10000E2").parse() == json(10000e2)); - CHECK(json::parser("10000E3").parse() == json(10000e3)); - CHECK(json::parser("10000E4").parse() == json(10000e4)); + CHECK(parse_string("10000E-4").parse() == json(10000e-4)); + CHECK(parse_string("10000E-3").parse() == json(10000e-3)); + CHECK(parse_string("10000E-2").parse() == json(10000e-2)); + CHECK(parse_string("10000E-1").parse() == json(10000e-1)); + CHECK(parse_string("10000E0").parse() == json(10000e0)); + CHECK(parse_string("10000E1").parse() == json(10000e1)); + CHECK(parse_string("10000E2").parse() == json(10000e2)); + CHECK(parse_string("10000E3").parse() == json(10000e3)); + CHECK(parse_string("10000E4").parse() == json(10000e4)); - CHECK(json::parser("10000e-4").parse() == json(10000e-4)); - CHECK(json::parser("10000e-3").parse() == json(10000e-3)); - CHECK(json::parser("10000e-2").parse() == json(10000e-2)); - CHECK(json::parser("10000e-1").parse() == json(10000e-1)); - CHECK(json::parser("10000e0").parse() == json(10000e0)); - CHECK(json::parser("10000e1").parse() == json(10000e1)); - CHECK(json::parser("10000e2").parse() == json(10000e2)); - CHECK(json::parser("10000e3").parse() == json(10000e3)); - CHECK(json::parser("10000e4").parse() == json(10000e4)); + CHECK(parse_string("10000e-4").parse() == json(10000e-4)); + CHECK(parse_string("10000e-3").parse() == json(10000e-3)); + CHECK(parse_string("10000e-2").parse() == json(10000e-2)); + CHECK(parse_string("10000e-1").parse() == json(10000e-1)); + CHECK(parse_string("10000e0").parse() == json(10000e0)); + CHECK(parse_string("10000e1").parse() == json(10000e1)); + CHECK(parse_string("10000e2").parse() == json(10000e2)); + CHECK(parse_string("10000e3").parse() == json(10000e3)); + CHECK(parse_string("10000e4").parse() == json(10000e4)); - CHECK(json::parser("-0e1").parse() == json(-0e1)); - CHECK(json::parser("-0E1").parse() == json(-0e1)); - CHECK(json::parser("-0E123").parse() == json(-0e123)); + CHECK(parse_string("-0e1").parse() == json(-0e1)); + CHECK(parse_string("-0E1").parse() == json(-0e1)); + CHECK(parse_string("-0E123").parse() == json(-0e123)); } SECTION("edge cases") @@ -232,9 +239,9 @@ TEST_CASE("parser class") // agree exactly on their numeric values. // -(2**53)+1 - CHECK(json::parser("-9007199254740991").parse().get() == -9007199254740991); + CHECK(parse_string("-9007199254740991").parse().get() == -9007199254740991); // (2**53)-1 - CHECK(json::parser("9007199254740991").parse().get() == 9007199254740991); + CHECK(parse_string("9007199254740991").parse().get() == 9007199254740991); } SECTION("over the edge cases") // issue #178 - Integer conversion to unsigned (incorrect handling of 64 bit integers) @@ -247,11 +254,11 @@ TEST_CASE("parser class") // i.e. -(2**63) -> (2**64)-1. // -(2**63) ** Note: compilers see negative literals as negated positive numbers (hence the -1)) - CHECK(json::parser("-9223372036854775808").parse().get() == -9223372036854775807 - 1); + CHECK(parse_string("-9223372036854775808").parse().get() == -9223372036854775807 - 1); // (2**63)-1 - CHECK(json::parser("9223372036854775807").parse().get() == 9223372036854775807); + CHECK(parse_string("9223372036854775807").parse().get() == 9223372036854775807); // (2**64)-1 - CHECK(json::parser("18446744073709551615").parse().get() == 18446744073709551615u); + CHECK(parse_string("18446744073709551615").parse().get() == 18446744073709551615u); } } @@ -259,85 +266,85 @@ TEST_CASE("parser class") { SECTION("without exponent") { - CHECK(json::parser("-128.5").parse() == json(-128.5)); - CHECK(json::parser("0.999").parse() == json(0.999)); - CHECK(json::parser("128.5").parse() == json(128.5)); - CHECK(json::parser("-0.0").parse() == json(-0.0)); + CHECK(parse_string("-128.5").parse() == json(-128.5)); + CHECK(parse_string("0.999").parse() == json(0.999)); + CHECK(parse_string("128.5").parse() == json(128.5)); + CHECK(parse_string("-0.0").parse() == json(-0.0)); } SECTION("with exponent") { - CHECK(json::parser("-128.5E3").parse() == json(-128.5E3)); - CHECK(json::parser("-128.5E-3").parse() == json(-128.5E-3)); - CHECK(json::parser("-0.0e1").parse() == json(-0.0e1)); - CHECK(json::parser("-0.0E1").parse() == json(-0.0e1)); + CHECK(parse_string("-128.5E3").parse() == json(-128.5E3)); + CHECK(parse_string("-128.5E-3").parse() == json(-128.5E-3)); + CHECK(parse_string("-0.0e1").parse() == json(-0.0e1)); + CHECK(parse_string("-0.0E1").parse() == json(-0.0e1)); } } SECTION("overflow") { // overflows during parsing yield an exception - CHECK_THROWS_AS(json::parser("1.18973e+4932").parse() == json(), json::out_of_range); - CHECK_THROWS_WITH(json::parser("1.18973e+4932").parse() == json(), + CHECK_THROWS_AS(parse_string("1.18973e+4932").parse() == json(), json::out_of_range); + CHECK_THROWS_WITH(parse_string("1.18973e+4932").parse() == json(), "[json.exception.out_of_range.406] number overflow parsing '1.18973e+4932'"); } SECTION("invalid numbers") { - CHECK_THROWS_AS(json::parser("01").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("--1").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("1.").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("1E").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("1E-").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("1.E1").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("-1E").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("-0E#").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("-0E-#").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("-0#").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("-0.0:").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("-0.0Z").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("-0E123:").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("-0e0-:").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("-0e-:").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("-0f").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("01").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("--1").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("1.").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("1E").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("1E-").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("1.E1").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("-1E").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("-0E#").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("-0E-#").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("-0#").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("-0.0:").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("-0.0Z").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("-0E123:").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("-0e0-:").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("-0e-:").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("-0f").parse(), json::parse_error); // numbers must not begin with "+" - CHECK_THROWS_AS(json::parser("+1").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("+0").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("+1").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("+0").parse(), json::parse_error); - CHECK_THROWS_WITH(json::parser("01").parse(), + CHECK_THROWS_WITH(parse_string("01").parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected number literal; expected end of input"); - CHECK_THROWS_WITH(json::parser("-01").parse(), + CHECK_THROWS_WITH(parse_string("-01").parse(), "[json.exception.parse_error.101] parse error at 3: syntax error - unexpected number literal; expected end of input"); - CHECK_THROWS_WITH(json::parser("--1").parse(), + CHECK_THROWS_WITH(parse_string("--1").parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - invalid number; expected digit after '-'; last read '--'"); - CHECK_THROWS_WITH(json::parser("1.").parse(), + CHECK_THROWS_WITH(parse_string("1.").parse(), "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected digit after '.'; last read '1.'"); - CHECK_THROWS_WITH(json::parser("1E").parse(), + CHECK_THROWS_WITH(parse_string("1E").parse(), "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected '+', '-', or digit after exponent; last read '1E'"); - CHECK_THROWS_WITH(json::parser("1E-").parse(), + CHECK_THROWS_WITH(parse_string("1E-").parse(), "[json.exception.parse_error.101] parse error at 4: syntax error - invalid number; expected digit after exponent sign; last read '1E-'"); - CHECK_THROWS_WITH(json::parser("1.E1").parse(), + CHECK_THROWS_WITH(parse_string("1.E1").parse(), "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected digit after '.'; last read '1.E'"); - CHECK_THROWS_WITH(json::parser("-1E").parse(), + CHECK_THROWS_WITH(parse_string("-1E").parse(), "[json.exception.parse_error.101] parse error at 4: syntax error - invalid number; expected '+', '-', or digit after exponent; last read '-1E'"); - CHECK_THROWS_WITH(json::parser("-0E#").parse(), + CHECK_THROWS_WITH(parse_string("-0E#").parse(), "[json.exception.parse_error.101] parse error at 4: syntax error - invalid number; expected '+', '-', or digit after exponent; last read '-0E#'"); - CHECK_THROWS_WITH(json::parser("-0E-#").parse(), + CHECK_THROWS_WITH(parse_string("-0E-#").parse(), "[json.exception.parse_error.101] parse error at 5: syntax error - invalid number; expected digit after exponent sign; last read '-0E-#'"); - CHECK_THROWS_WITH(json::parser("-0#").parse(), + CHECK_THROWS_WITH(parse_string("-0#").parse(), "[json.exception.parse_error.101] parse error at 3: syntax error - invalid literal; last read: '-0#'; expected end of input"); - CHECK_THROWS_WITH(json::parser("-0.0:").parse(), + CHECK_THROWS_WITH(parse_string("-0.0:").parse(), "[json.exception.parse_error.101] parse error at 5: syntax error - unexpected ':'; expected end of input"); - CHECK_THROWS_WITH(json::parser("-0.0Z").parse(), + CHECK_THROWS_WITH(parse_string("-0.0Z").parse(), "[json.exception.parse_error.101] parse error at 5: syntax error - invalid literal; last read: '-0.0Z'; expected end of input"); - CHECK_THROWS_WITH(json::parser("-0E123:").parse(), + CHECK_THROWS_WITH(parse_string("-0E123:").parse(), "[json.exception.parse_error.101] parse error at 7: syntax error - unexpected ':'; expected end of input"); - CHECK_THROWS_WITH(json::parser("-0e0-:").parse(), + CHECK_THROWS_WITH(parse_string("-0e0-:").parse(), "[json.exception.parse_error.101] parse error at 6: syntax error - invalid number; expected digit after '-'; last read: '-:'; expected end of input"); - CHECK_THROWS_WITH(json::parser("-0e-:").parse(), + CHECK_THROWS_WITH(parse_string("-0e-:").parse(), "[json.exception.parse_error.101] parse error at 5: syntax error - invalid number; expected digit after exponent sign; last read '-0e-:'"); - CHECK_THROWS_WITH(json::parser("-0f").parse(), + CHECK_THROWS_WITH(parse_string("-0f").parse(), "[json.exception.parse_error.101] parse error at 4: syntax error - invalid literal; expected 'false'; last read: '-0f'; expected end of input"); } } @@ -346,152 +353,152 @@ TEST_CASE("parser class") SECTION("parse errors") { // unexpected end of number - CHECK_THROWS_AS(json::parser("0.").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("-").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("--").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("-0.").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("-.").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("-:").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("0.:").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("e.").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("1e.").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("1e/").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("1e:").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("1E.").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("1E/").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("1E:").parse(), json::parse_error); - CHECK_THROWS_WITH(json::parser("0.").parse(), + CHECK_THROWS_AS(parse_string("0.").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("-").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("--").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("-0.").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("-.").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("-:").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("0.:").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("e.").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("1e.").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("1e/").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("1e:").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("1E.").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("1E/").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("1E:").parse(), json::parse_error); + CHECK_THROWS_WITH(parse_string("0.").parse(), "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected digit after '.'; last read '0.'"); - CHECK_THROWS_WITH(json::parser("-").parse(), + CHECK_THROWS_WITH(parse_string("-").parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - invalid number; expected digit after '-'; last read '-'"); - CHECK_THROWS_WITH(json::parser("--").parse(), + CHECK_THROWS_WITH(parse_string("--").parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - invalid number; expected digit after '-'; last read '--'"); - CHECK_THROWS_WITH(json::parser("-0.").parse(), + CHECK_THROWS_WITH(parse_string("-0.").parse(), "[json.exception.parse_error.101] parse error at 4: syntax error - invalid number; expected digit after '.'; last read '-0.'"); - CHECK_THROWS_WITH(json::parser("-.").parse(), + CHECK_THROWS_WITH(parse_string("-.").parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - invalid number; expected digit after '-'; last read '-.'"); - CHECK_THROWS_WITH(json::parser("-:").parse(), + CHECK_THROWS_WITH(parse_string("-:").parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - invalid number; expected digit after '-'; last read '-:'"); - CHECK_THROWS_WITH(json::parser("0.:").parse(), + CHECK_THROWS_WITH(parse_string("0.:").parse(), "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected digit after '.'; last read '0.:'"); - CHECK_THROWS_WITH(json::parser("e.").parse(), + CHECK_THROWS_WITH(parse_string("e.").parse(), "[json.exception.parse_error.101] parse error at 1: syntax error - invalid literal; last read 'e'"); - CHECK_THROWS_WITH(json::parser("1e.").parse(), + CHECK_THROWS_WITH(parse_string("1e.").parse(), "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected '+', '-', or digit after exponent; last read '1e.'"); - CHECK_THROWS_WITH(json::parser("1e/").parse(), + CHECK_THROWS_WITH(parse_string("1e/").parse(), "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected '+', '-', or digit after exponent; last read '1e/'"); - CHECK_THROWS_WITH(json::parser("1e:").parse(), + CHECK_THROWS_WITH(parse_string("1e:").parse(), "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected '+', '-', or digit after exponent; last read '1e:'"); - CHECK_THROWS_WITH(json::parser("1E.").parse(), + CHECK_THROWS_WITH(parse_string("1E.").parse(), "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected '+', '-', or digit after exponent; last read '1E.'"); - CHECK_THROWS_WITH(json::parser("1E/").parse(), + CHECK_THROWS_WITH(parse_string("1E/").parse(), "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected '+', '-', or digit after exponent; last read '1E/'"); - CHECK_THROWS_WITH(json::parser("1E:").parse(), + CHECK_THROWS_WITH(parse_string("1E:").parse(), "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected '+', '-', or digit after exponent; last read '1E:'"); // unexpected end of null - CHECK_THROWS_AS(json::parser("n").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("nu").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("nul").parse(), json::parse_error); - CHECK_THROWS_WITH(json::parser("n").parse(), + CHECK_THROWS_AS(parse_string("n").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("nu").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("nul").parse(), json::parse_error); + CHECK_THROWS_WITH(parse_string("n").parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - invalid literal; expected 'null'; last read 'n'"); - CHECK_THROWS_WITH(json::parser("nu").parse(), + CHECK_THROWS_WITH(parse_string("nu").parse(), "[json.exception.parse_error.101] parse error at 3: syntax error - invalid literal; expected 'null'; last read 'nu'"); - CHECK_THROWS_WITH(json::parser("nul").parse(), + CHECK_THROWS_WITH(parse_string("nul").parse(), "[json.exception.parse_error.101] parse error at 4: syntax error - invalid literal; expected 'null'; last read 'nul'"); // unexpected end of true - CHECK_THROWS_AS(json::parser("t").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("tr").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("tru").parse(), json::parse_error); - CHECK_THROWS_WITH(json::parser("t").parse(), + CHECK_THROWS_AS(parse_string("t").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("tr").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("tru").parse(), json::parse_error); + CHECK_THROWS_WITH(parse_string("t").parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - invalid literal; expected 'true'; last read 't'"); - CHECK_THROWS_WITH(json::parser("tr").parse(), + CHECK_THROWS_WITH(parse_string("tr").parse(), "[json.exception.parse_error.101] parse error at 3: syntax error - invalid literal; expected 'true'; last read 'tr'"); - CHECK_THROWS_WITH(json::parser("tru").parse(), + CHECK_THROWS_WITH(parse_string("tru").parse(), "[json.exception.parse_error.101] parse error at 4: syntax error - invalid literal; expected 'true'; last read 'tru'"); // unexpected end of false - CHECK_THROWS_AS(json::parser("f").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("fa").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("fal").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("fals").parse(), json::parse_error); - CHECK_THROWS_WITH(json::parser("f").parse(), + CHECK_THROWS_AS(parse_string("f").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("fa").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("fal").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("fals").parse(), json::parse_error); + CHECK_THROWS_WITH(parse_string("f").parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - invalid literal; expected 'false'; last read 'f'"); - CHECK_THROWS_WITH(json::parser("fa").parse(), + CHECK_THROWS_WITH(parse_string("fa").parse(), "[json.exception.parse_error.101] parse error at 3: syntax error - invalid literal; expected 'false'; last read 'fa'"); - CHECK_THROWS_WITH(json::parser("fal").parse(), + CHECK_THROWS_WITH(parse_string("fal").parse(), "[json.exception.parse_error.101] parse error at 4: syntax error - invalid literal; expected 'false'; last read 'fal'"); - CHECK_THROWS_WITH(json::parser("fals").parse(), + CHECK_THROWS_WITH(parse_string("fals").parse(), "[json.exception.parse_error.101] parse error at 5: syntax error - invalid literal; expected 'false'; last read 'fals'"); // missing/unexpected end of array - CHECK_THROWS_AS(json::parser("[").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("[1").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("[1,").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("[1,]").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("]").parse(), json::parse_error); - CHECK_THROWS_WITH(json::parser("[").parse(), + CHECK_THROWS_AS(parse_string("[").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("[1").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("[1,").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("[1,]").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("]").parse(), json::parse_error); + CHECK_THROWS_WITH(parse_string("[").parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected end of input"); - CHECK_THROWS_WITH(json::parser("[1").parse(), + CHECK_THROWS_WITH(parse_string("[1").parse(), "[json.exception.parse_error.101] parse error at 3: syntax error - unexpected end of input; expected ']'"); - CHECK_THROWS_WITH(json::parser("[1,").parse(), + CHECK_THROWS_WITH(parse_string("[1,").parse(), "[json.exception.parse_error.101] parse error at 4: syntax error - unexpected end of input"); - CHECK_THROWS_WITH(json::parser("[1,]").parse(), + CHECK_THROWS_WITH(parse_string("[1,]").parse(), "[json.exception.parse_error.101] parse error at 4: syntax error - unexpected ']'"); - CHECK_THROWS_WITH(json::parser("]").parse(), + CHECK_THROWS_WITH(parse_string("]").parse(), "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected ']'"); // missing/unexpected end of object - CHECK_THROWS_AS(json::parser("{").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("{\"foo\"").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("{\"foo\":").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("{\"foo\":}").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("{\"foo\":1,}").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("}").parse(), json::parse_error); - CHECK_THROWS_WITH(json::parser("{").parse(), + CHECK_THROWS_AS(parse_string("{").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("{\"foo\"").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("{\"foo\":").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("{\"foo\":}").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("{\"foo\":1,}").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("}").parse(), json::parse_error); + CHECK_THROWS_WITH(parse_string("{").parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected end of input; expected string literal"); - CHECK_THROWS_WITH(json::parser("{\"foo\"").parse(), + CHECK_THROWS_WITH(parse_string("{\"foo\"").parse(), "[json.exception.parse_error.101] parse error at 7: syntax error - unexpected end of input; expected ':'"); - CHECK_THROWS_WITH(json::parser("{\"foo\":").parse(), + CHECK_THROWS_WITH(parse_string("{\"foo\":").parse(), "[json.exception.parse_error.101] parse error at 8: syntax error - unexpected end of input"); - CHECK_THROWS_WITH(json::parser("{\"foo\":}").parse(), + CHECK_THROWS_WITH(parse_string("{\"foo\":}").parse(), "[json.exception.parse_error.101] parse error at 8: syntax error - unexpected '}'"); - CHECK_THROWS_WITH(json::parser("{\"foo\":1,}").parse(), + CHECK_THROWS_WITH(parse_string("{\"foo\":1,}").parse(), "[json.exception.parse_error.101] parse error at 10: syntax error - unexpected '}'; expected string literal"); - CHECK_THROWS_WITH(json::parser("}").parse(), + CHECK_THROWS_WITH(parse_string("}").parse(), "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '}'"); // missing/unexpected end of string - CHECK_THROWS_AS(json::parser("\"").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("\"\\\"").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("\"\\u\"").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("\"\\u0\"").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("\"\\u01\"").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("\"\\u012\"").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("\"\\u").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("\"\\u0").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("\"\\u01").parse(), json::parse_error); - CHECK_THROWS_AS(json::parser("\"\\u012").parse(), json::parse_error); - CHECK_THROWS_WITH(json::parser("\"").parse(), + CHECK_THROWS_AS(parse_string("\"").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("\"\\\"").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("\"\\u\"").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("\"\\u0\"").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("\"\\u01\"").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("\"\\u012\"").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("\"\\u").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("\"\\u0").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("\"\\u01").parse(), json::parse_error); + CHECK_THROWS_AS(parse_string("\"\\u012").parse(), json::parse_error); + CHECK_THROWS_WITH(parse_string("\"").parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: missing closing quote; last read '\"'"); - CHECK_THROWS_WITH(json::parser("\"\\\"").parse(), + CHECK_THROWS_WITH(parse_string("\"\\\"").parse(), "[json.exception.parse_error.101] parse error at 4: syntax error - invalid string: missing closing quote; last read '\"\\\"'"); - CHECK_THROWS_WITH(json::parser("\"\\u\"").parse(), + CHECK_THROWS_WITH(parse_string("\"\\u\"").parse(), "[json.exception.parse_error.101] parse error at 4: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read '\"\\u\"'"); - CHECK_THROWS_WITH(json::parser("\"\\u0\"").parse(), + CHECK_THROWS_WITH(parse_string("\"\\u0\"").parse(), "[json.exception.parse_error.101] parse error at 5: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read '\"\\u0\"'"); - CHECK_THROWS_WITH(json::parser("\"\\u01\"").parse(), + CHECK_THROWS_WITH(parse_string("\"\\u01\"").parse(), "[json.exception.parse_error.101] parse error at 6: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read '\"\\u01\"'"); - CHECK_THROWS_WITH(json::parser("\"\\u012\"").parse(), + CHECK_THROWS_WITH(parse_string("\"\\u012\"").parse(), "[json.exception.parse_error.101] parse error at 7: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read '\"\\u012\"'"); - CHECK_THROWS_WITH(json::parser("\"\\u").parse(), + CHECK_THROWS_WITH(parse_string("\"\\u").parse(), "[json.exception.parse_error.101] parse error at 4: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read '\"\\u'"); - CHECK_THROWS_WITH(json::parser("\"\\u0").parse(), + CHECK_THROWS_WITH(parse_string("\"\\u0").parse(), "[json.exception.parse_error.101] parse error at 5: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read '\"\\u0'"); - CHECK_THROWS_WITH(json::parser("\"\\u01").parse(), + CHECK_THROWS_WITH(parse_string("\"\\u01").parse(), "[json.exception.parse_error.101] parse error at 6: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read '\"\\u01'"); - CHECK_THROWS_WITH(json::parser("\"\\u012").parse(), + CHECK_THROWS_WITH(parse_string("\"\\u012").parse(), "[json.exception.parse_error.101] parse error at 7: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read '\"\\u012'"); // invalid escapes @@ -511,7 +518,7 @@ TEST_CASE("parser class") case ('r'): case ('t'): { - CHECK_NOTHROW(json::parser(s.c_str()).parse()); + CHECK_NOTHROW(parse_string(s.c_str()).parse()); break; } @@ -524,11 +531,11 @@ TEST_CASE("parser class") // any other combination of backslash and character is invalid default: { - CHECK_THROWS_AS(json::parser(s.c_str()).parse(), json::parse_error); + CHECK_THROWS_AS(parse_string(s.c_str()).parse(), json::parse_error); // only check error message if c is not a control character if (c > 0x1f) { - CHECK_THROWS_WITH(json::parser(s.c_str()).parse(), + CHECK_THROWS_WITH(parse_string(s.c_str()).parse(), "[json.exception.parse_error.101] parse error at 3: syntax error - invalid string: forbidden character after backspace; last read '\"\\" + std::string(1, static_cast(c)) + "'"); } break; @@ -589,49 +596,49 @@ TEST_CASE("parser class") if (valid(c)) { CAPTURE(s1); - CHECK_NOTHROW(json::parser(s1.c_str()).parse()); + CHECK_NOTHROW(parse_string(s1.c_str()).parse()); CAPTURE(s2); - CHECK_NOTHROW(json::parser(s2.c_str()).parse()); + CHECK_NOTHROW(parse_string(s2.c_str()).parse()); CAPTURE(s3); - CHECK_NOTHROW(json::parser(s3.c_str()).parse()); + CHECK_NOTHROW(parse_string(s3.c_str()).parse()); CAPTURE(s4); - CHECK_NOTHROW(json::parser(s4.c_str()).parse()); + CHECK_NOTHROW(parse_string(s4.c_str()).parse()); } else { CAPTURE(s1); - CHECK_THROWS_AS(json::parser(s1.c_str()).parse(), json::parse_error); + CHECK_THROWS_AS(parse_string(s1.c_str()).parse(), json::parse_error); // only check error message if c is not a control character if (c > 0x1f) { - CHECK_THROWS_WITH(json::parser(s1.c_str()).parse(), + CHECK_THROWS_WITH(parse_string(s1.c_str()).parse(), "[json.exception.parse_error.101] parse error at 7: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read '" + s1.substr(0, 7) + "'"); } CAPTURE(s2); - CHECK_THROWS_AS(json::parser(s2.c_str()).parse(), json::parse_error); + CHECK_THROWS_AS(parse_string(s2.c_str()).parse(), json::parse_error); // only check error message if c is not a control character if (c > 0x1f) { - CHECK_THROWS_WITH(json::parser(s2.c_str()).parse(), + CHECK_THROWS_WITH(parse_string(s2.c_str()).parse(), "[json.exception.parse_error.101] parse error at 6: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read '" + s2.substr(0, 6) + "'"); } CAPTURE(s3); - CHECK_THROWS_AS(json::parser(s3.c_str()).parse(), json::parse_error); + CHECK_THROWS_AS(parse_string(s3.c_str()).parse(), json::parse_error); // only check error message if c is not a control character if (c > 0x1f) { - CHECK_THROWS_WITH(json::parser(s3.c_str()).parse(), + CHECK_THROWS_WITH(parse_string(s3.c_str()).parse(), "[json.exception.parse_error.101] parse error at 5: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read '" + s3.substr(0, 5) + "'"); } CAPTURE(s4); - CHECK_THROWS_AS(json::parser(s4.c_str()).parse(), json::parse_error); + CHECK_THROWS_AS(parse_string(s4.c_str()).parse(), json::parse_error); // only check error message if c is not a control character if (c > 0x1f) { - CHECK_THROWS_WITH(json::parser(s4.c_str()).parse(), + CHECK_THROWS_WITH(parse_string(s4.c_str()).parse(), "[json.exception.parse_error.101] parse error at 4: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read '" + s4.substr(0, 4) + "'"); } } @@ -657,12 +664,12 @@ TEST_CASE("parser class") SECTION("tests found by mutate++") { // test case to make sure no comma preceeds the first key - CHECK_THROWS_AS(json::parser("{,\"key\": false}").parse(), json::parse_error); - CHECK_THROWS_WITH(json::parser("{,\"key\": false}").parse(), + CHECK_THROWS_AS(parse_string("{,\"key\": false}").parse(), json::parse_error); + CHECK_THROWS_WITH(parse_string("{,\"key\": false}").parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected ','; expected string literal"); // test case to make sure an object is properly closed - CHECK_THROWS_AS(json::parser("[{\"key\": false true]").parse(), json::parse_error); - CHECK_THROWS_WITH(json::parser("[{\"key\": false true]").parse(), + CHECK_THROWS_AS(parse_string("[{\"key\": false true]").parse(), json::parse_error); + CHECK_THROWS_WITH(parse_string("[{\"key\": false true]").parse(), "[json.exception.parse_error.101] parse error at 19: syntax error - unexpected true literal; expected '}'"); // test case to make sure the callback is properly evaluated after reading a key @@ -850,42 +857,42 @@ TEST_CASE("parser class") SECTION("from std::vector") { std::vector v = {'t', 'r', 'u', 'e'}; - CHECK(json::parser(std::begin(v), std::end(v)).parse() == json(true)); + CHECK(json::parser(json::input_adapter::create(std::begin(v), std::end(v))).parse() == json(true)); } SECTION("from std::array") { std::array v { {'t', 'r', 'u', 'e'} }; - CHECK(json::parser(std::begin(v), std::end(v)).parse() == json(true)); + CHECK(json::parser(json::input_adapter::create(std::begin(v), std::end(v))).parse() == json(true)); } SECTION("from array") { uint8_t v[] = {'t', 'r', 'u', 'e'}; - CHECK(json::parser(std::begin(v), std::end(v)).parse() == json(true)); + CHECK(json::parser(json::input_adapter::create(std::begin(v), std::end(v))).parse() == json(true)); } SECTION("from char literal") { - CHECK(json::parser("true").parse() == json(true)); + CHECK(parse_string("true").parse() == json(true)); } SECTION("from std::string") { std::string v = {'t', 'r', 'u', 'e'}; - CHECK(json::parser(std::begin(v), std::end(v)).parse() == json(true)); + CHECK(json::parser(json::input_adapter::create(std::begin(v), std::end(v))).parse() == json(true)); } SECTION("from std::initializer_list") { std::initializer_list v = {'t', 'r', 'u', 'e'}; - CHECK(json::parser(std::begin(v), std::end(v)).parse() == json(true)); + CHECK(json::parser(json::input_adapter::create(std::begin(v), std::end(v))).parse() == json(true)); } SECTION("from std::valarray") { std::valarray v = {'t', 'r', 'u', 'e'}; - CHECK(json::parser(std::begin(v), std::end(v)).parse() == json(true)); + CHECK(json::parser(json::input_adapter::create(std::begin(v), std::end(v))).parse() == json(true)); } } } From f3e43d7c6fcfc5f8c8b08020a13b9a44f431416d Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 9 Apr 2017 21:14:51 +0200 Subject: [PATCH 157/530] :hammer: some cleanup --- src/json.hpp | 19 +++++++++++-------- 1 file changed, 11 insertions(+), 8 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 20c5b2bc..f50650df 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -8626,6 +8626,9 @@ class basic_json } }; + // a type to simplify interfaces + using input_adapter_t = std::shared_ptr; + /// input adapter for cached stream input class cached_input_stream_adapter : public input_adapter { @@ -8783,8 +8786,8 @@ class basic_json class binary_reader { public: - explicit binary_reader(std::shared_ptr a) - : ia(a), is_little_endian(little_endianess()) + explicit binary_reader(input_adapter_t adapter) + : ia(adapter), is_little_endian(little_endianess()) {} /*! @@ -9807,7 +9810,7 @@ class basic_json private: /// input adapter - std::shared_ptr ia = nullptr; + input_adapter_t ia = nullptr; /// the current character int current = std::char_traits::eof(); @@ -10761,8 +10764,8 @@ class basic_json } } - explicit lexer(std::shared_ptr a) - : ia(a), decimal_point_char(get_decimal_point()) + explicit lexer(input_adapter_t adapter) + : ia(adapter), decimal_point_char(get_decimal_point()) {} private: @@ -12118,7 +12121,7 @@ scan_number_done: private: /// input adapter - std::shared_ptr ia = nullptr; + input_adapter_t ia = nullptr; /// the current character int current = std::char_traits::eof(); @@ -12157,9 +12160,9 @@ scan_number_done: { public: /// a parser reading from an input adapter - explicit parser(std::shared_ptr ia, + explicit parser(input_adapter_t adapter, const parser_callback_t cb = nullptr) - : callback(cb), m_lexer(ia) + : callback(cb), m_lexer(adapter) {} /*! From d7e57e3b737b4d49cbfd52c064940809bc32821c Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 9 Apr 2017 22:13:26 +0200 Subject: [PATCH 158/530] :hammer: added an output adapter for the binary writer --- src/json.hpp | 198 ++++++++++++++++++++++++++++++--------------------- 1 file changed, 117 insertions(+), 81 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index f50650df..eab01519 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -8775,6 +8775,46 @@ class basic_json const char* start; }; + ///////////////////// + // output adapters // + ///////////////////// + + class output_adapter + { + public: + virtual void write_character(uint8_t c) = 0; + virtual void write_characters(const uint8_t* s, size_t length) = 0; + virtual ~output_adapter() {} + + static std::shared_ptr create(std::vector& vec) + { + return std::shared_ptr(new output_vector_adapter(vec)); + } + }; + + using output_adapter_t = std::shared_ptr; + + class output_vector_adapter : public output_adapter + { + public: + output_vector_adapter(std::vector& vec) + : v(vec) + {} + + void write_character(uint8_t c) override + { + v.push_back(c); + } + + void write_characters(const uint8_t* s, size_t length) override + { + std::copy(s, s + length, std::back_inserter(v)); + } + + private: + std::vector& v; + }; + ////////////////////////////////////////// // binary serialization/deserialization // ////////////////////////////////////////// @@ -9829,32 +9869,23 @@ class basic_json : is_little_endian(little_endianess()) {} - std::vector write_cbor(const basic_json& j) - { - write_cbor_internal(j); - return v; - } + explicit binary_writer(output_adapter_t adapter) + : is_little_endian(little_endianess()), oa(adapter) + {} - std::vector write_msgpack(const basic_json& j) - { - write_msgpack_internal(j); - return v; - } - - private: - void write_cbor_internal(const basic_json& j) + void write_cbor(const basic_json& j) { switch (j.type()) { case value_t::null: { - v.push_back(0xf6); + oa->write_character(0xf6); break; } case value_t::boolean: { - v.push_back(j.m_value.boolean ? 0xf5 : 0xf4); + oa->write_character(j.m_value.boolean ? 0xf5 : 0xf4); break; } @@ -9871,22 +9902,22 @@ class basic_json } else if (j.m_value.number_integer <= (std::numeric_limits::max)()) { - v.push_back(0x18); + oa->write_character(0x18); write_number(static_cast(j.m_value.number_integer)); } else if (j.m_value.number_integer <= (std::numeric_limits::max)()) { - v.push_back(0x19); + oa->write_character(0x19); write_number(static_cast(j.m_value.number_integer)); } else if (j.m_value.number_integer <= (std::numeric_limits::max)()) { - v.push_back(0x1a); + oa->write_character(0x1a); write_number(static_cast(j.m_value.number_integer)); } else { - v.push_back(0x1b); + oa->write_character(0x1b); write_number(static_cast(j.m_value.number_integer)); } } @@ -9901,22 +9932,22 @@ class basic_json } else if (positive_number <= (std::numeric_limits::max)()) { - v.push_back(0x38); + oa->write_character(0x38); write_number(static_cast(positive_number)); } else if (positive_number <= (std::numeric_limits::max)()) { - v.push_back(0x39); + oa->write_character(0x39); write_number(static_cast(positive_number)); } else if (positive_number <= (std::numeric_limits::max)()) { - v.push_back(0x3a); + oa->write_character(0x3a); write_number(static_cast(positive_number)); } else { - v.push_back(0x3b); + oa->write_character(0x3b); write_number(static_cast(positive_number)); } } @@ -9931,22 +9962,22 @@ class basic_json } else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { - v.push_back(0x18); + oa->write_character(0x18); write_number(static_cast(j.m_value.number_unsigned)); } else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { - v.push_back(0x19); + oa->write_character(0x19); write_number(static_cast(j.m_value.number_unsigned)); } else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { - v.push_back(0x1a); + oa->write_character(0x1a); write_number(static_cast(j.m_value.number_unsigned)); } else { - v.push_back(0x1b); + oa->write_character(0x1b); write_number(static_cast(j.m_value.number_unsigned)); } break; @@ -9955,7 +9986,7 @@ class basic_json case value_t::number_float: { // Double-Precision Float - v.push_back(0xfb); + oa->write_character(0xfb); write_number(j.m_value.number_float); break; } @@ -9969,30 +10000,30 @@ class basic_json } else if (N <= 0xff) { - v.push_back(0x78); + oa->write_character(0x78); write_number(static_cast(N)); } else if (N <= 0xffff) { - v.push_back(0x79); + oa->write_character(0x79); write_number(static_cast(N)); } else if (N <= 0xffffffff) { - v.push_back(0x7a); + oa->write_character(0x7a); write_number(static_cast(N)); } // LCOV_EXCL_START else if (N <= 0xffffffffffffffff) { - v.push_back(0x7b); + oa->write_character(0x7b); write_number(static_cast(N)); } // LCOV_EXCL_STOP // append string - std::copy(j.m_value.string->begin(), j.m_value.string->end(), - std::back_inserter(v)); + oa->write_characters(reinterpret_cast(j.m_value.string->c_str()), + j.m_value.string->size()); break; } @@ -10005,23 +10036,23 @@ class basic_json } else if (N <= 0xff) { - v.push_back(0x98); + oa->write_character(0x98); write_number(static_cast(N)); } else if (N <= 0xffff) { - v.push_back(0x99); + oa->write_character(0x99); write_number(static_cast(N)); } else if (N <= 0xffffffff) { - v.push_back(0x9a); + oa->write_character(0x9a); write_number(static_cast(N)); } // LCOV_EXCL_START else if (N <= 0xffffffffffffffff) { - v.push_back(0x9b); + oa->write_character(0x9b); write_number(static_cast(N)); } // LCOV_EXCL_STOP @@ -10029,7 +10060,7 @@ class basic_json // append each element for (const auto& el : *j.m_value.array) { - write_cbor_internal(el); + write_cbor(el); } break; } @@ -10043,23 +10074,23 @@ class basic_json } else if (N <= 0xff) { - v.push_back(0xb8); + oa->write_character(0xb8); write_number(static_cast(N)); } else if (N <= 0xffff) { - v.push_back(0xb9); + oa->write_character(0xb9); write_number(static_cast(N)); } else if (N <= 0xffffffff) { - v.push_back(0xba); + oa->write_character(0xba); write_number(static_cast(N)); } // LCOV_EXCL_START else if (N <= 0xffffffffffffffff) { - v.push_back(0xbb); + oa->write_character(0xbb); write_number(static_cast(N)); } // LCOV_EXCL_STOP @@ -10067,8 +10098,8 @@ class basic_json // append each element for (const auto& el : *j.m_value.object) { - write_cbor_internal(el.first); - write_cbor_internal(el.second); + write_cbor(el.first); + write_cbor(el.second); } break; } @@ -10080,21 +10111,21 @@ class basic_json } } - void write_msgpack_internal(const basic_json& j) + void write_msgpack(const basic_json& j) { switch (j.type()) { case value_t::null: { // nil - v.push_back(0xc0); + oa->write_character(0xc0); break; } case value_t::boolean: { // true and false - v.push_back(j.m_value.boolean ? 0xc3 : 0xc2); + oa->write_character(j.m_value.boolean ? 0xc3 : 0xc2); break; } @@ -10114,25 +10145,25 @@ class basic_json else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { // uint 8 - v.push_back(0xcc); + oa->write_character(0xcc); write_number(static_cast(j.m_value.number_integer)); } else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { // uint 16 - v.push_back(0xcd); + oa->write_character(0xcd); write_number(static_cast(j.m_value.number_integer)); } else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { // uint 32 - v.push_back(0xce); + oa->write_character(0xce); write_number(static_cast(j.m_value.number_integer)); } else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { // uint 64 - v.push_back(0xcf); + oa->write_character(0xcf); write_number(static_cast(j.m_value.number_integer)); } } @@ -10146,25 +10177,25 @@ class basic_json else if (j.m_value.number_integer >= (std::numeric_limits::min)() and j.m_value.number_integer <= (std::numeric_limits::max)()) { // int 8 - v.push_back(0xd0); + oa->write_character(0xd0); write_number(static_cast(j.m_value.number_integer)); } else if (j.m_value.number_integer >= (std::numeric_limits::min)() and j.m_value.number_integer <= (std::numeric_limits::max)()) { // int 16 - v.push_back(0xd1); + oa->write_character(0xd1); write_number(static_cast(j.m_value.number_integer)); } else if (j.m_value.number_integer >= (std::numeric_limits::min)() and j.m_value.number_integer <= (std::numeric_limits::max)()) { // int 32 - v.push_back(0xd2); + oa->write_character(0xd2); write_number(static_cast(j.m_value.number_integer)); } else if (j.m_value.number_integer >= (std::numeric_limits::min)() and j.m_value.number_integer <= (std::numeric_limits::max)()) { // int 64 - v.push_back(0xd3); + oa->write_character(0xd3); write_number(static_cast(j.m_value.number_integer)); } } @@ -10181,25 +10212,25 @@ class basic_json else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { // uint 8 - v.push_back(0xcc); + oa->write_character(0xcc); write_number(static_cast(j.m_value.number_integer)); } else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { // uint 16 - v.push_back(0xcd); + oa->write_character(0xcd); write_number(static_cast(j.m_value.number_integer)); } else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { // uint 32 - v.push_back(0xce); + oa->write_character(0xce); write_number(static_cast(j.m_value.number_integer)); } else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { // uint 64 - v.push_back(0xcf); + oa->write_character(0xcf); write_number(static_cast(j.m_value.number_integer)); } break; @@ -10208,7 +10239,7 @@ class basic_json case value_t::number_float: { // float 64 - v.push_back(0xcb); + oa->write_character(0xcb); write_number(j.m_value.number_float); break; } @@ -10224,25 +10255,25 @@ class basic_json else if (N <= 255) { // str 8 - v.push_back(0xd9); + oa->write_character(0xd9); write_number(static_cast(N)); } else if (N <= 65535) { // str 16 - v.push_back(0xda); + oa->write_character(0xda); write_number(static_cast(N)); } else if (N <= 4294967295) { // str 32 - v.push_back(0xdb); + oa->write_character(0xdb); write_number(static_cast(N)); } // append string - std::copy(j.m_value.string->begin(), j.m_value.string->end(), - std::back_inserter(v)); + oa->write_characters(reinterpret_cast(j.m_value.string->c_str()), + j.m_value.string->size()); break; } @@ -10257,20 +10288,20 @@ class basic_json else if (N <= 0xffff) { // array 16 - v.push_back(0xdc); + oa->write_character(0xdc); write_number(static_cast(N)); } else if (N <= 0xffffffff) { // array 32 - v.push_back(0xdd); + oa->write_character(0xdd); write_number(static_cast(N)); } // append each element for (const auto& el : *j.m_value.array) { - write_msgpack_internal(el); + write_msgpack(el); } break; } @@ -10286,21 +10317,21 @@ class basic_json else if (N <= 65535) { // map 16 - v.push_back(0xde); + oa->write_character(0xde); write_number(static_cast(N)); } else if (N <= 4294967295) { // map 32 - v.push_back(0xdf); + oa->write_character(0xdf); write_number(static_cast(N)); } // append each element for (const auto& el : *j.m_value.object) { - write_msgpack_internal(el.first); - write_msgpack_internal(el.second); + write_msgpack(el.first); + write_msgpack(el.second); } break; } @@ -10312,6 +10343,7 @@ class basic_json } } + private: template void write_number(T n) { @@ -10323,11 +10355,11 @@ class basic_json // reverse byte order prior to conversion if necessary if (is_little_endian) { - v.push_back(vec[sizeof(T) - i - 1]); + oa->write_character(vec[sizeof(T) - i - 1]); } else { - v.push_back(vec[i]); + oa->write_character(vec[i]); } } } @@ -10343,8 +10375,8 @@ class basic_json /// whether we can assume little endianess const bool is_little_endian = true; - /// the vector that is used as output - std::vector v {}; + /// the output + output_adapter_t oa = nullptr; }; public: @@ -10432,8 +10464,10 @@ class basic_json */ static std::vector to_cbor(const basic_json& j) { - binary_writer bw; - return bw.write_cbor(j); + std::vector result; + binary_writer bw(output_adapter::create(result)); + bw.write_cbor(j); + return result; } /*! @@ -10512,8 +10546,10 @@ class basic_json */ static std::vector to_msgpack(const basic_json& j) { - binary_writer bw; - return bw.write_msgpack(j); + std::vector result; + binary_writer bw(output_adapter::create(result)); + bw.write_msgpack(j); + return result; } /*! From 5bea6f383af7e2d019f4f95d03775daf8860b908 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 11 Apr 2017 18:57:53 +0200 Subject: [PATCH 159/530] :pencil2: fixed a typo --- README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.md b/README.md index f8174fa8..265ec06f 100644 --- a/README.md +++ b/README.md @@ -841,7 +841,7 @@ Thanks a lot for helping out! Please [let me know](mailto:mail@nlohmann.me) if I ## Used third-party tools -The library itself contains of a single header file licensed under the MIT license. However, it is built, tested, documented, and whatnot using a lot of thirs-party tools and services. Thanks a lot! +The library itself contains of a single header file licensed under the MIT license. However, it is built, tested, documented, and whatnot using a lot of third-party tools and services. Thanks a lot! - [**American fuzzy lop**](http://lcamtuf.coredump.cx/afl/) for fuzz testing - [**AppVeyor**](https://www.appveyor.com) for [continuous integration](https://ci.appveyor.com/project/nlohmann/json) on Windows From ef55626296db745f382629ba0c31ef58e2f3b58a Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 11 Apr 2017 20:37:38 +0200 Subject: [PATCH 160/530] :construction_worker: added Clang 3.9-5.0 Taken from https://github.com/travis-ci/docs-travis-ci-com/pull/746/files. --- .travis.yml | 21 +++++++++++++++++++++ 1 file changed, 21 insertions(+) diff --git a/.travis.yml b/.travis.yml index 8b9ba53e..f69283b0 100644 --- a/.travis.yml +++ b/.travis.yml @@ -207,6 +207,27 @@ matrix: env: LLVM_VERSION=3.8.1 compiler: clang + - os: linux + addons: + apt: + sources: llvm-toolchain-trusty-3.9 + packages: clang-3.9 + env: COMPILER=clang++-3.9 + + - os: linux + addons: + apt: + sources: llvm-toolchain-trusty-4.0 + packages: clang-4.0 + env: COMPILER=clang++-4.0 + + - os: linux + addons: + apt: + sources: llvm-toolchain-trusty + packages: clang-5.0 + env: COMPILER=clang++-5.0 + ##################### # installation step # ##################### From 42b4cdd4d00a0289932fe6af1f71c0a875041cc3 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 11 Apr 2017 22:13:26 +0200 Subject: [PATCH 161/530] :construction_worker: Travis is very picky --- .travis.yml | 40 ++++++++++++++++++++-------------------- 1 file changed, 20 insertions(+), 20 deletions(-) diff --git a/.travis.yml b/.travis.yml index f69283b0..0d7026b0 100644 --- a/.travis.yml +++ b/.travis.yml @@ -207,26 +207,26 @@ matrix: env: LLVM_VERSION=3.8.1 compiler: clang - - os: linux - addons: - apt: - sources: llvm-toolchain-trusty-3.9 - packages: clang-3.9 - env: COMPILER=clang++-3.9 - - - os: linux - addons: - apt: - sources: llvm-toolchain-trusty-4.0 - packages: clang-4.0 - env: COMPILER=clang++-4.0 - - - os: linux - addons: - apt: - sources: llvm-toolchain-trusty - packages: clang-5.0 - env: COMPILER=clang++-5.0 +# - os: linux +# addons: +# apt: +# sources: llvm-toolchain-trusty-3.9 +# packages: clang-3.9 +# env: COMPILER=clang++-3.9 +# +# - os: linux +# addons: +# apt: +# sources: llvm-toolchain-trusty-4.0 +# packages: clang-4.0 +# env: COMPILER=clang++-4.0 +# +# - os: linux +# addons: +# apt: +# sources: llvm-toolchain-trusty +# packages: clang-5.0 +# env: COMPILER=clang++-5.0 ##################### # installation step # From e2f6cf7f46707e03fc634a0a968058f4b73432a0 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Fri, 14 Apr 2017 17:35:31 +0200 Subject: [PATCH 162/530] :fire: removed .test files These files were never meant to be versioned. --- doc/examples/at__object_t_key_type.test | 3 --- doc/examples/get_ref.test | 2 -- doc/examples/json_pointer.test | 3 --- doc/examples/operator__equal.test | 4 ---- doc/examples/operator__notequal.test | 4 ---- 5 files changed, 16 deletions(-) delete mode 100644 doc/examples/at__object_t_key_type.test delete mode 100644 doc/examples/get_ref.test delete mode 100644 doc/examples/json_pointer.test delete mode 100644 doc/examples/operator__equal.test delete mode 100644 doc/examples/operator__notequal.test diff --git a/doc/examples/at__object_t_key_type.test b/doc/examples/at__object_t_key_type.test deleted file mode 100644 index 654b9eb6..00000000 --- a/doc/examples/at__object_t_key_type.test +++ /dev/null @@ -1,3 +0,0 @@ -"il brutto" -{"the bad":"il cattivo","the good":"il buono","the ugly":"il brutto"} -[json.exception.out_of_range.403] key 'the fast' not found diff --git a/doc/examples/get_ref.test b/doc/examples/get_ref.test deleted file mode 100644 index 3811afa2..00000000 --- a/doc/examples/get_ref.test +++ /dev/null @@ -1,2 +0,0 @@ -17 17 -[json.exception.type_error.303] incompatible ReferenceType for get_ref, actual type is number diff --git a/doc/examples/json_pointer.test b/doc/examples/json_pointer.test deleted file mode 100644 index 33e2cc68..00000000 --- a/doc/examples/json_pointer.test +++ /dev/null @@ -1,3 +0,0 @@ -[json.exception.parse_error.107] parse error at 1: JSON pointer must be empty or begin with '/' - was: 'foo' -[json.exception.parse_error.108] parse error: escape character '~' must be followed with '0' or '1' -[json.exception.parse_error.108] parse error: escape character '~' must be followed with '0' or '1' diff --git a/doc/examples/operator__equal.test b/doc/examples/operator__equal.test deleted file mode 100644 index e9dfd755..00000000 --- a/doc/examples/operator__equal.test +++ /dev/null @@ -1,4 +0,0 @@ -[1,2,3] == [1,2,4] false -{"A":"a","B":"b"} == {"A":"a","B":"b"} true -17 == 17 true -"foo" == "bar" false diff --git a/doc/examples/operator__notequal.test b/doc/examples/operator__notequal.test deleted file mode 100644 index ddd838b4..00000000 --- a/doc/examples/operator__notequal.test +++ /dev/null @@ -1,4 +0,0 @@ -[1,2,3] == [1,2,4] true -{"A":"a","B":"b"} == {"A":"a","B":"b"} false -17 == 17 false -"foo" == "bar" true From 951d0920fc0a42476846bc2e6eb3ed7428dc81c1 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Fri, 14 Apr 2017 17:36:45 +0200 Subject: [PATCH 163/530] :bug: fixed README example The example with the size() operator is bad: using operator[] already changes the size of the object. The fix makes it clearer. --- doc/examples/README.cpp | 4 +++- doc/examples/README.output | 27 --------------------------- 2 files changed, 3 insertions(+), 28 deletions(-) delete mode 100644 doc/examples/README.output diff --git a/doc/examples/README.cpp b/doc/examples/README.cpp index b928e116..d8cd4d67 100644 --- a/doc/examples/README.cpp +++ b/doc/examples/README.cpp @@ -1,4 +1,5 @@ #include +#include // for std::setw using json = nlohmann::json; @@ -29,7 +30,8 @@ int main() j["new"]["key"]["value"] = {"another", "list"}; // count elements - j["size"] = j.size(); + auto s = j.size(); + j["size"] = s; // pretty print with indent of 4 spaces std::cout << std::setw(4) << j << '\n'; diff --git a/doc/examples/README.output b/doc/examples/README.output deleted file mode 100644 index 3226f729..00000000 --- a/doc/examples/README.output +++ /dev/null @@ -1,27 +0,0 @@ -{ - "answer": { - "everything": 42 - }, - "happy": true, - "list": [ - 1, - 0, - 2 - ], - "name": "Niels", - "new": { - "key": { - "value": [ - "another", - "list" - ] - } - }, - "nothing": null, - "object": { - "currency": "USD", - "value": 42.99 - }, - "pi": 3.141, - "size": 9 -} From 6b6e5540673338839f10ff40cee3661007ab17ec Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Fri, 14 Apr 2017 17:37:28 +0200 Subject: [PATCH 164/530] :bug: added missing header std::setw needs the iomanip header (at least with GCC). --- doc/examples/diff.cpp | 1 + doc/examples/flatten.cpp | 1 + doc/examples/from_cbor.cpp | 1 + doc/examples/from_msgpack.cpp | 1 + doc/examples/meta.cpp | 1 + doc/examples/operator_deserialize.cpp | 1 + doc/examples/operator_serialize.cpp | 1 + doc/examples/operatorarray__key_type.cpp | 1 + doc/examples/parse__array__parser_callback_t.cpp | 1 + doc/examples/parse__contiguouscontainer__parser_callback_t.cpp | 1 + doc/examples/parse__istream__parser_callback_t.cpp | 1 + doc/examples/parse__iteratortype__parser_callback_t.cpp | 1 + doc/examples/parse__string__parser_callback_t.cpp | 1 + doc/examples/patch.cpp | 1 + doc/examples/to_cbor.cpp | 1 + doc/examples/to_msgpack.cpp | 1 + doc/examples/unflatten.cpp | 1 + 17 files changed, 17 insertions(+) diff --git a/doc/examples/diff.cpp b/doc/examples/diff.cpp index d81a58db..a377fc81 100644 --- a/doc/examples/diff.cpp +++ b/doc/examples/diff.cpp @@ -1,4 +1,5 @@ #include +#include // for std::setw using json = nlohmann::json; diff --git a/doc/examples/flatten.cpp b/doc/examples/flatten.cpp index ace53a21..e7b7ee87 100644 --- a/doc/examples/flatten.cpp +++ b/doc/examples/flatten.cpp @@ -1,4 +1,5 @@ #include +#include // for std::setw using json = nlohmann::json; diff --git a/doc/examples/from_cbor.cpp b/doc/examples/from_cbor.cpp index 92b05225..fc44d4a5 100644 --- a/doc/examples/from_cbor.cpp +++ b/doc/examples/from_cbor.cpp @@ -1,4 +1,5 @@ #include +#include // for std::setw using json = nlohmann::json; diff --git a/doc/examples/from_msgpack.cpp b/doc/examples/from_msgpack.cpp index d275f13a..7c4202c9 100644 --- a/doc/examples/from_msgpack.cpp +++ b/doc/examples/from_msgpack.cpp @@ -1,4 +1,5 @@ #include +#include // for std::setw using json = nlohmann::json; diff --git a/doc/examples/meta.cpp b/doc/examples/meta.cpp index 3a31ca24..28c88b8d 100644 --- a/doc/examples/meta.cpp +++ b/doc/examples/meta.cpp @@ -1,4 +1,5 @@ #include +#include // for std::setw using json = nlohmann::json; diff --git a/doc/examples/operator_deserialize.cpp b/doc/examples/operator_deserialize.cpp index a43cdf0f..06a98db9 100644 --- a/doc/examples/operator_deserialize.cpp +++ b/doc/examples/operator_deserialize.cpp @@ -1,4 +1,5 @@ #include +#include // for std::setw using json = nlohmann::json; diff --git a/doc/examples/operator_serialize.cpp b/doc/examples/operator_serialize.cpp index c1568d99..c3861b46 100644 --- a/doc/examples/operator_serialize.cpp +++ b/doc/examples/operator_serialize.cpp @@ -1,4 +1,5 @@ #include +#include // for std::setw using json = nlohmann::json; diff --git a/doc/examples/operatorarray__key_type.cpp b/doc/examples/operatorarray__key_type.cpp index e83a2ac0..795abc08 100644 --- a/doc/examples/operatorarray__key_type.cpp +++ b/doc/examples/operatorarray__key_type.cpp @@ -1,4 +1,5 @@ #include +#include // for std::setw using json = nlohmann::json; diff --git a/doc/examples/parse__array__parser_callback_t.cpp b/doc/examples/parse__array__parser_callback_t.cpp index 8e086d20..68ecf602 100644 --- a/doc/examples/parse__array__parser_callback_t.cpp +++ b/doc/examples/parse__array__parser_callback_t.cpp @@ -1,4 +1,5 @@ #include +#include // for std::setw using json = nlohmann::json; diff --git a/doc/examples/parse__contiguouscontainer__parser_callback_t.cpp b/doc/examples/parse__contiguouscontainer__parser_callback_t.cpp index 5a339079..45e376d9 100644 --- a/doc/examples/parse__contiguouscontainer__parser_callback_t.cpp +++ b/doc/examples/parse__contiguouscontainer__parser_callback_t.cpp @@ -1,4 +1,5 @@ #include +#include // for std::setw using json = nlohmann::json; diff --git a/doc/examples/parse__istream__parser_callback_t.cpp b/doc/examples/parse__istream__parser_callback_t.cpp index 6812a5e2..b5f84da4 100644 --- a/doc/examples/parse__istream__parser_callback_t.cpp +++ b/doc/examples/parse__istream__parser_callback_t.cpp @@ -1,4 +1,5 @@ #include +#include // for std::setw using json = nlohmann::json; diff --git a/doc/examples/parse__iteratortype__parser_callback_t.cpp b/doc/examples/parse__iteratortype__parser_callback_t.cpp index 3f723c5f..c7a5bf83 100644 --- a/doc/examples/parse__iteratortype__parser_callback_t.cpp +++ b/doc/examples/parse__iteratortype__parser_callback_t.cpp @@ -1,4 +1,5 @@ #include +#include // for std::setw using json = nlohmann::json; diff --git a/doc/examples/parse__string__parser_callback_t.cpp b/doc/examples/parse__string__parser_callback_t.cpp index 0a4f3b53..bea4914f 100644 --- a/doc/examples/parse__string__parser_callback_t.cpp +++ b/doc/examples/parse__string__parser_callback_t.cpp @@ -1,4 +1,5 @@ #include +#include // for std::setw using json = nlohmann::json; diff --git a/doc/examples/patch.cpp b/doc/examples/patch.cpp index 24a52d59..4dfada32 100644 --- a/doc/examples/patch.cpp +++ b/doc/examples/patch.cpp @@ -1,4 +1,5 @@ #include +#include // for std::setw using json = nlohmann::json; diff --git a/doc/examples/to_cbor.cpp b/doc/examples/to_cbor.cpp index 21a5ce86..2d07e8ab 100644 --- a/doc/examples/to_cbor.cpp +++ b/doc/examples/to_cbor.cpp @@ -1,4 +1,5 @@ #include +#include // for std::setw using json = nlohmann::json; diff --git a/doc/examples/to_msgpack.cpp b/doc/examples/to_msgpack.cpp index 21c8817a..dad83542 100644 --- a/doc/examples/to_msgpack.cpp +++ b/doc/examples/to_msgpack.cpp @@ -1,4 +1,5 @@ #include +#include // for std::setw using json = nlohmann::json; diff --git a/doc/examples/unflatten.cpp b/doc/examples/unflatten.cpp index e2b9b6b8..a01a6349 100644 --- a/doc/examples/unflatten.cpp +++ b/doc/examples/unflatten.cpp @@ -1,4 +1,5 @@ #include +#include // for std::setw using json = nlohmann::json; From aea47422a2ba99813ad0c623067fe62f59e1ad51 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Fri, 14 Apr 2017 18:13:01 +0200 Subject: [PATCH 165/530] :memo: updated links and output --- doc/examples/README.link | 2 +- doc/examples/README.output | 27 +++++++++++++++++++++++++++ doc/examples/diff.link | 2 +- doc/examples/flatten.link | 2 +- doc/examples/from_cbor.link | 2 +- doc/examples/from_msgpack.link | 2 +- 6 files changed, 32 insertions(+), 5 deletions(-) create mode 100644 doc/examples/README.output diff --git a/doc/examples/README.link b/doc/examples/README.link index a774c8ff..9b154c4a 100644 --- a/doc/examples/README.link +++ b/doc/examples/README.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/README.output b/doc/examples/README.output new file mode 100644 index 00000000..31188d45 --- /dev/null +++ b/doc/examples/README.output @@ -0,0 +1,27 @@ +{ + "answer": { + "everything": 42 + }, + "happy": true, + "list": [ + 1, + 0, + 2 + ], + "name": "Niels", + "new": { + "key": { + "value": [ + "another", + "list" + ] + } + }, + "nothing": null, + "object": { + "currency": "USD", + "value": 42.99 + }, + "pi": 3.141, + "size": 8 +} diff --git a/doc/examples/diff.link b/doc/examples/diff.link index c3e3fa4d..459ecfcc 100644 --- a/doc/examples/diff.link +++ b/doc/examples/diff.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/flatten.link b/doc/examples/flatten.link index 50d3841f..ab2128fa 100644 --- a/doc/examples/flatten.link +++ b/doc/examples/flatten.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/from_cbor.link b/doc/examples/from_cbor.link index 81204989..c621bd2d 100644 --- a/doc/examples/from_cbor.link +++ b/doc/examples/from_cbor.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/from_msgpack.link b/doc/examples/from_msgpack.link index 0d5e7831..22bac001 100644 --- a/doc/examples/from_msgpack.link +++ b/doc/examples/from_msgpack.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file From 717106ecedb9cc42882a3964a5afc4016dae6382 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Fri, 14 Apr 2017 19:49:05 +0200 Subject: [PATCH 166/530] :hammer: templated output_adapter and used in class serializer --- src/json.hpp | 248 +++++++++++++++++++++------------- test/src/unit-convenience.cpp | 2 +- 2 files changed, 154 insertions(+), 96 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 2fcb368e..f7fbfd71 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -2897,8 +2897,8 @@ class basic_json */ string_t dump(const int indent = -1) const { - std::stringstream ss; - serializer s(ss); + string_t result; + serializer s(output_adapter::create(result)); if (indent >= 0) { @@ -2909,7 +2909,7 @@ class basic_json s.dump(*this, false, 0); } - return ss.str(); + return result; } /*! @@ -6554,6 +6554,104 @@ class basic_json /// @} + private: + ///////////////////// + // output adapters // + ///////////////////// + + template + class output_adapter + { + public: + virtual void write_character(CharType c) = 0; + virtual void write_characters(const CharType* s, size_t length) = 0; + virtual ~output_adapter() {} + + static std::shared_ptr> create(std::vector& vec) + { + return std::shared_ptr(new output_vector_adapter(vec)); + } + + static std::shared_ptr> create(std::ostream& s) + { + return std::shared_ptr(new output_stream_adapter(s)); + } + + static std::shared_ptr> create(std::string& s) + { + return std::shared_ptr(new output_string_adapter(s)); + } + }; + + template + using output_adapter_t = std::shared_ptr>; + + template + class output_vector_adapter : public output_adapter + { + public: + output_vector_adapter(std::vector& vec) + : v(vec) + {} + + void write_character(CharType c) override + { + v.push_back(c); + } + + void write_characters(const CharType* s, size_t length) override + { + std::copy(s, s + length, std::back_inserter(v)); + } + + private: + std::vector& v; + }; + + template + class output_stream_adapter : public output_adapter + { + public: + output_stream_adapter(std::basic_ostream& s) + : stream(s) + {} + + void write_character(CharType c) override + { + stream.put(c); + } + + void write_characters(const CharType* s, size_t length) override + { + stream.write(s, static_cast(length)); + } + + private: + std::basic_ostream& stream; + }; + + template + class output_string_adapter : public output_adapter + { + public: + output_string_adapter(std::string& s) + : str(s) + {} + + void write_character(CharType c) override + { + str.push_back(c); + } + + void write_characters(const CharType* s, size_t length) override + { + str.append(s, length); + } + + private: + std::basic_string& str; + }; + /////////////////// // serialization // @@ -6576,7 +6674,7 @@ class basic_json /*! @param[in] s output stream to serialize to */ - serializer(std::ostream& s) + serializer(output_adapter_t s) : o(s), loc(std::localeconv()), thousands_sep(!loc->thousands_sep ? '\0' : loc->thousands_sep[0]), decimal_point(!loc->decimal_point ? '\0' : loc->decimal_point[0]) @@ -6610,13 +6708,13 @@ class basic_json { if (val.m_value.object->empty()) { - o.write("{}", 2); + o->write_characters("{}", 2); return; } if (pretty_print) { - o.write("{\n", 2); + o->write_characters("{\n", 2); // variable to hold indentation for recursive calls const auto new_indent = current_indent + indent_step; @@ -6629,49 +6727,49 @@ class basic_json auto i = val.m_value.object->cbegin(); for (size_t cnt = 0; cnt < val.m_value.object->size() - 1; ++cnt, ++i) { - o.write(indent_string.c_str(), static_cast(new_indent)); - o.put('\"'); + o->write_characters(indent_string.c_str(), new_indent); + o->write_character('\"'); dump_escaped(i->first); - o.write("\": ", 3); + o->write_characters("\": ", 3); dump(i->second, true, indent_step, new_indent); - o.write(",\n", 2); + o->write_characters(",\n", 2); } // last element assert(i != val.m_value.object->cend()); - o.write(indent_string.c_str(), static_cast(new_indent)); - o.put('\"'); + o->write_characters(indent_string.c_str(), new_indent); + o->write_character('\"'); dump_escaped(i->first); - o.write("\": ", 3); + o->write_characters("\": ", 3); dump(i->second, true, indent_step, new_indent); - o.put('\n'); - o.write(indent_string.c_str(), static_cast(current_indent)); - o.put('}'); + o->write_character('\n'); + o->write_characters(indent_string.c_str(), current_indent); + o->write_character('}'); } else { - o.put('{'); + o->write_character('{'); // first n-1 elements auto i = val.m_value.object->cbegin(); for (size_t cnt = 0; cnt < val.m_value.object->size() - 1; ++cnt, ++i) { - o.put('\"'); + o->write_character('\"'); dump_escaped(i->first); - o.write("\":", 2); + o->write_characters("\":", 2); dump(i->second, false, indent_step, current_indent); - o.put(','); + o->write_character(','); } // last element assert(i != val.m_value.object->cend()); - o.put('\"'); + o->write_character('\"'); dump_escaped(i->first); - o.write("\":", 2); + o->write_characters("\":", 2); dump(i->second, false, indent_step, current_indent); - o.put('}'); + o->write_character('}'); } return; @@ -6681,13 +6779,13 @@ class basic_json { if (val.m_value.array->empty()) { - o.write("[]", 2); + o->write_characters("[]", 2); return; } if (pretty_print) { - o.write("[\n", 2); + o->write_characters("[\n", 2); // variable to hold indentation for recursive calls const auto new_indent = current_indent + indent_step; @@ -6699,36 +6797,36 @@ class basic_json // first n-1 elements for (auto i = val.m_value.array->cbegin(); i != val.m_value.array->cend() - 1; ++i) { - o.write(indent_string.c_str(), static_cast(new_indent)); + o->write_characters(indent_string.c_str(), new_indent); dump(*i, true, indent_step, new_indent); - o.write(",\n", 2); + o->write_characters(",\n", 2); } // last element assert(not val.m_value.array->empty()); - o.write(indent_string.c_str(), static_cast(new_indent)); + o->write_characters(indent_string.c_str(), new_indent); dump(val.m_value.array->back(), true, indent_step, new_indent); - o.put('\n'); - o.write(indent_string.c_str(), static_cast(current_indent)); - o.put(']'); + o->write_character('\n'); + o->write_characters(indent_string.c_str(), current_indent); + o->write_character(']'); } else { - o.put('['); + o->write_character('['); // first n-1 elements for (auto i = val.m_value.array->cbegin(); i != val.m_value.array->cend() - 1; ++i) { dump(*i, false, indent_step, current_indent); - o.put(','); + o->write_character(','); } // last element assert(not val.m_value.array->empty()); dump(val.m_value.array->back(), false, indent_step, current_indent); - o.put(']'); + o->write_character(']'); } return; @@ -6736,9 +6834,9 @@ class basic_json case value_t::string: { - o.put('\"'); + o->write_character('\"'); dump_escaped(*val.m_value.string); - o.put('\"'); + o->write_character('\"'); return; } @@ -6746,11 +6844,11 @@ class basic_json { if (val.m_value.boolean) { - o.write("true", 4); + o->write_characters("true", 4); } else { - o.write("false", 5); + o->write_characters("false", 5); } return; } @@ -6775,13 +6873,13 @@ class basic_json case value_t::discarded: { - o.write("", 11); + o->write_characters("", 11); return; } case value_t::null: { - o.write("null", 4); + o->write_characters("null", 4); return; } } @@ -6872,7 +6970,7 @@ class basic_json const auto space = extra_space(s); if (space == 0) { - o.write(s.c_str(), static_cast(s.size())); + o->write_characters(s.c_str(), s.size()); return; } @@ -6998,7 +7096,7 @@ class basic_json } assert(pos == s.size() + space); - o.write(result.c_str(), static_cast(result.size())); + o->write_characters(result.c_str(), result.size()); } /*! @@ -7018,7 +7116,7 @@ class basic_json // special case for "0" if (x == 0) { - o.put('0'); + o->write_character('0'); return; } @@ -7044,7 +7142,7 @@ class basic_json } std::reverse(number_buffer.begin(), number_buffer.begin() + i); - o.write(number_buffer.data(), static_cast(i)); + o->write_characters(number_buffer.data(), i); } /*! @@ -7060,7 +7158,7 @@ class basic_json // NaN / inf if (not std::isfinite(x) or std::isnan(x)) { - o.write("null", 4); + o->write_characters("null", 4); return; } @@ -7069,11 +7167,11 @@ class basic_json { if (std::signbit(x)) { - o.write("-0.0", 4); + o->write_characters("-0.0", 4); } else { - o.write("0.0", 3); + o->write_characters("0.0", 3); } return; } @@ -7114,7 +7212,7 @@ class basic_json } } - o.write(number_buffer.data(), static_cast(len)); + o->write_characters(number_buffer.data(), static_cast(len)); // determine if need to append ".0" const bool value_is_int_like = std::none_of(number_buffer.begin(), @@ -7126,13 +7224,13 @@ class basic_json if (value_is_int_like) { - o.write(".0", 2); + o->write_characters(".0", 2); } } private: /// the output of the serializer - std::ostream& o; + output_adapter_t o = nullptr; /// a (hopefully) large enough character buffer std::array number_buffer{{}}; @@ -7181,7 +7279,7 @@ class basic_json o.width(0); // do the actual serialization - serializer s(o); + serializer s(output_adapter::create(o)); s.dump(j, pretty_print, static_cast(indentation)); return o; } @@ -8778,46 +8876,6 @@ class basic_json const char* start; }; - ///////////////////// - // output adapters // - ///////////////////// - - class output_adapter - { - public: - virtual void write_character(uint8_t c) = 0; - virtual void write_characters(const uint8_t* s, size_t length) = 0; - virtual ~output_adapter() {} - - static std::shared_ptr create(std::vector& vec) - { - return std::shared_ptr(new output_vector_adapter(vec)); - } - }; - - using output_adapter_t = std::shared_ptr; - - class output_vector_adapter : public output_adapter - { - public: - output_vector_adapter(std::vector& vec) - : v(vec) - {} - - void write_character(uint8_t c) override - { - v.push_back(c); - } - - void write_characters(const uint8_t* s, size_t length) override - { - std::copy(s, s + length, std::back_inserter(v)); - } - - private: - std::vector& v; - }; - ////////////////////////////////////////// // binary serialization/deserialization // ////////////////////////////////////////// @@ -9872,7 +9930,7 @@ class basic_json : is_little_endian(little_endianess()) {} - explicit binary_writer(output_adapter_t adapter) + explicit binary_writer(output_adapter_t adapter) : is_little_endian(little_endianess()), oa(adapter) {} @@ -10379,7 +10437,7 @@ class basic_json const bool is_little_endian = true; /// the output - output_adapter_t oa = nullptr; + output_adapter_t oa = nullptr; }; public: @@ -10468,7 +10526,7 @@ class basic_json static std::vector to_cbor(const basic_json& j) { std::vector result; - binary_writer bw(output_adapter::create(result)); + binary_writer bw(output_adapter::create(result)); bw.write_cbor(j); return result; } @@ -10550,7 +10608,7 @@ class basic_json static std::vector to_msgpack(const basic_json& j) { std::vector result; - binary_writer bw(output_adapter::create(result)); + binary_writer bw(output_adapter::create(result)); bw.write_msgpack(j); return result; } diff --git a/test/src/unit-convenience.cpp b/test/src/unit-convenience.cpp index 33556311..5e16962d 100644 --- a/test/src/unit-convenience.cpp +++ b/test/src/unit-convenience.cpp @@ -53,7 +53,7 @@ TEST_CASE("convenience functions") const char* escaped) { std::stringstream ss; - json::serializer s(ss); + json::serializer s(json::output_adapter::create(ss)); s.dump_escaped(original); CHECK(ss.str() == escaped); }; From db9bf953f310014835e3bf5108877e1c3d2b25f7 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sat, 15 Apr 2017 10:40:10 +0200 Subject: [PATCH 167/530] :hammer: improved diagnostic output --- src/json.hpp | 51 +++++++++++++++++++++------------- test/src/unit-class_parser.cpp | 16 +++++------ 2 files changed, 39 insertions(+), 28 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index f7fbfd71..82f69abe 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -11127,6 +11127,13 @@ class basic_json return codepoint; } + static std::string codepoint_to_string(int codepoint) + { + std::stringstream ss; + ss << "U+" << std::setw(4) << std::uppercase << std::setfill('0') << std::hex << codepoint; + return ss.str(); + } + token_type scan_string() { // reset yytext (ignore opening quote) @@ -11237,13 +11244,13 @@ class basic_json } else { - error_message = "invalid string: invalid low surrogate"; + error_message = "invalid string: surrogate " + codepoint_to_string(codepoint1) + " must be followed by U+DC00..U+DFFF instead of " + codepoint_to_string(codepoint2); return token_type::parse_error; } } else { - error_message = "invalid string: missing low surrogate"; + error_message = "invalid string: surrogate " + codepoint_to_string(codepoint1) + " must be followed by U+DC00..U+DFFF"; return token_type::parse_error; } } @@ -11251,7 +11258,7 @@ class basic_json { if (JSON_UNLIKELY(0xDC00 <= codepoint1 and codepoint1 <= 0xDFFF)) { - error_message = "invalid string: missing high surrogate"; + error_message = "invalid string: surrogate " + codepoint_to_string(codepoint1) + " must follow U+D800..U+DBFF"; return token_type::parse_error; } @@ -11336,7 +11343,7 @@ class basic_json case 0x1e: case 0x1f: { - error_message = "invalid string: control characters (U+0000 through U+001f) must be escaped"; + error_message = "invalid string: control character " + codepoint_to_string(current) + " must be escaped"; return token_type::parse_error; } @@ -11480,7 +11487,7 @@ class basic_json continue; } - error_message = "invalid string: not well-formed UTF-8 byte"; + error_message = "invalid string: ill-formed UTF-8 byte"; return token_type::parse_error; } @@ -11500,7 +11507,7 @@ class basic_json } } - error_message = "invalid string: not well-formed UTF-8 byte"; + error_message = "invalid string: ill-formed UTF-8 byte"; return token_type::parse_error; } @@ -11534,7 +11541,7 @@ class basic_json } } - error_message = "invalid string: not well-formed UTF-8 byte"; + error_message = "invalid string: ill-formed UTF-8 byte"; return token_type::parse_error; } @@ -11554,7 +11561,7 @@ class basic_json } } - error_message = "invalid string: not well-formed UTF-8 byte"; + error_message = "invalid string: ill-formed UTF-8 byte"; return token_type::parse_error; } @@ -11579,7 +11586,7 @@ class basic_json } } - error_message = "invalid string: not well-formed UTF-8 byte"; + error_message = "invalid string: ill-formed UTF-8 byte"; return token_type::parse_error; } @@ -11606,7 +11613,7 @@ class basic_json } } - error_message = "invalid string: not well-formed UTF-8 byte"; + error_message = "invalid string: ill-formed UTF-8 byte"; return token_type::parse_error; } @@ -11631,14 +11638,14 @@ class basic_json } } - error_message = "invalid string: not well-formed UTF-8 byte"; + error_message = "invalid string: ill-formed UTF-8 byte"; return token_type::parse_error; } - // remaining bytes (80..C1 and F5..FF) are not well-formed + // remaining bytes (80..C1 and F5..FF) are ill-formed default: { - error_message = "invalid string: not well-formed UTF-8 byte"; + error_message = "invalid string: ill-formed UTF-8 byte"; return token_type::parse_error; } } @@ -11681,7 +11688,7 @@ class basic_json // be changed if minus sign, decimal point or exponent is read token_type number_type = token_type::value_unsigned; - // state: we just found out we need to scan a number + // state (init): we just found out we need to scan a number switch (current) { case '-': @@ -12001,6 +12008,8 @@ scan_number_done: } } + // this code is reached if we parse a floating-point number or if + // an integer conversion above failed strtof(value_float, yytext.data(), nullptr); return token_type::value_float; } @@ -12064,7 +12073,8 @@ scan_number_done: /// add a character to yytext void add(int c) { - // resize yytext if necessary + // resize yytext if necessary; this condition is deemed unlikely, + // because we start with a 1024-byte buffer if (JSON_UNLIKELY((yylen + 1 > yytext.capacity()))) { yytext.resize(2 * yytext.capacity(), '\0'); @@ -12120,7 +12130,7 @@ scan_number_done: std::string s = ia->read(start_pos, chars_read - start_pos); // escape control characters - std::stringstream ss; + std::string result; for (auto c : s) { if (c == '\0' or c == std::char_traits::eof()) @@ -12131,16 +12141,16 @@ scan_number_done: else if ('\x00' <= c and c <= '\x1f') { // escape control characters - ss << ""; + result += "<" + codepoint_to_string(c) + ">"; } else { // add character as is - ss << c; + result.append(1, c); } } - return ss.str(); + return result; } /// return syntax error message @@ -12204,7 +12214,8 @@ scan_number_done: case '9': return scan_number(); - // end of input + // end of input (the null byte is needed when parsing from + // string literals) case '\0': case std::char_traits::eof(): return token_type::end_of_input; diff --git a/test/src/unit-class_parser.cpp b/test/src/unit-class_parser.cpp index e0fffac4..b631a978 100644 --- a/test/src/unit-class_parser.cpp +++ b/test/src/unit-class_parser.cpp @@ -98,18 +98,18 @@ TEST_CASE("parser class") // error: tab in string CHECK_THROWS_AS(parse_string("\"\t\"").parse(), json::parse_error); CHECK_THROWS_WITH(parse_string("\"\t\"").parse(), - "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: control characters (U+0000 through U+001f) must be escaped; last read '\"'"); + "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: control character U+0009 must be escaped; last read '\"'"); // error: newline in string CHECK_THROWS_AS(parse_string("\"\n\"").parse(), json::parse_error); CHECK_THROWS_AS(parse_string("\"\r\"").parse(), json::parse_error); CHECK_THROWS_WITH(parse_string("\"\n\"").parse(), - "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: control characters (U+0000 through U+001f) must be escaped; last read '\"'"); + "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: control character U+000A must be escaped; last read '\"'"); CHECK_THROWS_WITH(parse_string("\"\r\"").parse(), - "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: control characters (U+0000 through U+001f) must be escaped; last read '\"'"); + "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: control character U+000D must be escaped; last read '\"'"); // error: backspace in string CHECK_THROWS_AS(parse_string("\"\b\"").parse(), json::parse_error); CHECK_THROWS_WITH(parse_string("\"\b\"").parse(), - "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: control characters (U+0000 through U+001f) must be escaped; last read '\"'"); + "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: control character U+0008 must be escaped; last read '\"'"); // improve code coverage CHECK_THROWS_AS(parse_string("\uFF01").parse(), json::parse_error); CHECK_THROWS_AS(parse_string("[-4:1,]").parse(), json::parse_error); @@ -648,17 +648,17 @@ TEST_CASE("parser class") // missing part of a surrogate pair CHECK_THROWS_AS(json::parse("\"\\uD80C\""), json::parse_error); CHECK_THROWS_WITH(json::parse("\"\\uD80C\""), - "[json.exception.parse_error.101] parse error at 8: syntax error - invalid string: missing low surrogate; last read '\"\\uD80C\"'"); + "[json.exception.parse_error.101] parse error at 8: syntax error - invalid string: surrogate U+D80C must be followed by U+DC00..U+DFFF; last read '\"\\uD80C\"'"); // invalid surrogate pair CHECK_THROWS_AS(json::parse("\"\\uD80C\\uD80C\""), json::parse_error); CHECK_THROWS_AS(json::parse("\"\\uD80C\\u0000\""), json::parse_error); CHECK_THROWS_AS(json::parse("\"\\uD80C\\uFFFF\""), json::parse_error); CHECK_THROWS_WITH(json::parse("\"\\uD80C\\uD80C\""), - "[json.exception.parse_error.101] parse error at 13: syntax error - invalid string: invalid low surrogate; last read '\"\\uD80C\\uD80C'"); + "[json.exception.parse_error.101] parse error at 13: syntax error - invalid string: surrogate U+D80C must be followed by U+DC00..U+DFFF instead of U+D80C; last read '\"\\uD80C\\uD80C'"); CHECK_THROWS_WITH(json::parse("\"\\uD80C\\u0000\""), - "[json.exception.parse_error.101] parse error at 13: syntax error - invalid string: invalid low surrogate; last read '\"\\uD80C\\u0000'"); + "[json.exception.parse_error.101] parse error at 13: syntax error - invalid string: surrogate U+D80C must be followed by U+DC00..U+DFFF instead of U+0000; last read '\"\\uD80C\\u0000'"); CHECK_THROWS_WITH(json::parse("\"\\uD80C\\uFFFF\""), - "[json.exception.parse_error.101] parse error at 13: syntax error - invalid string: invalid low surrogate; last read '\"\\uD80C\\uFFFF'"); + "[json.exception.parse_error.101] parse error at 13: syntax error - invalid string: surrogate U+D80C must be followed by U+DC00..U+DFFF instead of U+FFFF; last read '\"\\uD80C\\uFFFF'"); } SECTION("tests found by mutate++") From 9b32f7258421cf747d0984ed28d273530e6f8587 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Fri, 21 Apr 2017 22:07:07 +0200 Subject: [PATCH 168/530] :memo: fixed examples for Wandbox As I learned in https://github.com/melpon/wandbox/issues/209, this library is already installed at Wandbox, so we need to adjust the examples to use `#include "json.hpp"` insteas of `#include `. --- doc/examples/README.cpp | 2 +- doc/examples/README.link | 2 +- doc/examples/array.cpp | 2 +- doc/examples/array.link | 2 +- doc/examples/at__object_t_key_type.cpp | 2 +- doc/examples/at__object_t_key_type.link | 2 +- doc/examples/at__object_t_key_type_const.cpp | 2 +- doc/examples/at__object_t_key_type_const.link | 2 +- doc/examples/at__size_type.cpp | 2 +- doc/examples/at__size_type.link | 2 +- doc/examples/at__size_type_const.cpp | 2 +- doc/examples/at__size_type_const.link | 2 +- doc/examples/at_json_pointer.cpp | 2 +- doc/examples/at_json_pointer.link | 2 +- doc/examples/at_json_pointer_const.cpp | 2 +- doc/examples/at_json_pointer_const.link | 2 +- doc/examples/back.cpp | 2 +- doc/examples/back.link | 2 +- doc/examples/basic_json__CompatibleType.cpp | 2 +- doc/examples/basic_json__CompatibleType.link | 2 +- doc/examples/basic_json__InputIt_InputIt.cpp | 2 +- doc/examples/basic_json__InputIt_InputIt.link | 2 +- doc/examples/basic_json__basic_json.cpp | 2 +- doc/examples/basic_json__basic_json.link | 2 +- doc/examples/basic_json__copyassignment.cpp | 2 +- doc/examples/basic_json__copyassignment.link | 2 +- doc/examples/basic_json__list_init_t.cpp | 2 +- doc/examples/basic_json__list_init_t.link | 2 +- doc/examples/basic_json__moveconstructor.cpp | 2 +- doc/examples/basic_json__moveconstructor.link | 2 +- doc/examples/basic_json__nullptr_t.cpp | 2 +- doc/examples/basic_json__nullptr_t.link | 2 +- doc/examples/basic_json__size_type_basic_json.cpp | 2 +- doc/examples/basic_json__size_type_basic_json.link | 2 +- doc/examples/basic_json__value.cpp | 2 +- doc/examples/basic_json__value.link | 2 +- doc/examples/basic_json__value_ptr.cpp | 2 +- doc/examples/basic_json__value_ptr.link | 2 +- doc/examples/basic_json__value_t.cpp | 2 +- doc/examples/basic_json__value_t.link | 2 +- doc/examples/begin.cpp | 2 +- doc/examples/begin.link | 2 +- doc/examples/cbegin.cpp | 2 +- doc/examples/cbegin.link | 2 +- doc/examples/cend.cpp | 2 +- doc/examples/cend.link | 2 +- doc/examples/clear.cpp | 2 +- doc/examples/clear.link | 2 +- doc/examples/count.cpp | 2 +- doc/examples/count.link | 2 +- doc/examples/crbegin.cpp | 2 +- doc/examples/crbegin.link | 2 +- doc/examples/crend.cpp | 2 +- doc/examples/crend.link | 2 +- doc/examples/diff.cpp | 2 +- doc/examples/diff.link | 2 +- doc/examples/dump.cpp | 2 +- doc/examples/dump.link | 2 +- doc/examples/emplace.cpp | 2 +- doc/examples/emplace.link | 2 +- doc/examples/emplace_back.cpp | 2 +- doc/examples/emplace_back.link | 2 +- doc/examples/empty.cpp | 2 +- doc/examples/empty.link | 2 +- doc/examples/end.cpp | 2 +- doc/examples/end.link | 2 +- doc/examples/erase__IteratorType.cpp | 2 +- doc/examples/erase__IteratorType.link | 2 +- doc/examples/erase__IteratorType_IteratorType.cpp | 2 +- doc/examples/erase__IteratorType_IteratorType.link | 2 +- doc/examples/erase__key_type.cpp | 2 +- doc/examples/erase__key_type.link | 2 +- doc/examples/erase__size_type.cpp | 2 +- doc/examples/erase__size_type.link | 2 +- doc/examples/find__key_type.cpp | 2 +- doc/examples/find__key_type.link | 2 +- doc/examples/flatten.cpp | 2 +- doc/examples/flatten.link | 2 +- doc/examples/from_cbor.cpp | 2 +- doc/examples/from_cbor.link | 2 +- doc/examples/from_msgpack.cpp | 2 +- doc/examples/from_msgpack.link | 2 +- doc/examples/front.cpp | 2 +- doc/examples/front.link | 2 +- doc/examples/get__PointerType.cpp | 2 +- doc/examples/get__PointerType.link | 2 +- doc/examples/get__ValueType_const.cpp | 2 +- doc/examples/get__ValueType_const.link | 2 +- doc/examples/get_ptr.cpp | 2 +- doc/examples/get_ptr.link | 2 +- doc/examples/get_ref.cpp | 2 +- doc/examples/get_ref.link | 2 +- doc/examples/insert.cpp | 2 +- doc/examples/insert.link | 2 +- doc/examples/insert__count.cpp | 2 +- doc/examples/insert__count.link | 2 +- doc/examples/insert__ilist.cpp | 2 +- doc/examples/insert__ilist.link | 2 +- doc/examples/insert__range.cpp | 2 +- doc/examples/insert__range.link | 2 +- doc/examples/insert__range_object.cpp | 2 +- doc/examples/insert__range_object.link | 1 + doc/examples/is_array.cpp | 2 +- doc/examples/is_array.link | 2 +- doc/examples/is_boolean.cpp | 2 +- doc/examples/is_boolean.link | 2 +- doc/examples/is_discarded.cpp | 2 +- doc/examples/is_discarded.link | 2 +- doc/examples/is_null.cpp | 2 +- doc/examples/is_null.link | 2 +- doc/examples/is_number.cpp | 2 +- doc/examples/is_number.link | 2 +- doc/examples/is_number_float.cpp | 2 +- doc/examples/is_number_float.link | 2 +- doc/examples/is_number_integer.cpp | 2 +- doc/examples/is_number_integer.link | 2 +- doc/examples/is_number_unsigned.cpp | 2 +- doc/examples/is_number_unsigned.link | 2 +- doc/examples/is_object.cpp | 2 +- doc/examples/is_object.link | 2 +- doc/examples/is_primitive.cpp | 2 +- doc/examples/is_primitive.link | 2 +- doc/examples/is_string.cpp | 2 +- doc/examples/is_string.link | 2 +- doc/examples/is_structured.cpp | 2 +- doc/examples/is_structured.link | 2 +- doc/examples/json_pointer.cpp | 2 +- doc/examples/json_pointer.link | 2 +- doc/examples/json_pointer__to_string.cpp | 2 +- doc/examples/json_pointer__to_string.link | 2 +- doc/examples/max_size.cpp | 2 +- doc/examples/max_size.link | 2 +- doc/examples/meta.cpp | 2 +- doc/examples/meta.link | 2 +- doc/examples/meta.output | 2 +- doc/examples/object.cpp | 2 +- doc/examples/object.link | 2 +- doc/examples/operator__ValueType.cpp | 2 +- doc/examples/operator__ValueType.link | 2 +- doc/examples/operator__equal.cpp | 2 +- doc/examples/operator__equal.link | 2 +- doc/examples/operator__equal.output | 2 +- doc/examples/operator__equal__nullptr_t.cpp | 2 +- doc/examples/operator__equal__nullptr_t.link | 2 +- doc/examples/operator__greater.cpp | 2 +- doc/examples/operator__greater.link | 2 +- doc/examples/operator__greaterequal.cpp | 2 +- doc/examples/operator__greaterequal.link | 2 +- doc/examples/operator__less.cpp | 2 +- doc/examples/operator__less.link | 2 +- doc/examples/operator__lessequal.cpp | 2 +- doc/examples/operator__lessequal.link | 2 +- doc/examples/operator__notequal.cpp | 2 +- doc/examples/operator__notequal.link | 2 +- doc/examples/operator__notequal.output | 2 +- doc/examples/operator__notequal__nullptr_t.cpp | 2 +- doc/examples/operator__notequal__nullptr_t.link | 2 +- doc/examples/operator__value_t.cpp | 2 +- doc/examples/operator__value_t.link | 2 +- doc/examples/operator_deserialize.cpp | 2 +- doc/examples/operator_deserialize.link | 2 +- doc/examples/operator_serialize.cpp | 2 +- doc/examples/operator_serialize.link | 2 +- doc/examples/operatorarray__key_type.cpp | 2 +- doc/examples/operatorarray__key_type.link | 2 +- doc/examples/operatorarray__key_type_const.cpp | 2 +- doc/examples/operatorarray__key_type_const.link | 2 +- doc/examples/operatorarray__size_type.cpp | 2 +- doc/examples/operatorarray__size_type.link | 2 +- doc/examples/operatorarray__size_type_const.cpp | 2 +- doc/examples/operatorarray__size_type_const.link | 2 +- doc/examples/operatorjson_pointer.cpp | 2 +- doc/examples/operatorjson_pointer.link | 2 +- doc/examples/operatorjson_pointer_const.cpp | 2 +- doc/examples/operatorjson_pointer_const.link | 2 +- doc/examples/parse__array__parser_callback_t.cpp | 2 +- doc/examples/parse__array__parser_callback_t.link | 2 +- doc/examples/parse__contiguouscontainer__parser_callback_t.cpp | 2 +- doc/examples/parse__contiguouscontainer__parser_callback_t.link | 2 +- doc/examples/parse__istream__parser_callback_t.cpp | 2 +- doc/examples/parse__istream__parser_callback_t.link | 2 +- doc/examples/parse__iteratortype__parser_callback_t.cpp | 2 +- doc/examples/parse__iteratortype__parser_callback_t.link | 2 +- doc/examples/parse__string__parser_callback_t.cpp | 2 +- doc/examples/parse__string__parser_callback_t.link | 2 +- doc/examples/patch.cpp | 2 +- doc/examples/patch.link | 2 +- doc/examples/push_back.cpp | 2 +- doc/examples/push_back.link | 2 +- doc/examples/push_back__initializer_list.cpp | 2 +- doc/examples/push_back__initializer_list.link | 2 +- doc/examples/push_back__object_t__value.cpp | 2 +- doc/examples/push_back__object_t__value.link | 2 +- doc/examples/rbegin.cpp | 2 +- doc/examples/rbegin.link | 2 +- doc/examples/rend.cpp | 2 +- doc/examples/rend.link | 2 +- doc/examples/size.cpp | 2 +- doc/examples/size.link | 2 +- doc/examples/swap__array_t.cpp | 2 +- doc/examples/swap__array_t.link | 2 +- doc/examples/swap__object_t.cpp | 2 +- doc/examples/swap__object_t.link | 2 +- doc/examples/swap__reference.cpp | 2 +- doc/examples/swap__reference.link | 2 +- doc/examples/swap__string_t.cpp | 2 +- doc/examples/swap__string_t.link | 2 +- doc/examples/to_cbor.cpp | 2 +- doc/examples/to_cbor.link | 2 +- doc/examples/to_msgpack.cpp | 2 +- doc/examples/to_msgpack.link | 2 +- doc/examples/type.cpp | 2 +- doc/examples/type.link | 2 +- doc/examples/type_name.cpp | 2 +- doc/examples/type_name.link | 2 +- doc/examples/unflatten.cpp | 2 +- doc/examples/unflatten.link | 2 +- 217 files changed, 217 insertions(+), 216 deletions(-) create mode 100644 doc/examples/insert__range_object.link diff --git a/doc/examples/README.cpp b/doc/examples/README.cpp index d8cd4d67..a93ee60a 100644 --- a/doc/examples/README.cpp +++ b/doc/examples/README.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" #include // for std::setw using json = nlohmann::json; diff --git a/doc/examples/README.link b/doc/examples/README.link index 9b154c4a..8758bf05 100644 --- a/doc/examples/README.link +++ b/doc/examples/README.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/array.cpp b/doc/examples/array.cpp index 3ec0a133..d3cc4777 100644 --- a/doc/examples/array.cpp +++ b/doc/examples/array.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/array.link b/doc/examples/array.link index cafca79c..a7e40831 100644 --- a/doc/examples/array.link +++ b/doc/examples/array.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/at__object_t_key_type.cpp b/doc/examples/at__object_t_key_type.cpp index 7797418f..611ca226 100644 --- a/doc/examples/at__object_t_key_type.cpp +++ b/doc/examples/at__object_t_key_type.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/at__object_t_key_type.link b/doc/examples/at__object_t_key_type.link index 8874783b..cff20bea 100644 --- a/doc/examples/at__object_t_key_type.link +++ b/doc/examples/at__object_t_key_type.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/at__object_t_key_type_const.cpp b/doc/examples/at__object_t_key_type_const.cpp index f6042404..eae7da96 100644 --- a/doc/examples/at__object_t_key_type_const.cpp +++ b/doc/examples/at__object_t_key_type_const.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/at__object_t_key_type_const.link b/doc/examples/at__object_t_key_type_const.link index cd8594e6..43bb6c87 100644 --- a/doc/examples/at__object_t_key_type_const.link +++ b/doc/examples/at__object_t_key_type_const.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/at__size_type.cpp b/doc/examples/at__size_type.cpp index 28def1dd..17381738 100644 --- a/doc/examples/at__size_type.cpp +++ b/doc/examples/at__size_type.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/at__size_type.link b/doc/examples/at__size_type.link index cba0fa00..fea3bcd5 100644 --- a/doc/examples/at__size_type.link +++ b/doc/examples/at__size_type.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/at__size_type_const.cpp b/doc/examples/at__size_type_const.cpp index 213d22fd..3f67d378 100644 --- a/doc/examples/at__size_type_const.cpp +++ b/doc/examples/at__size_type_const.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/at__size_type_const.link b/doc/examples/at__size_type_const.link index ce3647ac..70dd9151 100644 --- a/doc/examples/at__size_type_const.link +++ b/doc/examples/at__size_type_const.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/at_json_pointer.cpp b/doc/examples/at_json_pointer.cpp index 3ef91282..2cb62e2d 100644 --- a/doc/examples/at_json_pointer.cpp +++ b/doc/examples/at_json_pointer.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/at_json_pointer.link b/doc/examples/at_json_pointer.link index c8563ec2..2e6a3e70 100644 --- a/doc/examples/at_json_pointer.link +++ b/doc/examples/at_json_pointer.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/at_json_pointer_const.cpp b/doc/examples/at_json_pointer_const.cpp index a1d065f2..3b09f65d 100644 --- a/doc/examples/at_json_pointer_const.cpp +++ b/doc/examples/at_json_pointer_const.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/at_json_pointer_const.link b/doc/examples/at_json_pointer_const.link index f421faf4..1ea3b078 100644 --- a/doc/examples/at_json_pointer_const.link +++ b/doc/examples/at_json_pointer_const.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/back.cpp b/doc/examples/back.cpp index 45f9483c..f4dcced3 100644 --- a/doc/examples/back.cpp +++ b/doc/examples/back.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/back.link b/doc/examples/back.link index 15b1102b..8d757cde 100644 --- a/doc/examples/back.link +++ b/doc/examples/back.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/basic_json__CompatibleType.cpp b/doc/examples/basic_json__CompatibleType.cpp index ff564a72..85dbccd0 100644 --- a/doc/examples/basic_json__CompatibleType.cpp +++ b/doc/examples/basic_json__CompatibleType.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" #include #include #include diff --git a/doc/examples/basic_json__CompatibleType.link b/doc/examples/basic_json__CompatibleType.link index a78f01bb..3ced745b 100644 --- a/doc/examples/basic_json__CompatibleType.link +++ b/doc/examples/basic_json__CompatibleType.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/basic_json__InputIt_InputIt.cpp b/doc/examples/basic_json__InputIt_InputIt.cpp index ce07e07b..60043ff3 100644 --- a/doc/examples/basic_json__InputIt_InputIt.cpp +++ b/doc/examples/basic_json__InputIt_InputIt.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/basic_json__InputIt_InputIt.link b/doc/examples/basic_json__InputIt_InputIt.link index d360496a..c7c09d38 100644 --- a/doc/examples/basic_json__InputIt_InputIt.link +++ b/doc/examples/basic_json__InputIt_InputIt.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/basic_json__basic_json.cpp b/doc/examples/basic_json__basic_json.cpp index 28f1fd07..c218a8e5 100644 --- a/doc/examples/basic_json__basic_json.cpp +++ b/doc/examples/basic_json__basic_json.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/basic_json__basic_json.link b/doc/examples/basic_json__basic_json.link index 78fcbbe5..e8ac682d 100644 --- a/doc/examples/basic_json__basic_json.link +++ b/doc/examples/basic_json__basic_json.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/basic_json__copyassignment.cpp b/doc/examples/basic_json__copyassignment.cpp index 9e9c2ecc..b44e5993 100644 --- a/doc/examples/basic_json__copyassignment.cpp +++ b/doc/examples/basic_json__copyassignment.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/basic_json__copyassignment.link b/doc/examples/basic_json__copyassignment.link index efcac306..1bff32a7 100644 --- a/doc/examples/basic_json__copyassignment.link +++ b/doc/examples/basic_json__copyassignment.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/basic_json__list_init_t.cpp b/doc/examples/basic_json__list_init_t.cpp index 8734dff1..5d77dbbc 100644 --- a/doc/examples/basic_json__list_init_t.cpp +++ b/doc/examples/basic_json__list_init_t.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/basic_json__list_init_t.link b/doc/examples/basic_json__list_init_t.link index bcdf6689..4419ca9a 100644 --- a/doc/examples/basic_json__list_init_t.link +++ b/doc/examples/basic_json__list_init_t.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/basic_json__moveconstructor.cpp b/doc/examples/basic_json__moveconstructor.cpp index e470c95b..ecfdb603 100644 --- a/doc/examples/basic_json__moveconstructor.cpp +++ b/doc/examples/basic_json__moveconstructor.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/basic_json__moveconstructor.link b/doc/examples/basic_json__moveconstructor.link index 9580d02c..7bad9910 100644 --- a/doc/examples/basic_json__moveconstructor.link +++ b/doc/examples/basic_json__moveconstructor.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/basic_json__nullptr_t.cpp b/doc/examples/basic_json__nullptr_t.cpp index d0156d53..041a3269 100644 --- a/doc/examples/basic_json__nullptr_t.cpp +++ b/doc/examples/basic_json__nullptr_t.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/basic_json__nullptr_t.link b/doc/examples/basic_json__nullptr_t.link index f911caa5..143d20a2 100644 --- a/doc/examples/basic_json__nullptr_t.link +++ b/doc/examples/basic_json__nullptr_t.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/basic_json__size_type_basic_json.cpp b/doc/examples/basic_json__size_type_basic_json.cpp index 15f8c0d5..1add4925 100644 --- a/doc/examples/basic_json__size_type_basic_json.cpp +++ b/doc/examples/basic_json__size_type_basic_json.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/basic_json__size_type_basic_json.link b/doc/examples/basic_json__size_type_basic_json.link index 78305dcf..e699f9ad 100644 --- a/doc/examples/basic_json__size_type_basic_json.link +++ b/doc/examples/basic_json__size_type_basic_json.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/basic_json__value.cpp b/doc/examples/basic_json__value.cpp index dfb91d84..fced71c3 100644 --- a/doc/examples/basic_json__value.cpp +++ b/doc/examples/basic_json__value.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/basic_json__value.link b/doc/examples/basic_json__value.link index 27e1ddeb..3dc8a407 100644 --- a/doc/examples/basic_json__value.link +++ b/doc/examples/basic_json__value.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/basic_json__value_ptr.cpp b/doc/examples/basic_json__value_ptr.cpp index f45fb8b0..262fbbf6 100644 --- a/doc/examples/basic_json__value_ptr.cpp +++ b/doc/examples/basic_json__value_ptr.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/basic_json__value_ptr.link b/doc/examples/basic_json__value_ptr.link index 2f8fc83c..482c75f0 100644 --- a/doc/examples/basic_json__value_ptr.link +++ b/doc/examples/basic_json__value_ptr.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/basic_json__value_t.cpp b/doc/examples/basic_json__value_t.cpp index faf5dfae..20cd3b4d 100644 --- a/doc/examples/basic_json__value_t.cpp +++ b/doc/examples/basic_json__value_t.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/basic_json__value_t.link b/doc/examples/basic_json__value_t.link index e7e744c1..8e999a8d 100644 --- a/doc/examples/basic_json__value_t.link +++ b/doc/examples/basic_json__value_t.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/begin.cpp b/doc/examples/begin.cpp index 46c17218..c2bb43ae 100644 --- a/doc/examples/begin.cpp +++ b/doc/examples/begin.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/begin.link b/doc/examples/begin.link index 64eac702..da79b631 100644 --- a/doc/examples/begin.link +++ b/doc/examples/begin.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/cbegin.cpp b/doc/examples/cbegin.cpp index fe45e49b..f4353ef0 100644 --- a/doc/examples/cbegin.cpp +++ b/doc/examples/cbegin.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/cbegin.link b/doc/examples/cbegin.link index c298c653..dee6deb2 100644 --- a/doc/examples/cbegin.link +++ b/doc/examples/cbegin.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/cend.cpp b/doc/examples/cend.cpp index a140fb0b..08914124 100644 --- a/doc/examples/cend.cpp +++ b/doc/examples/cend.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/cend.link b/doc/examples/cend.link index 61a3203d..6dfc2277 100644 --- a/doc/examples/cend.link +++ b/doc/examples/cend.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/clear.cpp b/doc/examples/clear.cpp index 60016753..055211cf 100644 --- a/doc/examples/clear.cpp +++ b/doc/examples/clear.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/clear.link b/doc/examples/clear.link index 9ad0a14f..91fa1f25 100644 --- a/doc/examples/clear.link +++ b/doc/examples/clear.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/count.cpp b/doc/examples/count.cpp index b97f71da..b3f10d06 100644 --- a/doc/examples/count.cpp +++ b/doc/examples/count.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/count.link b/doc/examples/count.link index 0893d922..6167a673 100644 --- a/doc/examples/count.link +++ b/doc/examples/count.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/crbegin.cpp b/doc/examples/crbegin.cpp index 4ea71d97..df0868ab 100644 --- a/doc/examples/crbegin.cpp +++ b/doc/examples/crbegin.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/crbegin.link b/doc/examples/crbegin.link index 1188be1e..49072710 100644 --- a/doc/examples/crbegin.link +++ b/doc/examples/crbegin.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/crend.cpp b/doc/examples/crend.cpp index 079eeea6..20c2f82a 100644 --- a/doc/examples/crend.cpp +++ b/doc/examples/crend.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/crend.link b/doc/examples/crend.link index 4bdfef3a..7c244144 100644 --- a/doc/examples/crend.link +++ b/doc/examples/crend.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/diff.cpp b/doc/examples/diff.cpp index a377fc81..982b8f9c 100644 --- a/doc/examples/diff.cpp +++ b/doc/examples/diff.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" #include // for std::setw using json = nlohmann::json; diff --git a/doc/examples/diff.link b/doc/examples/diff.link index 459ecfcc..d0bed27a 100644 --- a/doc/examples/diff.link +++ b/doc/examples/diff.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/dump.cpp b/doc/examples/dump.cpp index 3145d453..8190ae49 100644 --- a/doc/examples/dump.cpp +++ b/doc/examples/dump.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/dump.link b/doc/examples/dump.link index 5cf3e634..3ced4448 100644 --- a/doc/examples/dump.link +++ b/doc/examples/dump.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/emplace.cpp b/doc/examples/emplace.cpp index c3b3c3e3..2af633eb 100644 --- a/doc/examples/emplace.cpp +++ b/doc/examples/emplace.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/emplace.link b/doc/examples/emplace.link index a9366c32..b689c476 100644 --- a/doc/examples/emplace.link +++ b/doc/examples/emplace.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/emplace_back.cpp b/doc/examples/emplace_back.cpp index 4e9ec89a..f76e599f 100644 --- a/doc/examples/emplace_back.cpp +++ b/doc/examples/emplace_back.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/emplace_back.link b/doc/examples/emplace_back.link index 4363e4c7..824c95df 100644 --- a/doc/examples/emplace_back.link +++ b/doc/examples/emplace_back.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/empty.cpp b/doc/examples/empty.cpp index 2edda6dc..a4c71b52 100644 --- a/doc/examples/empty.cpp +++ b/doc/examples/empty.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/empty.link b/doc/examples/empty.link index 12aa9678..c16e0296 100644 --- a/doc/examples/empty.link +++ b/doc/examples/empty.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/end.cpp b/doc/examples/end.cpp index 848623a4..5eeec81d 100644 --- a/doc/examples/end.cpp +++ b/doc/examples/end.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/end.link b/doc/examples/end.link index 462a463f..4ced3b90 100644 --- a/doc/examples/end.link +++ b/doc/examples/end.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/erase__IteratorType.cpp b/doc/examples/erase__IteratorType.cpp index 5743c327..2ec6204f 100644 --- a/doc/examples/erase__IteratorType.cpp +++ b/doc/examples/erase__IteratorType.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/erase__IteratorType.link b/doc/examples/erase__IteratorType.link index 854a9be5..50571c91 100644 --- a/doc/examples/erase__IteratorType.link +++ b/doc/examples/erase__IteratorType.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/erase__IteratorType_IteratorType.cpp b/doc/examples/erase__IteratorType_IteratorType.cpp index 9250c3ca..ef3f070c 100644 --- a/doc/examples/erase__IteratorType_IteratorType.cpp +++ b/doc/examples/erase__IteratorType_IteratorType.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/erase__IteratorType_IteratorType.link b/doc/examples/erase__IteratorType_IteratorType.link index 26e0b903..fb7fca42 100644 --- a/doc/examples/erase__IteratorType_IteratorType.link +++ b/doc/examples/erase__IteratorType_IteratorType.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/erase__key_type.cpp b/doc/examples/erase__key_type.cpp index ca0962aa..e035c23b 100644 --- a/doc/examples/erase__key_type.cpp +++ b/doc/examples/erase__key_type.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/erase__key_type.link b/doc/examples/erase__key_type.link index e4c2fba1..8a1579c9 100644 --- a/doc/examples/erase__key_type.link +++ b/doc/examples/erase__key_type.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/erase__size_type.cpp b/doc/examples/erase__size_type.cpp index 7df3d062..68bbab64 100644 --- a/doc/examples/erase__size_type.cpp +++ b/doc/examples/erase__size_type.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/erase__size_type.link b/doc/examples/erase__size_type.link index c63abe52..cd20c392 100644 --- a/doc/examples/erase__size_type.link +++ b/doc/examples/erase__size_type.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/find__key_type.cpp b/doc/examples/find__key_type.cpp index 79eed509..cb4a51c7 100644 --- a/doc/examples/find__key_type.cpp +++ b/doc/examples/find__key_type.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/find__key_type.link b/doc/examples/find__key_type.link index cd3d4fd2..bf003fb1 100644 --- a/doc/examples/find__key_type.link +++ b/doc/examples/find__key_type.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/flatten.cpp b/doc/examples/flatten.cpp index e7b7ee87..b0cf9b6c 100644 --- a/doc/examples/flatten.cpp +++ b/doc/examples/flatten.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" #include // for std::setw using json = nlohmann::json; diff --git a/doc/examples/flatten.link b/doc/examples/flatten.link index ab2128fa..b5285f3c 100644 --- a/doc/examples/flatten.link +++ b/doc/examples/flatten.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/from_cbor.cpp b/doc/examples/from_cbor.cpp index fc44d4a5..5d78be8f 100644 --- a/doc/examples/from_cbor.cpp +++ b/doc/examples/from_cbor.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" #include // for std::setw using json = nlohmann::json; diff --git a/doc/examples/from_cbor.link b/doc/examples/from_cbor.link index c621bd2d..15a2bf40 100644 --- a/doc/examples/from_cbor.link +++ b/doc/examples/from_cbor.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/from_msgpack.cpp b/doc/examples/from_msgpack.cpp index 7c4202c9..a16c1bdd 100644 --- a/doc/examples/from_msgpack.cpp +++ b/doc/examples/from_msgpack.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" #include // for std::setw using json = nlohmann::json; diff --git a/doc/examples/from_msgpack.link b/doc/examples/from_msgpack.link index 22bac001..a4818bd7 100644 --- a/doc/examples/from_msgpack.link +++ b/doc/examples/from_msgpack.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/front.cpp b/doc/examples/front.cpp index 7c5a2ae9..e9f09d0c 100644 --- a/doc/examples/front.cpp +++ b/doc/examples/front.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/front.link b/doc/examples/front.link index d69c7f90..e3f57b3d 100644 --- a/doc/examples/front.link +++ b/doc/examples/front.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/get__PointerType.cpp b/doc/examples/get__PointerType.cpp index 7c15c5aa..613305f9 100644 --- a/doc/examples/get__PointerType.cpp +++ b/doc/examples/get__PointerType.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/get__PointerType.link b/doc/examples/get__PointerType.link index 68a46b57..5c0713d5 100644 --- a/doc/examples/get__PointerType.link +++ b/doc/examples/get__PointerType.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/get__ValueType_const.cpp b/doc/examples/get__ValueType_const.cpp index 40a96184..3fdd2834 100644 --- a/doc/examples/get__ValueType_const.cpp +++ b/doc/examples/get__ValueType_const.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" #include using json = nlohmann::json; diff --git a/doc/examples/get__ValueType_const.link b/doc/examples/get__ValueType_const.link index 95862c67..d58c1789 100644 --- a/doc/examples/get__ValueType_const.link +++ b/doc/examples/get__ValueType_const.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/get_ptr.cpp b/doc/examples/get_ptr.cpp index 2c319dc4..54288450 100644 --- a/doc/examples/get_ptr.cpp +++ b/doc/examples/get_ptr.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/get_ptr.link b/doc/examples/get_ptr.link index 13ed7d81..2b158c8f 100644 --- a/doc/examples/get_ptr.link +++ b/doc/examples/get_ptr.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/get_ref.cpp b/doc/examples/get_ref.cpp index a152d347..c07d8488 100644 --- a/doc/examples/get_ref.cpp +++ b/doc/examples/get_ref.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/get_ref.link b/doc/examples/get_ref.link index c5bd9c03..1181b7c6 100644 --- a/doc/examples/get_ref.link +++ b/doc/examples/get_ref.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/insert.cpp b/doc/examples/insert.cpp index 967f443e..99879566 100644 --- a/doc/examples/insert.cpp +++ b/doc/examples/insert.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/insert.link b/doc/examples/insert.link index 9b0a1d90..f3f4e4bc 100644 --- a/doc/examples/insert.link +++ b/doc/examples/insert.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/insert__count.cpp b/doc/examples/insert__count.cpp index 258515e6..11460e51 100644 --- a/doc/examples/insert__count.cpp +++ b/doc/examples/insert__count.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/insert__count.link b/doc/examples/insert__count.link index f6a6441d..103795f5 100644 --- a/doc/examples/insert__count.link +++ b/doc/examples/insert__count.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/insert__ilist.cpp b/doc/examples/insert__ilist.cpp index c0138d2a..fa9e8612 100644 --- a/doc/examples/insert__ilist.cpp +++ b/doc/examples/insert__ilist.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/insert__ilist.link b/doc/examples/insert__ilist.link index eba2b560..13805fbf 100644 --- a/doc/examples/insert__ilist.link +++ b/doc/examples/insert__ilist.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/insert__range.cpp b/doc/examples/insert__range.cpp index b5d0001b..8e8a5977 100644 --- a/doc/examples/insert__range.cpp +++ b/doc/examples/insert__range.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/insert__range.link b/doc/examples/insert__range.link index 3a802b57..30440793 100644 --- a/doc/examples/insert__range.link +++ b/doc/examples/insert__range.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/insert__range_object.cpp b/doc/examples/insert__range_object.cpp index 4cd58938..3f266b22 100644 --- a/doc/examples/insert__range_object.cpp +++ b/doc/examples/insert__range_object.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/insert__range_object.link b/doc/examples/insert__range_object.link new file mode 100644 index 00000000..f6eb481a --- /dev/null +++ b/doc/examples/insert__range_object.link @@ -0,0 +1 @@ +online \ No newline at end of file diff --git a/doc/examples/is_array.cpp b/doc/examples/is_array.cpp index 11a63115..7a5d20bf 100644 --- a/doc/examples/is_array.cpp +++ b/doc/examples/is_array.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/is_array.link b/doc/examples/is_array.link index ccd72750..ad18000c 100644 --- a/doc/examples/is_array.link +++ b/doc/examples/is_array.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/is_boolean.cpp b/doc/examples/is_boolean.cpp index 9efcf694..d3a0e178 100644 --- a/doc/examples/is_boolean.cpp +++ b/doc/examples/is_boolean.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/is_boolean.link b/doc/examples/is_boolean.link index 5ebc9acb..bb89c4a0 100644 --- a/doc/examples/is_boolean.link +++ b/doc/examples/is_boolean.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/is_discarded.cpp b/doc/examples/is_discarded.cpp index a1d42a7a..f4953bf9 100644 --- a/doc/examples/is_discarded.cpp +++ b/doc/examples/is_discarded.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/is_discarded.link b/doc/examples/is_discarded.link index d2172392..407667aa 100644 --- a/doc/examples/is_discarded.link +++ b/doc/examples/is_discarded.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/is_null.cpp b/doc/examples/is_null.cpp index 0d69c05c..9dd0528c 100644 --- a/doc/examples/is_null.cpp +++ b/doc/examples/is_null.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/is_null.link b/doc/examples/is_null.link index d40569d2..6ead545c 100644 --- a/doc/examples/is_null.link +++ b/doc/examples/is_null.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/is_number.cpp b/doc/examples/is_number.cpp index fd2a99b1..a0baff32 100644 --- a/doc/examples/is_number.cpp +++ b/doc/examples/is_number.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/is_number.link b/doc/examples/is_number.link index 089d7893..7e291e79 100644 --- a/doc/examples/is_number.link +++ b/doc/examples/is_number.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/is_number_float.cpp b/doc/examples/is_number_float.cpp index 92bcfa10..a67d9271 100644 --- a/doc/examples/is_number_float.cpp +++ b/doc/examples/is_number_float.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/is_number_float.link b/doc/examples/is_number_float.link index 3fa73533..3f56570a 100644 --- a/doc/examples/is_number_float.link +++ b/doc/examples/is_number_float.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/is_number_integer.cpp b/doc/examples/is_number_integer.cpp index ecac7cd6..b8f772ef 100644 --- a/doc/examples/is_number_integer.cpp +++ b/doc/examples/is_number_integer.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/is_number_integer.link b/doc/examples/is_number_integer.link index 474cded4..692d0208 100644 --- a/doc/examples/is_number_integer.link +++ b/doc/examples/is_number_integer.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/is_number_unsigned.cpp b/doc/examples/is_number_unsigned.cpp index ad457591..7cd04a31 100644 --- a/doc/examples/is_number_unsigned.cpp +++ b/doc/examples/is_number_unsigned.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/is_number_unsigned.link b/doc/examples/is_number_unsigned.link index 0849b8e5..930ec498 100644 --- a/doc/examples/is_number_unsigned.link +++ b/doc/examples/is_number_unsigned.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/is_object.cpp b/doc/examples/is_object.cpp index 56db66e5..1fe66ec2 100644 --- a/doc/examples/is_object.cpp +++ b/doc/examples/is_object.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/is_object.link b/doc/examples/is_object.link index 5bd827c2..5bfaa41a 100644 --- a/doc/examples/is_object.link +++ b/doc/examples/is_object.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/is_primitive.cpp b/doc/examples/is_primitive.cpp index 48872ab5..fb863b61 100644 --- a/doc/examples/is_primitive.cpp +++ b/doc/examples/is_primitive.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/is_primitive.link b/doc/examples/is_primitive.link index 6de65c79..aa46ead7 100644 --- a/doc/examples/is_primitive.link +++ b/doc/examples/is_primitive.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/is_string.cpp b/doc/examples/is_string.cpp index 2d6ed069..df887007 100644 --- a/doc/examples/is_string.cpp +++ b/doc/examples/is_string.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/is_string.link b/doc/examples/is_string.link index 86ef2b45..b4d8cc39 100644 --- a/doc/examples/is_string.link +++ b/doc/examples/is_string.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/is_structured.cpp b/doc/examples/is_structured.cpp index ed573e3c..66016083 100644 --- a/doc/examples/is_structured.cpp +++ b/doc/examples/is_structured.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/is_structured.link b/doc/examples/is_structured.link index 43385c8b..ef6804b7 100644 --- a/doc/examples/is_structured.link +++ b/doc/examples/is_structured.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/json_pointer.cpp b/doc/examples/json_pointer.cpp index 3b23dfb2..333d3963 100644 --- a/doc/examples/json_pointer.cpp +++ b/doc/examples/json_pointer.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/json_pointer.link b/doc/examples/json_pointer.link index 78881dd9..541a97d8 100644 --- a/doc/examples/json_pointer.link +++ b/doc/examples/json_pointer.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/json_pointer__to_string.cpp b/doc/examples/json_pointer__to_string.cpp index 4cb053c5..fc7f2557 100644 --- a/doc/examples/json_pointer__to_string.cpp +++ b/doc/examples/json_pointer__to_string.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/json_pointer__to_string.link b/doc/examples/json_pointer__to_string.link index 407609b2..5cae597e 100644 --- a/doc/examples/json_pointer__to_string.link +++ b/doc/examples/json_pointer__to_string.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/max_size.cpp b/doc/examples/max_size.cpp index d2f1b727..27989271 100644 --- a/doc/examples/max_size.cpp +++ b/doc/examples/max_size.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/max_size.link b/doc/examples/max_size.link index 5719ccaa..2bca668d 100644 --- a/doc/examples/max_size.link +++ b/doc/examples/max_size.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/meta.cpp b/doc/examples/meta.cpp index 28c88b8d..8e4ad3de 100644 --- a/doc/examples/meta.cpp +++ b/doc/examples/meta.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" #include // for std::setw using json = nlohmann::json; diff --git a/doc/examples/meta.link b/doc/examples/meta.link index 6a5ad0c1..05b7ef06 100644 --- a/doc/examples/meta.link +++ b/doc/examples/meta.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/meta.output b/doc/examples/meta.output index 933aac0f..d5ef8260 100644 --- a/doc/examples/meta.output +++ b/doc/examples/meta.output @@ -2,7 +2,7 @@ "compiler": { "c++": "201103", "family": "clang", - "version": "8.0.0 (clang-800.0.42.1)" + "version": "8.1.0 (clang-802.0.42)" }, "copyright": "(C) 2013-2017 Niels Lohmann", "name": "JSON for Modern C++", diff --git a/doc/examples/object.cpp b/doc/examples/object.cpp index 5b8ba589..3c089b0f 100644 --- a/doc/examples/object.cpp +++ b/doc/examples/object.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/object.link b/doc/examples/object.link index d9e63e53..0a435eb1 100644 --- a/doc/examples/object.link +++ b/doc/examples/object.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/operator__ValueType.cpp b/doc/examples/operator__ValueType.cpp index 77af700f..4895fe1e 100644 --- a/doc/examples/operator__ValueType.cpp +++ b/doc/examples/operator__ValueType.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" #include using json = nlohmann::json; diff --git a/doc/examples/operator__ValueType.link b/doc/examples/operator__ValueType.link index 9a689f82..84497ae4 100644 --- a/doc/examples/operator__ValueType.link +++ b/doc/examples/operator__ValueType.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/operator__equal.cpp b/doc/examples/operator__equal.cpp index 12bf92ed..e9910f32 100644 --- a/doc/examples/operator__equal.cpp +++ b/doc/examples/operator__equal.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/operator__equal.link b/doc/examples/operator__equal.link index 86707520..910bf449 100644 --- a/doc/examples/operator__equal.link +++ b/doc/examples/operator__equal.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/operator__equal.output b/doc/examples/operator__equal.output index e9dfd755..78067355 100644 --- a/doc/examples/operator__equal.output +++ b/doc/examples/operator__equal.output @@ -1,4 +1,4 @@ [1,2,3] == [1,2,4] false {"A":"a","B":"b"} == {"A":"a","B":"b"} true -17 == 17 true +17 == 17.0 true "foo" == "bar" false diff --git a/doc/examples/operator__equal__nullptr_t.cpp b/doc/examples/operator__equal__nullptr_t.cpp index 49c632cf..d7c2fbdc 100644 --- a/doc/examples/operator__equal__nullptr_t.cpp +++ b/doc/examples/operator__equal__nullptr_t.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/operator__equal__nullptr_t.link b/doc/examples/operator__equal__nullptr_t.link index d52acab4..30030502 100644 --- a/doc/examples/operator__equal__nullptr_t.link +++ b/doc/examples/operator__equal__nullptr_t.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/operator__greater.cpp b/doc/examples/operator__greater.cpp index bee3b2b5..1860625c 100644 --- a/doc/examples/operator__greater.cpp +++ b/doc/examples/operator__greater.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/operator__greater.link b/doc/examples/operator__greater.link index 5499771d..f28c8fe0 100644 --- a/doc/examples/operator__greater.link +++ b/doc/examples/operator__greater.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/operator__greaterequal.cpp b/doc/examples/operator__greaterequal.cpp index cd398966..e1d5d8b7 100644 --- a/doc/examples/operator__greaterequal.cpp +++ b/doc/examples/operator__greaterequal.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/operator__greaterequal.link b/doc/examples/operator__greaterequal.link index e476f396..a03af050 100644 --- a/doc/examples/operator__greaterequal.link +++ b/doc/examples/operator__greaterequal.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/operator__less.cpp b/doc/examples/operator__less.cpp index d4a7b794..53a595a9 100644 --- a/doc/examples/operator__less.cpp +++ b/doc/examples/operator__less.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/operator__less.link b/doc/examples/operator__less.link index d50c2207..9d8a489a 100644 --- a/doc/examples/operator__less.link +++ b/doc/examples/operator__less.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/operator__lessequal.cpp b/doc/examples/operator__lessequal.cpp index 20654d9e..6d82e894 100644 --- a/doc/examples/operator__lessequal.cpp +++ b/doc/examples/operator__lessequal.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/operator__lessequal.link b/doc/examples/operator__lessequal.link index 3b71a46d..637846fb 100644 --- a/doc/examples/operator__lessequal.link +++ b/doc/examples/operator__lessequal.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/operator__notequal.cpp b/doc/examples/operator__notequal.cpp index 499a00d1..1310a8d9 100644 --- a/doc/examples/operator__notequal.cpp +++ b/doc/examples/operator__notequal.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/operator__notequal.link b/doc/examples/operator__notequal.link index 49f1529c..fa7acee8 100644 --- a/doc/examples/operator__notequal.link +++ b/doc/examples/operator__notequal.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/operator__notequal.output b/doc/examples/operator__notequal.output index ddd838b4..9eba626e 100644 --- a/doc/examples/operator__notequal.output +++ b/doc/examples/operator__notequal.output @@ -1,4 +1,4 @@ [1,2,3] == [1,2,4] true {"A":"a","B":"b"} == {"A":"a","B":"b"} false -17 == 17 false +17 == 17.0 false "foo" == "bar" true diff --git a/doc/examples/operator__notequal__nullptr_t.cpp b/doc/examples/operator__notequal__nullptr_t.cpp index 014e6343..e65dcd7c 100644 --- a/doc/examples/operator__notequal__nullptr_t.cpp +++ b/doc/examples/operator__notequal__nullptr_t.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/operator__notequal__nullptr_t.link b/doc/examples/operator__notequal__nullptr_t.link index 21033623..f8be3cbe 100644 --- a/doc/examples/operator__notequal__nullptr_t.link +++ b/doc/examples/operator__notequal__nullptr_t.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/operator__value_t.cpp b/doc/examples/operator__value_t.cpp index d281df0d..dfcf0c3b 100644 --- a/doc/examples/operator__value_t.cpp +++ b/doc/examples/operator__value_t.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/operator__value_t.link b/doc/examples/operator__value_t.link index 3139efa1..2acdc711 100644 --- a/doc/examples/operator__value_t.link +++ b/doc/examples/operator__value_t.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/operator_deserialize.cpp b/doc/examples/operator_deserialize.cpp index 06a98db9..bf9d7795 100644 --- a/doc/examples/operator_deserialize.cpp +++ b/doc/examples/operator_deserialize.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" #include // for std::setw using json = nlohmann::json; diff --git a/doc/examples/operator_deserialize.link b/doc/examples/operator_deserialize.link index fea1503e..caf6025a 100644 --- a/doc/examples/operator_deserialize.link +++ b/doc/examples/operator_deserialize.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/operator_serialize.cpp b/doc/examples/operator_serialize.cpp index c3861b46..889bc108 100644 --- a/doc/examples/operator_serialize.cpp +++ b/doc/examples/operator_serialize.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" #include // for std::setw using json = nlohmann::json; diff --git a/doc/examples/operator_serialize.link b/doc/examples/operator_serialize.link index 45b5f4b1..a77519c2 100644 --- a/doc/examples/operator_serialize.link +++ b/doc/examples/operator_serialize.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/operatorarray__key_type.cpp b/doc/examples/operatorarray__key_type.cpp index 795abc08..2438f991 100644 --- a/doc/examples/operatorarray__key_type.cpp +++ b/doc/examples/operatorarray__key_type.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" #include // for std::setw using json = nlohmann::json; diff --git a/doc/examples/operatorarray__key_type.link b/doc/examples/operatorarray__key_type.link index d597ad78..7990937b 100644 --- a/doc/examples/operatorarray__key_type.link +++ b/doc/examples/operatorarray__key_type.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/operatorarray__key_type_const.cpp b/doc/examples/operatorarray__key_type_const.cpp index 5ad97355..36aa1db2 100644 --- a/doc/examples/operatorarray__key_type_const.cpp +++ b/doc/examples/operatorarray__key_type_const.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/operatorarray__key_type_const.link b/doc/examples/operatorarray__key_type_const.link index d34adb46..3b6f2f4a 100644 --- a/doc/examples/operatorarray__key_type_const.link +++ b/doc/examples/operatorarray__key_type_const.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/operatorarray__size_type.cpp b/doc/examples/operatorarray__size_type.cpp index 4ccbf270..27d7d9ef 100644 --- a/doc/examples/operatorarray__size_type.cpp +++ b/doc/examples/operatorarray__size_type.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/operatorarray__size_type.link b/doc/examples/operatorarray__size_type.link index aa5f718a..e978214e 100644 --- a/doc/examples/operatorarray__size_type.link +++ b/doc/examples/operatorarray__size_type.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/operatorarray__size_type_const.cpp b/doc/examples/operatorarray__size_type_const.cpp index 2ef79478..44264172 100644 --- a/doc/examples/operatorarray__size_type_const.cpp +++ b/doc/examples/operatorarray__size_type_const.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/operatorarray__size_type_const.link b/doc/examples/operatorarray__size_type_const.link index a5c351e6..3dc0ce3e 100644 --- a/doc/examples/operatorarray__size_type_const.link +++ b/doc/examples/operatorarray__size_type_const.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/operatorjson_pointer.cpp b/doc/examples/operatorjson_pointer.cpp index d7e8faff..63b233d2 100644 --- a/doc/examples/operatorjson_pointer.cpp +++ b/doc/examples/operatorjson_pointer.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/operatorjson_pointer.link b/doc/examples/operatorjson_pointer.link index debbdedb..67b3be46 100644 --- a/doc/examples/operatorjson_pointer.link +++ b/doc/examples/operatorjson_pointer.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/operatorjson_pointer_const.cpp b/doc/examples/operatorjson_pointer_const.cpp index 20ac36cb..8f3bc613 100644 --- a/doc/examples/operatorjson_pointer_const.cpp +++ b/doc/examples/operatorjson_pointer_const.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/operatorjson_pointer_const.link b/doc/examples/operatorjson_pointer_const.link index 1bc14234..38d54c6d 100644 --- a/doc/examples/operatorjson_pointer_const.link +++ b/doc/examples/operatorjson_pointer_const.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/parse__array__parser_callback_t.cpp b/doc/examples/parse__array__parser_callback_t.cpp index 68ecf602..790cc21d 100644 --- a/doc/examples/parse__array__parser_callback_t.cpp +++ b/doc/examples/parse__array__parser_callback_t.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" #include // for std::setw using json = nlohmann::json; diff --git a/doc/examples/parse__array__parser_callback_t.link b/doc/examples/parse__array__parser_callback_t.link index 3389916f..76e50355 100644 --- a/doc/examples/parse__array__parser_callback_t.link +++ b/doc/examples/parse__array__parser_callback_t.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/parse__contiguouscontainer__parser_callback_t.cpp b/doc/examples/parse__contiguouscontainer__parser_callback_t.cpp index 45e376d9..c953fe67 100644 --- a/doc/examples/parse__contiguouscontainer__parser_callback_t.cpp +++ b/doc/examples/parse__contiguouscontainer__parser_callback_t.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" #include // for std::setw using json = nlohmann::json; diff --git a/doc/examples/parse__contiguouscontainer__parser_callback_t.link b/doc/examples/parse__contiguouscontainer__parser_callback_t.link index 57d6dc3a..27f6f961 100644 --- a/doc/examples/parse__contiguouscontainer__parser_callback_t.link +++ b/doc/examples/parse__contiguouscontainer__parser_callback_t.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/parse__istream__parser_callback_t.cpp b/doc/examples/parse__istream__parser_callback_t.cpp index b5f84da4..1d25f705 100644 --- a/doc/examples/parse__istream__parser_callback_t.cpp +++ b/doc/examples/parse__istream__parser_callback_t.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" #include // for std::setw using json = nlohmann::json; diff --git a/doc/examples/parse__istream__parser_callback_t.link b/doc/examples/parse__istream__parser_callback_t.link index 4d2d0a9c..f1552bf6 100644 --- a/doc/examples/parse__istream__parser_callback_t.link +++ b/doc/examples/parse__istream__parser_callback_t.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/parse__iteratortype__parser_callback_t.cpp b/doc/examples/parse__iteratortype__parser_callback_t.cpp index c7a5bf83..1d752052 100644 --- a/doc/examples/parse__iteratortype__parser_callback_t.cpp +++ b/doc/examples/parse__iteratortype__parser_callback_t.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" #include // for std::setw using json = nlohmann::json; diff --git a/doc/examples/parse__iteratortype__parser_callback_t.link b/doc/examples/parse__iteratortype__parser_callback_t.link index 63f58fe6..f48f2fd1 100644 --- a/doc/examples/parse__iteratortype__parser_callback_t.link +++ b/doc/examples/parse__iteratortype__parser_callback_t.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/parse__string__parser_callback_t.cpp b/doc/examples/parse__string__parser_callback_t.cpp index bea4914f..b044eb32 100644 --- a/doc/examples/parse__string__parser_callback_t.cpp +++ b/doc/examples/parse__string__parser_callback_t.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" #include // for std::setw using json = nlohmann::json; diff --git a/doc/examples/parse__string__parser_callback_t.link b/doc/examples/parse__string__parser_callback_t.link index 292046b6..3b44d9dc 100644 --- a/doc/examples/parse__string__parser_callback_t.link +++ b/doc/examples/parse__string__parser_callback_t.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/patch.cpp b/doc/examples/patch.cpp index 4dfada32..ad30b78b 100644 --- a/doc/examples/patch.cpp +++ b/doc/examples/patch.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" #include // for std::setw using json = nlohmann::json; diff --git a/doc/examples/patch.link b/doc/examples/patch.link index 5d5032b7..93997149 100644 --- a/doc/examples/patch.link +++ b/doc/examples/patch.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/push_back.cpp b/doc/examples/push_back.cpp index 5d0877f0..8d1b1c5e 100644 --- a/doc/examples/push_back.cpp +++ b/doc/examples/push_back.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/push_back.link b/doc/examples/push_back.link index d0ca11be..efb1b6bd 100644 --- a/doc/examples/push_back.link +++ b/doc/examples/push_back.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/push_back__initializer_list.cpp b/doc/examples/push_back__initializer_list.cpp index 9fe01ad7..443c3639 100644 --- a/doc/examples/push_back__initializer_list.cpp +++ b/doc/examples/push_back__initializer_list.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/push_back__initializer_list.link b/doc/examples/push_back__initializer_list.link index 2b5abaed..bb56fb16 100644 --- a/doc/examples/push_back__initializer_list.link +++ b/doc/examples/push_back__initializer_list.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/push_back__object_t__value.cpp b/doc/examples/push_back__object_t__value.cpp index b55ee640..fe3318c0 100644 --- a/doc/examples/push_back__object_t__value.cpp +++ b/doc/examples/push_back__object_t__value.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/push_back__object_t__value.link b/doc/examples/push_back__object_t__value.link index 04bdbd9b..0fad4a34 100644 --- a/doc/examples/push_back__object_t__value.link +++ b/doc/examples/push_back__object_t__value.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/rbegin.cpp b/doc/examples/rbegin.cpp index 2ced56d3..c4f632eb 100644 --- a/doc/examples/rbegin.cpp +++ b/doc/examples/rbegin.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/rbegin.link b/doc/examples/rbegin.link index 79ed80c3..a780148a 100644 --- a/doc/examples/rbegin.link +++ b/doc/examples/rbegin.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/rend.cpp b/doc/examples/rend.cpp index ece1c551..d746ab4f 100644 --- a/doc/examples/rend.cpp +++ b/doc/examples/rend.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/rend.link b/doc/examples/rend.link index 6e8850f0..9d33126d 100644 --- a/doc/examples/rend.link +++ b/doc/examples/rend.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/size.cpp b/doc/examples/size.cpp index 5949ed63..34186499 100644 --- a/doc/examples/size.cpp +++ b/doc/examples/size.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/size.link b/doc/examples/size.link index 9b546197..6b27e7b9 100644 --- a/doc/examples/size.link +++ b/doc/examples/size.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/swap__array_t.cpp b/doc/examples/swap__array_t.cpp index 3c2cc6b7..b58c02c4 100644 --- a/doc/examples/swap__array_t.cpp +++ b/doc/examples/swap__array_t.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/swap__array_t.link b/doc/examples/swap__array_t.link index 1dca248c..ed7862c8 100644 --- a/doc/examples/swap__array_t.link +++ b/doc/examples/swap__array_t.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/swap__object_t.cpp b/doc/examples/swap__object_t.cpp index d3a0bae2..15f061de 100644 --- a/doc/examples/swap__object_t.cpp +++ b/doc/examples/swap__object_t.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/swap__object_t.link b/doc/examples/swap__object_t.link index cdb94e9a..6dd2dc18 100644 --- a/doc/examples/swap__object_t.link +++ b/doc/examples/swap__object_t.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/swap__reference.cpp b/doc/examples/swap__reference.cpp index 178a60be..ee033651 100644 --- a/doc/examples/swap__reference.cpp +++ b/doc/examples/swap__reference.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/swap__reference.link b/doc/examples/swap__reference.link index 262d1961..6bb80015 100644 --- a/doc/examples/swap__reference.link +++ b/doc/examples/swap__reference.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/swap__string_t.cpp b/doc/examples/swap__string_t.cpp index af166849..87ee1ed8 100644 --- a/doc/examples/swap__string_t.cpp +++ b/doc/examples/swap__string_t.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/swap__string_t.link b/doc/examples/swap__string_t.link index f3c448d5..5d204f86 100644 --- a/doc/examples/swap__string_t.link +++ b/doc/examples/swap__string_t.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/to_cbor.cpp b/doc/examples/to_cbor.cpp index 2d07e8ab..a4fbfbf1 100644 --- a/doc/examples/to_cbor.cpp +++ b/doc/examples/to_cbor.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" #include // for std::setw using json = nlohmann::json; diff --git a/doc/examples/to_cbor.link b/doc/examples/to_cbor.link index 3ab655c1..06695fec 100644 --- a/doc/examples/to_cbor.link +++ b/doc/examples/to_cbor.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/to_msgpack.cpp b/doc/examples/to_msgpack.cpp index dad83542..5e0d1884 100644 --- a/doc/examples/to_msgpack.cpp +++ b/doc/examples/to_msgpack.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" #include // for std::setw using json = nlohmann::json; diff --git a/doc/examples/to_msgpack.link b/doc/examples/to_msgpack.link index 4c7d3a27..904a6540 100644 --- a/doc/examples/to_msgpack.link +++ b/doc/examples/to_msgpack.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/type.cpp b/doc/examples/type.cpp index 07ec7885..eb764bf2 100644 --- a/doc/examples/type.cpp +++ b/doc/examples/type.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/type.link b/doc/examples/type.link index ccb4422b..c4d92739 100644 --- a/doc/examples/type.link +++ b/doc/examples/type.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/type_name.cpp b/doc/examples/type_name.cpp index 815e92d7..b849c10a 100644 --- a/doc/examples/type_name.cpp +++ b/doc/examples/type_name.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/type_name.link b/doc/examples/type_name.link index 39d1f973..8f36e649 100644 --- a/doc/examples/type_name.link +++ b/doc/examples/type_name.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/unflatten.cpp b/doc/examples/unflatten.cpp index a01a6349..f17e10ce 100644 --- a/doc/examples/unflatten.cpp +++ b/doc/examples/unflatten.cpp @@ -1,4 +1,4 @@ -#include +#include "json.hpp" #include // for std::setw using json = nlohmann::json; diff --git a/doc/examples/unflatten.link b/doc/examples/unflatten.link index 386f5e11..71d3d55d 100644 --- a/doc/examples/unflatten.link +++ b/doc/examples/unflatten.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file From e24df7eca98ae8590616c103cafcff9ed3fa5b7c Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 23 Apr 2017 15:10:40 +0200 Subject: [PATCH 169/530] :memo: improved documentation --- src/json.hpp | 250 +++++++++++++++++++++++++++++++++++++++++++-------- 1 file changed, 214 insertions(+), 36 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 1db82e12..99a6e854 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -6605,6 +6605,7 @@ class basic_json // output adapters // ///////////////////// + /// abstract output adapter interface template class output_adapter { @@ -6629,9 +6630,11 @@ class basic_json } }; + /// a type to simplify interfaces template using output_adapter_t = std::shared_ptr>; + /// output adapter for byte vectors template class output_vector_adapter : public output_adapter { @@ -6654,6 +6657,7 @@ class basic_json std::vector& v; }; + /// putput adatpter for output streams template class output_stream_adapter : public output_adapter { @@ -6676,6 +6680,7 @@ class basic_json std::basic_ostream& stream; }; + /// output adapter for basic_string template class output_string_adapter : public output_adapter { @@ -8773,7 +8778,7 @@ class basic_json } }; - // a type to simplify interfaces + /// a type to simplify interfaces using input_adapter_t = std::shared_ptr; /// input adapter for cached stream input @@ -8930,17 +8935,34 @@ class basic_json /// @{ private: + /*! + @brief deserialization of CBOR and MessagePack values + */ class binary_reader { public: + /*! + @brief create a binary reader + + @param[in] adapter input adapter to read from + */ explicit binary_reader(input_adapter_t adapter) : ia(adapter), is_little_endian(little_endianess()) - {} + { + assert(ia); + } /*! + @brief create a JSON value from CBOR input + @param[in] get_char whether a new character should be retrieved from the input (true, default) or whether the last read character should be considered instead + + @return JSON value created from CBOR input + + @throw parse_error.110 if input ended unexpectedly + @throw parse_error.112 if unsupported byte was read */ basic_json parse_cbor(const bool get_char = true) { @@ -9297,12 +9319,13 @@ class basic_json check_eof(); // code from RFC 7049, Appendix D, Figure 3: - // As half-precision floating-point numbers were only added to - // IEEE 754 in 2008, today's programming platforms often still - // only have limited support for them. It is very easy to - // include at least decoding support for them even without such - // support. An example of a small decoder for half-precision - // floating-point numbers in the C language is shown in Fig. 3. + // As half-precision floating-point numbers were only added + // to IEEE 754 in 2008, today's programming platforms often + // still only have limited support for them. It is very + // easy to include at least decoding support for them even + // without such support. An example of a small decoder for + // half-precision floating-point numbers in the C language + // is shown in Fig. 3. const int half = (byte1 << 8) + byte2; const int exp = (half >> 10) & 0x1f; const int mant = half & 0x3ff; @@ -9343,6 +9366,14 @@ class basic_json } } + /*! + @brief create a JSON value from MessagePack input + + @return JSON value created from MessagePack input + + @throw parse_error.110 if input ended unexpectedly + @throw parse_error.112 if unsupported byte was read + */ basic_json parse_msgpack() { switch (get()) @@ -9745,23 +9776,52 @@ class basic_json } } - private: - // from http://stackoverflow.com/a/1001328/266378 - static bool little_endianess() + /*! + @brief determine system byte order + + @return true iff system's byte order is little endian + + @note from http://stackoverflow.com/a/1001328/266378 + */ + static bool little_endianess() noexcept { int num = 1; return (*reinterpret_cast(&num) == 1); } + private: + /*! + @brief get next character from the input + + This function provides the interface to the used input adapter. It does + not throw in case the input reached EOF, but returns + `std::char_traits::eof()` in that case. + + @return character read from the input + */ int get() { ++chars_read; return (current = ia->get_character()); } + /* + @brief read a number from the input + + @tparam T the type of the number + + @return number of type @a T + + @note This function needs to respect the system's endianess, because + bytes in CBOR and MessagePack are stored in network order (big + endian) and therefore need reordering on little endian systems. + + @throw parse_error.110 if input has less than `sizeof(T)` bytes + */ template T get_number() { + // step 1: read input into array with system's byte order std::array vec; for (size_t i = 0; i < sizeof(T); ++i) { @@ -9779,11 +9839,21 @@ class basic_json } } + // step 2: convert array into number of type T and return T result; std::memcpy(&result, vec.data(), sizeof(T)); return result; } + /*! + @brief create a string by reading characters from the input + + @param[in] len number of bytes to read + + @return string created by reading @a len bytes + + @throw parse_error.110 if input has less than @a len bytes + */ std::string get_string(const size_t len) { std::string result; @@ -9796,6 +9866,18 @@ class basic_json return result; } + /*! + @brief reads a CBOR string + + This function first reads starting bytes to determine the expected + string length and then copies this number of bytes into a string. + Additionally, CBOR's strings with indefinite lengths are supported. + + @return string + + @throw parse_error.110 if input ended + @throw parse_error.113 if an unexpexted byte is read + */ std::string get_cbor_string() { check_eof(); @@ -9876,6 +9958,17 @@ class basic_json } } + /*! + @brief reads a MessagePack string + + This function first reads starting bytes to determine the expected + string length and then copies this number of bytes into a string. + + @return string + + @throw parse_error.110 if input ended + @throw parse_error.113 if an unexpexted byte is read + */ std::string get_msgpack_string() { check_eof(); @@ -9947,7 +10040,11 @@ class basic_json } } - void check_eof() + /*! + @brief check if input ended + @throw parse_error.110 if input ended + */ + void check_eof() const { if (JSON_UNLIKELY(current == std::char_traits::eof())) { @@ -9969,17 +10066,26 @@ class basic_json const bool is_little_endian = true; }; + /*! + @brief serialization to CBOR and MessagePack values + */ class binary_writer { public: - binary_writer() - : is_little_endian(little_endianess()) - {} + /*! + @brief create a binary writer + @param[in] adapter output adapter to write to + */ explicit binary_writer(output_adapter_t adapter) - : is_little_endian(little_endianess()), oa(adapter) - {} + : is_little_endian(binary_reader::little_endianess()), oa(adapter) + { + assert(oa); + } + /*! + @brief[in] j JSON value to serialize + */ void write_cbor(const basic_json& j) { switch (j.type()) @@ -10100,6 +10206,7 @@ class basic_json case value_t::string: { + // step 1: write control byte and the string length const auto N = j.m_value.string->size(); if (N <= 0x17) { @@ -10128,7 +10235,7 @@ class basic_json } // LCOV_EXCL_STOP - // append string + // step 2: write the string oa->write_characters(reinterpret_cast(j.m_value.string->c_str()), j.m_value.string->size()); break; @@ -10136,6 +10243,7 @@ class basic_json case value_t::array: { + // step 1: write control byte and the array size const auto N = j.m_value.array->size(); if (N <= 0x17) { @@ -10164,7 +10272,7 @@ class basic_json } // LCOV_EXCL_STOP - // append each element + // step 2: write each element for (const auto& el : *j.m_value.array) { write_cbor(el); @@ -10174,6 +10282,7 @@ class basic_json case value_t::object: { + // step 1: write control byte and the object size const auto N = j.m_value.object->size(); if (N <= 0x17) { @@ -10202,7 +10311,7 @@ class basic_json } // LCOV_EXCL_STOP - // append each element + // step 2: write each element for (const auto& el : *j.m_value.object) { write_cbor(el.first); @@ -10218,6 +10327,9 @@ class basic_json } } + /*! + @brief[in] j JSON value to serialize + */ void write_msgpack(const basic_json& j) { switch (j.type()) @@ -10353,6 +10465,7 @@ class basic_json case value_t::string: { + // step 1: write control byte and the string length const auto N = j.m_value.string->size(); if (N <= 31) { @@ -10378,7 +10491,7 @@ class basic_json write_number(static_cast(N)); } - // append string + // step 2: write the string oa->write_characters(reinterpret_cast(j.m_value.string->c_str()), j.m_value.string->size()); break; @@ -10386,6 +10499,7 @@ class basic_json case value_t::array: { + // step 1: write control byte and the array size const auto N = j.m_value.array->size(); if (N <= 15) { @@ -10405,7 +10519,7 @@ class basic_json write_number(static_cast(N)); } - // append each element + // step 2: write each element for (const auto& el : *j.m_value.array) { write_msgpack(el); @@ -10415,6 +10529,7 @@ class basic_json case value_t::object: { + // step 1: write control byte and the object size const auto N = j.m_value.object->size(); if (N <= 15) { @@ -10434,7 +10549,7 @@ class basic_json write_number(static_cast(N)); } - // append each element + // step 2: write each element for (const auto& el : *j.m_value.object) { write_msgpack(el.first); @@ -10451,12 +10566,24 @@ class basic_json } private: + /* + @brief write a number to output input + + @param[in] n number of type @a T + @tparam T the type of the number + + @note This function needs to respect the system's endianess, because + bytes in CBOR and MessagePack are stored in network order (big + endian) and therefore need reordering on little endian systems. + */ template void write_number(T n) { + // step 1: write number to array of length T std::array vec; std::memcpy(vec.data(), &n, sizeof(T)); + // step 2: write array to output (with possible reordering) for (size_t i = 0; i < sizeof(T); ++i) { // reverse byte order prior to conversion if necessary @@ -10471,13 +10598,6 @@ class basic_json } } - // from http://stackoverflow.com/a/1001328/266378 - static bool little_endianess() - { - int num = 1; - return (*reinterpret_cast(&num) == 1); - } - private: /// whether we can assume little endianess const bool is_little_endian = true; @@ -10928,12 +11048,19 @@ class basic_json // scan functions ///////////////////// - // must be called after \u was read; returns following xxxx as hex or -1 when error + /*! + @brief get codepoint from 4 hex characters following `\u` + + @return codepoint or -1 in case of an error (e.g. EOF or non-hex + character) + */ int get_codepoint() { + // this function only makes sense after reading `\u` assert(current == 'u'); int codepoint = 0; + // byte 1: \uXxxx switch (get()) { case '0': @@ -10993,6 +11120,7 @@ class basic_json return -1; } + // byte 2: \uxXxx switch (get()) { case '0': @@ -11052,6 +11180,7 @@ class basic_json return -1; } + // byte 3: \uxxXx switch (get()) { case '0': @@ -11111,6 +11240,7 @@ class basic_json return -1; } + // byte 4: \uxxxX switch (get()) { case '0': @@ -11173,6 +11303,10 @@ class basic_json return codepoint; } + /*! + @brief create diagnostic representation of a codepoint + @return string "U+XXXX" for codepoint XXXX + */ static std::string codepoint_to_string(int codepoint) { std::stringstream ss; @@ -11180,6 +11314,20 @@ class basic_json return ss.str(); } + /*! + @brief scan a string literal + + This function scans a string according to Sect. 7 of RFC 7159. While + scanning, bytes are escaped and copied into buffer yytext. Then the + function returns successfully, yytext is null-terminated and yylen + contains the number of bytes in the string. + + @return token_type::value_string if string could be successfully + scanned, token_type::parse_error otherwise + + @note In case of errors, variable error_message contains a textual + description. + */ token_type scan_string() { // reset yytext (ignore opening quote) @@ -11714,6 +11862,17 @@ class basic_json } /*! + @brief scan a number literal + + This function scans a string according to Sect. 6 of RFC 7159. + + The function is realized with a deterministic finite state machine + derived from the grammar described in RFC 7159. Starting in state + "init", the input is read and used to determined the next state. Only + state "done" accepts the number. State "error" is a trap state to model + errors. In the table below, "anything" means any character but the ones + listed before. + state | 0 | 1-9 | e E | + | - | . | anything ---------|----------|----------|----------|---------|---------|----------|----------- init | zero | any1 | [error] | [error] | minus | [error] | [error] @@ -11725,13 +11884,31 @@ class basic_json exponent | any2 | any2 | [error] | sign | sign | [error] | [error] sign | any2 | any2 | [error] | [error] | [error] | [error] | [error] any2 | any2 | any2 | done | done | done | done | done + + The state machine is realized with one label per state (prefixed with + "scan_number_") and `goto` statements between them. The state machine + contains cycles, but any cycle can be left when EOF is read. Therefore, + the function is guaranteed to terminate. + + During scanning, the read bytes are stored in yytext. This string is + then converted to a signed integer, an unsigned integer, or a + floating-point number. + + @return token_type::value_unsigned, token_type::value_integer, or + token_type::value_float if number could be successfully scanned, + token_type::parse_error otherwise + + @note The scanner is independent of the current locale. Internally, the + locale's decimal point is used instead of `.` to work with the + locale-dependent converters. */ token_type scan_number() { + // reset yytext to store the number's bytes reset(); - // the type of the parsed number; initially set to unsigned; will - // be changed if minus sign, decimal point or exponent is read + // the type of the parsed number; initially set to unsigned; will be + // changed if minus sign, decimal point or exponent is read token_type number_type = token_type::value_unsigned; // state (init): we just found out we need to scan a number @@ -12008,7 +12185,8 @@ scan_number_any2: } scan_number_done: - // unget the character after the number + // unget the character after the number (we only read it to know + // that we are done scanning a number) --chars_read; next_unget = true; @@ -12155,7 +12333,7 @@ scan_number_done: const std::string get_string() { // yytext cannot be returned as char*, because it may contain a - // null byte + // null byte (parsed as "\u0000") return std::string(yytext.data(), yylen); } @@ -12302,7 +12480,7 @@ scan_number_done: number_float_t value_float = 0; /// the decimal point - const char decimal_point_char = '\0'; + const char decimal_point_char = '.'; }; /*! From 4aedae400ecb0c99424d2f6721db40b5816c1f44 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 23 Apr 2017 15:12:50 +0200 Subject: [PATCH 170/530] :white_check_mark: added exhaustive UTF-8 tests Creates all well-formed sequences of bytes up to length 4. Furthermore, creates ill-formed sequences by removing required trailing bytes or changing bytes. As the tests can take a lot of time, preprocessor symbols are introduced. --- test/src/unit-unicode.cpp | 1052 +++++++++++++++++++++++++++++++++++++ 1 file changed, 1052 insertions(+) diff --git a/test/src/unit-unicode.cpp b/test/src/unit-unicode.cpp index 9fff61af..d889a87b 100644 --- a/test/src/unit-unicode.cpp +++ b/test/src/unit-unicode.cpp @@ -34,6 +34,1058 @@ using nlohmann::json; #include +TEST_CASE("RFC 3629") +{ + /* + RFC 3629 describes in Sect. 4 the syntax of UTF-8 byte sequences as + follows: + + A UTF-8 string is a sequence of octets representing a sequence of UCS + characters. An octet sequence is valid UTF-8 only if it matches the + following syntax, which is derived from the rules for encoding UTF-8 + and is expressed in the ABNF of [RFC2234]. + + UTF8-octets = *( UTF8-char ) + UTF8-char = UTF8-1 / UTF8-2 / UTF8-3 / UTF8-4 + UTF8-1 = %x00-7F + UTF8-2 = %xC2-DF UTF8-tail + UTF8-3 = %xE0 %xA0-BF UTF8-tail / %xE1-EC 2( UTF8-tail ) / + %xED %x80-9F UTF8-tail / %xEE-EF 2( UTF8-tail ) + UTF8-4 = %xF0 %x90-BF 2( UTF8-tail ) / %xF1-F3 3( UTF8-tail ) / + %xF4 %x80-8F 2( UTF8-tail ) + UTF8-tail = %x80-BF + */ + + auto create_string = [](int byte1, int byte2 = -1, int byte3 = -1, int byte4 = -1) + { + std::string result = "\"" + std::string(1, static_cast(byte1)); + if (byte2 != -1) + { + result += std::string(1, static_cast(byte2)); + } + if (byte3 != -1) + { + result += std::string(1, static_cast(byte3)); + } + if (byte4 != -1) + { + result += std::string(1, static_cast(byte4)); + } + result += "\""; + return result; + }; + + SECTION("ill-formed first byte") + { + for (int byte1 = 0x80; byte1 <= 0xC1; ++byte1) + { + const auto json_string = create_string(byte1); + CAPTURE(byte1); + CAPTURE(json_string); + CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + } + + for (int byte1 = 0xF5; byte1 <= 0xFF; ++byte1) + { + const auto json_string = create_string(byte1); + CAPTURE(byte1); + CAPTURE(json_string); + CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + } + } + + SECTION("UTF8-1 (x00-x7F)") + { + SECTION("well-formed") + { + for (int byte1 = 0x00; byte1 <= 0x7F; ++byte1) + { + const auto json_string = create_string(byte1); + CAPTURE(byte1); + CAPTURE(json_string); + + // unescaped control characters are parse errors in JSON + if (0x00 <= byte1 and byte1 <= 0x1F) + { + CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + continue; + } + + // a single quote is a parse error in JSON + if (byte1 == 0x22) + { + CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + continue; + } + + // a single backslash is a parse error in JSON + if (byte1 == 0x5C) + { + CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + continue; + } + + // all other characters are OK + CHECK_NOTHROW(json::parse(json_string)); + } + } + } + + SECTION("UTF8-2 (xC2-xDF UTF8-tail)") + { + SECTION("well-formed") + { + for (int byte1 = 0xC2; byte1 <= 0xDF; ++byte1) + { + for (int byte2 = 0x80; byte2 <= 0xBF; ++byte2) + { + const auto json_string = create_string(byte1, byte2); + CAPTURE(byte1); + CAPTURE(byte2); + CAPTURE(json_string); + CHECK_NOTHROW(json::parse(json_string)); + } + } + } + + SECTION("ill-formed: missing second byte") + { + for (int byte1 = 0xC2; byte1 <= 0xDF; ++byte1) + { + const auto json_string = create_string(byte1); + CAPTURE(byte1); + CAPTURE(json_string); + CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + } + } + +#ifdef WRONG_SECOND + SECTION("ill-formed: wrong second byte") + { + for (int byte1 = 0xC2; byte1 <= 0xDF; ++byte1) + { + for (int byte2 = 0x00; byte2 <= 0xFF; ++byte2) + { + // skip correct second byte + if (0x80 <= byte2 and byte2 <= 0xBF) + { + continue; + } + + const auto json_string = create_string(byte1, byte2); + CAPTURE(byte1); + CAPTURE(byte2); + CAPTURE(json_string); + CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + } + } + } +#endif + } + + SECTION("UTF8-3 (xE0 xA0-BF UTF8-tail)") + { + SECTION("well-formed") + { + for (int byte1 = 0xE0; byte1 <= 0xE0; ++byte1) + { + for (int byte2 = 0xA0; byte2 <= 0xBF; ++byte2) + { + for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) + { + const auto json_string = create_string(byte1, byte2, byte3); + CAPTURE(byte1); + CAPTURE(byte2); + CAPTURE(byte3); + CAPTURE(json_string); + CHECK_NOTHROW(json::parse(json_string)); + } + } + } + } + + SECTION("ill-formed: missing second byte") + { + for (int byte1 = 0xE0; byte1 <= 0xE0; ++byte1) + { + const auto json_string = create_string(byte1); + CAPTURE(byte1); + CAPTURE(json_string); + CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + } + } + + SECTION("ill-formed: missing third byte") + { + for (int byte1 = 0xE0; byte1 <= 0xE0; ++byte1) + { + for (int byte2 = 0xA0; byte2 <= 0xBF; ++byte2) + { + const auto json_string = create_string(byte1, byte2); + CAPTURE(byte1); + CAPTURE(byte2); + CAPTURE(json_string); + CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + } + } + } + +#ifdef WRONG_SECOND + SECTION("ill-formed: wrong second byte") + { + for (int byte1 = 0xE0; byte1 <= 0xE0; ++byte1) + { + for (int byte2 = 0x00; byte2 <= 0xFF; ++byte2) + { + // skip correct second byte + if (0xA0 <= byte2 and byte2 <= 0xBF) + { + continue; + } + + for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) + { + const auto json_string = create_string(byte1, byte2, byte3); + CAPTURE(byte1); + CAPTURE(byte2); + CAPTURE(byte3); + CAPTURE(json_string); + CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + } + } + } + } +#endif + +#ifdef WRONG_THIRD + SECTION("ill-formed: wrong third byte") + { + for (int byte1 = 0xE0; byte1 <= 0xE0; ++byte1) + { + for (int byte2 = 0xA0; byte2 <= 0xBF; ++byte2) + { + for (int byte3 = 0x00; byte3 <= 0xFF; ++byte2) + { + // skip correct third byte + if (0x80 <= byte3 and byte3 <= 0xBF) + { + continue; + } + + const auto json_string = create_string(byte1, byte2, byte3); + CAPTURE(byte1); + CAPTURE(byte2); + CAPTURE(byte3); + CAPTURE(json_string); + CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + } + } + } + } +#endif + } + + SECTION("UTF8-3 (xE1-xEC UTF8-tail UTF8-tail)") + { + SECTION("well-formed") + { + for (int byte1 = 0xE1; byte1 <= 0xEC; ++byte1) + { + for (int byte2 = 0x80; byte2 <= 0xBF; ++byte2) + { + for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) + { + const auto json_string = create_string(byte1, byte2, byte3); + CAPTURE(byte1); + CAPTURE(byte2); + CAPTURE(byte3); + CAPTURE(json_string); + CHECK_NOTHROW(json::parse(json_string)); + } + } + } + } + + SECTION("ill-formed: missing second byte") + { + for (int byte1 = 0xE1; byte1 <= 0xEC; ++byte1) + { + const auto json_string = create_string(byte1); + CAPTURE(byte1); + CAPTURE(json_string); + CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + } + } + + SECTION("ill-formed: missing third byte") + { + for (int byte1 = 0xE1; byte1 <= 0xEC; ++byte1) + { + for (int byte2 = 0x80; byte2 <= 0xBF; ++byte2) + { + const auto json_string = create_string(byte1, byte2); + CAPTURE(byte1); + CAPTURE(byte2); + CAPTURE(json_string); + CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + } + } + } + +#ifdef WRONG_SECOND + SECTION("ill-formed: wrong second byte") + { + for (int byte1 = 0xE1; byte1 <= 0xEC; ++byte1) + { + for (int byte2 = 0x00; byte2 <= 0xFF; ++byte2) + { + // skip correct second byte + if (0x80 <= byte2 and byte2 <= 0xBF) + { + continue; + } + + for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) + { + const auto json_string = create_string(byte1, byte2, byte3); + CAPTURE(byte1); + CAPTURE(byte2); + CAPTURE(byte3); + CAPTURE(json_string); + CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + } + } + } + } +#endif + +#ifdef WRONG_THIRD + SECTION("ill-formed: wrong third byte") + { + for (int byte1 = 0xE1; byte1 <= 0xEC; ++byte1) + { + for (int byte2 = 0x80; byte2 <= 0xBF; ++byte2) + { + for (int byte3 = 0x00; byte3 <= 0xFF; ++byte2) + { + // skip correct third byte + if (0x80 <= byte3 and byte3 <= 0xBF) + { + continue; + } + + const auto json_string = create_string(byte1, byte2, byte3); + CAPTURE(byte1); + CAPTURE(byte2); + CAPTURE(byte3); + CAPTURE(json_string); + CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + } + } + } + } +#endif + } + + SECTION("UTF8-3 (xED x80-9F UTF8-tail)") + { + SECTION("well-formed") + { + for (int byte1 = 0xED; byte1 <= 0xED; ++byte1) + { + for (int byte2 = 0x80; byte2 <= 0x9F; ++byte2) + { + for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) + { + const auto json_string = create_string(byte1, byte2, byte3); + CAPTURE(byte1); + CAPTURE(byte2); + CAPTURE(byte3); + CAPTURE(json_string); + CHECK_NOTHROW(json::parse(json_string)); + } + } + } + } + + SECTION("ill-formed: missing second byte") + { + for (int byte1 = 0xED; byte1 <= 0xED; ++byte1) + { + const auto json_string = create_string(byte1); + CAPTURE(byte1); + CAPTURE(json_string); + CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + } + } + + SECTION("ill-formed: missing third byte") + { + for (int byte1 = 0xED; byte1 <= 0xED; ++byte1) + { + for (int byte2 = 0x80; byte2 <= 0x9F; ++byte2) + { + const auto json_string = create_string(byte1, byte2); + CAPTURE(byte1); + CAPTURE(byte2); + CAPTURE(json_string); + CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + } + } + } + +#ifdef WRONG_SECOND + SECTION("ill-formed: wrong second byte") + { + for (int byte1 = 0xED; byte1 <= 0xED; ++byte1) + { + for (int byte2 = 0x00; byte2 <= 0xFF; ++byte2) + { + // skip correct second byte + if (0x80 <= byte2 and byte2 <= 0x9F) + { + continue; + } + + for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) + { + const auto json_string = create_string(byte1, byte2, byte3); + CAPTURE(byte1); + CAPTURE(byte2); + CAPTURE(byte3); + CAPTURE(json_string); + CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + } + } + } + } +#endif + +#ifdef WRONG_THIRD + SECTION("ill-formed: wrong third byte") + { + for (int byte1 = 0xED; byte1 <= 0xED; ++byte1) + { + for (int byte2 = 0x80; byte2 <= 0x9F; ++byte2) + { + for (int byte3 = 0x00; byte3 <= 0xFF; ++byte2) + { + // skip correct third byte + if (0x80 <= byte3 and byte3 <= 0xBF) + { + continue; + } + + const auto json_string = create_string(byte1, byte2, byte3); + CAPTURE(byte1); + CAPTURE(byte2); + CAPTURE(byte3); + CAPTURE(json_string); + CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + } + } + } + } +#endif + } + + SECTION("UTF8-3 (xEE-xEF UTF8-tail UTF8-tail)") + { + SECTION("well-formed") + { + for (int byte1 = 0xEE; byte1 <= 0xEF; ++byte1) + { + for (int byte2 = 0x80; byte2 <= 0xBF; ++byte2) + { + for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) + { + const auto json_string = create_string(byte1, byte2, byte3); + CAPTURE(byte1); + CAPTURE(byte2); + CAPTURE(byte3); + CAPTURE(json_string); + CHECK_NOTHROW(json::parse(json_string)); + } + } + } + } + + SECTION("ill-formed: missing second byte") + { + for (int byte1 = 0xEE; byte1 <= 0xEF; ++byte1) + { + const auto json_string = create_string(byte1); + CAPTURE(byte1); + CAPTURE(json_string); + CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + } + } + + SECTION("ill-formed: missing third byte") + { + for (int byte1 = 0xEE; byte1 <= 0xEF; ++byte1) + { + for (int byte2 = 0x80; byte2 <= 0xBF; ++byte2) + { + const auto json_string = create_string(byte1, byte2); + CAPTURE(byte1); + CAPTURE(byte2); + CAPTURE(json_string); + CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + } + } + } + +#ifdef WRONG_SECOND + SECTION("ill-formed: wrong second byte") + { + for (int byte1 = 0xEE; byte1 <= 0xEF; ++byte1) + { + for (int byte2 = 0x00; byte2 <= 0xFF; ++byte2) + { + // skip correct second byte + if (0x80 <= byte2 and byte2 <= 0xBF) + { + continue; + } + + for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) + { + const auto json_string = create_string(byte1, byte2, byte3); + CAPTURE(byte1); + CAPTURE(byte2); + CAPTURE(byte3); + CAPTURE(json_string); + CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + } + } + } + } +#endif + +#ifdef WRONG_THIRD + SECTION("ill-formed: wrong third byte") + { + for (int byte1 = 0xEE; byte1 <= 0xEF; ++byte1) + { + for (int byte2 = 0x80; byte2 <= 0xBF; ++byte2) + { + for (int byte3 = 0x00; byte3 <= 0xFF; ++byte2) + { + // skip correct third byte + if (0x80 <= byte3 and byte3 <= 0xBF) + { + continue; + } + + const auto json_string = create_string(byte1, byte2, byte3); + CAPTURE(byte1); + CAPTURE(byte2); + CAPTURE(byte3); + CAPTURE(json_string); + CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + } + } + } + } +#endif + } + + SECTION("UTF8-4 (xF0 x90-BF UTF8-tail UTF8-tail)") + { + SECTION("well-formed") + { + for (int byte1 = 0xF0; byte1 <= 0xF0; ++byte1) + { + for (int byte2 = 0x90; byte2 <= 0xBF; ++byte2) + { + for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) + { + for (int byte4 = 0x80; byte4 <= 0xBF; ++byte4) + { + const auto json_string = create_string(byte1, byte2, byte3, byte4); + CAPTURE(byte1); + CAPTURE(byte2); + CAPTURE(byte3); + CAPTURE(byte4); + CAPTURE(json_string); + CHECK_NOTHROW(json::parse(json_string)); + } + } + } + } + } + + SECTION("ill-formed: missing second byte") + { + for (int byte1 = 0xF0; byte1 <= 0xF0; ++byte1) + { + const auto json_string = create_string(byte1); + CAPTURE(byte1); + CAPTURE(json_string); + CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + } + } + + SECTION("ill-formed: missing third byte") + { + for (int byte1 = 0xF0; byte1 <= 0xF0; ++byte1) + { + for (int byte2 = 0x90; byte2 <= 0xBF; ++byte2) + { + const auto json_string = create_string(byte1, byte2); + CAPTURE(byte1); + CAPTURE(byte2); + CAPTURE(json_string); + CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + } + } + } + + SECTION("ill-formed: missing fourth byte") + { + for (int byte1 = 0xF0; byte1 <= 0xF0; ++byte1) + { + for (int byte2 = 0x90; byte2 <= 0xBF; ++byte2) + { + for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) + { + const auto json_string = create_string(byte1, byte2, byte3); + CAPTURE(byte1); + CAPTURE(byte2); + CAPTURE(byte3); + CAPTURE(json_string); + CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + } + } + } + } + +#ifdef WRONG_SECOND + SECTION("ill-formed: wrong second byte") + { + for (int byte1 = 0xF0; byte1 <= 0xF0; ++byte1) + { + for (int byte2 = 0x00; byte2 <= 0xFF; ++byte2) + { + // skip correct second byte + if (0x90 <= byte2 and byte2 <= 0xBF) + { + continue; + } + + for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) + { + for (int byte4 = 0x80; byte4 <= 0xBF; ++byte4) + { + const auto json_string = create_string(byte1, byte2, byte3, byte4); + CAPTURE(byte1); + CAPTURE(byte2); + CAPTURE(byte3); + CAPTURE(byte4); + CAPTURE(json_string); + CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + } + } + } + } + } +#endif + +#ifdef WRONG_THIRD + SECTION("ill-formed: wrong third byte") + { + for (int byte1 = 0xF0; byte1 <= 0xF0; ++byte1) + { + for (int byte2 = 0x90; byte2 <= 0xBF; ++byte2) + { + for (int byte3 = 0x00; byte3 <= 0xFF; ++byte2) + { + // skip correct third byte + if (0x80 <= byte3 and byte3 <= 0xBF) + { + continue; + } + + for (int byte4 = 0x80; byte4 <= 0xBF; ++byte4) + { + const auto json_string = create_string(byte1, byte2, byte3, byte4); + CAPTURE(byte1); + CAPTURE(byte2); + CAPTURE(byte3); + CAPTURE(byte4); + CAPTURE(json_string); + CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + } + } + } + } + } +#endif + +#ifdef WRONG_FOURTH + SECTION("ill-formed: wrong fourth byte") + { + for (int byte1 = 0xF0; byte1 <= 0xF0; ++byte1) + { + for (int byte2 = 0x90; byte2 <= 0xBF; ++byte2) + { + for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) + { + { + for (int byte4 = 0x00; byte4 <= 0xFF; ++byte4) + { + // skip correct second byte + if (0x80 <= byte3 and byte3 <= 0xBF) + { + continue; + } + + const auto json_string = create_string(byte1, byte2, byte3, byte4); + CAPTURE(byte1); + CAPTURE(byte2); + CAPTURE(byte3); + CAPTURE(byte4); + CAPTURE(json_string); + CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + } + } + } + } + } + } +#endif + } + + SECTION("UTF8-4 (xF1-F3 UTF8-tail UTF8-tail UTF8-tail)") + { + SECTION("well-formed") + { + for (int byte1 = 0xF1; byte1 <= 0xF3; ++byte1) + { + for (int byte2 = 0x80; byte2 <= 0xBF; ++byte2) + { + for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) + { + for (int byte4 = 0x80; byte4 <= 0xBF; ++byte4) + { + const auto json_string = create_string(byte1, byte2, byte3, byte4); + CAPTURE(byte1); + CAPTURE(byte2); + CAPTURE(byte3); + CAPTURE(byte4); + CAPTURE(json_string); + CHECK_NOTHROW(json::parse(json_string)); + } + } + } + } + } + + SECTION("ill-formed: missing second byte") + { + for (int byte1 = 0xF1; byte1 <= 0xF3; ++byte1) + { + const auto json_string = create_string(byte1); + CAPTURE(byte1); + CAPTURE(json_string); + CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + } + } + + SECTION("ill-formed: missing third byte") + { + for (int byte1 = 0xF1; byte1 <= 0xF3; ++byte1) + { + for (int byte2 = 0x80; byte2 <= 0xBF; ++byte2) + { + const auto json_string = create_string(byte1, byte2); + CAPTURE(byte1); + CAPTURE(byte2); + CAPTURE(json_string); + CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + } + } + } + + SECTION("ill-formed: missing fourth byte") + { + for (int byte1 = 0xF1; byte1 <= 0xF3; ++byte1) + { + for (int byte2 = 0x80; byte2 <= 0xBF; ++byte2) + { + for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) + { + const auto json_string = create_string(byte1, byte2, byte3); + CAPTURE(byte1); + CAPTURE(byte2); + CAPTURE(byte3); + CAPTURE(json_string); + CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + } + } + } + } + +#ifdef WRONG_SECOND + SECTION("ill-formed: wrong second byte") + { + for (int byte1 = 0xF1; byte1 <= 0xF3; ++byte1) + { + for (int byte2 = 0x00; byte2 <= 0xFF; ++byte2) + { + // skip correct second byte + if (0x80 <= byte2 and byte2 <= 0xBF) + { + continue; + } + + for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) + { + for (int byte4 = 0x80; byte4 <= 0xBF; ++byte4) + { + const auto json_string = create_string(byte1, byte2, byte3, byte4); + CAPTURE(byte1); + CAPTURE(byte2); + CAPTURE(byte3); + CAPTURE(byte4); + CAPTURE(json_string); + CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + } + } + } + } + } +#endif + +#ifdef WRONG_THIRD + SECTION("ill-formed: wrong third byte") + { + for (int byte1 = 0xF1; byte1 <= 0xF3; ++byte1) + { + for (int byte2 = 0x80; byte2 <= 0xBF; ++byte2) + { + for (int byte3 = 0x00; byte3 <= 0xFF; ++byte2) + { + // skip correct third byte + if (0x80 <= byte3 and byte3 <= 0xBF) + { + continue; + } + + for (int byte4 = 0x80; byte4 <= 0xBF; ++byte4) + { + const auto json_string = create_string(byte1, byte2, byte3, byte4); + CAPTURE(byte1); + CAPTURE(byte2); + CAPTURE(byte3); + CAPTURE(byte4); + CAPTURE(json_string); + CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + } + } + } + } + } +#endif + +#ifdef WRONG_FOURTH + SECTION("ill-formed: wrong fourth byte") + { + for (int byte1 = 0xF1; byte1 <= 0xF3; ++byte1) + { + for (int byte2 = 0x80; byte2 <= 0xBF; ++byte2) + { + for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) + { + for (int byte4 = 0x00; byte4 <= 0xFF; ++byte4) + { + // skip correct second byte + if (0x80 <= byte3 and byte3 <= 0xBF) + { + continue; + } + + const auto json_string = create_string(byte1, byte2, byte3, byte4); + CAPTURE(byte1); + CAPTURE(byte2); + CAPTURE(byte3); + CAPTURE(byte4); + CAPTURE(json_string); + CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + } + } + } + } + } +#endif + } + + SECTION("UTF8-4 (xF4 x80-8F UTF8-tail UTF8-tail)") + { + SECTION("well-formed") + { + for (int byte1 = 0xF4; byte1 <= 0xF4; ++byte1) + { + for (int byte2 = 0x80; byte2 <= 0x8F; ++byte2) + { + for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) + { + for (int byte4 = 0x80; byte4 <= 0xBF; ++byte4) + { + const auto json_string = create_string(byte1, byte2, byte3, byte4); + CAPTURE(byte1); + CAPTURE(byte2); + CAPTURE(byte3); + CAPTURE(byte4); + CAPTURE(json_string); + CHECK_NOTHROW(json::parse(json_string)); + } + } + } + } + } + + SECTION("ill-formed: missing second byte") + { + for (int byte1 = 0xF4; byte1 <= 0xF4; ++byte1) + { + const auto json_string = create_string(byte1); + CAPTURE(byte1); + CAPTURE(json_string); + CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + } + } + + SECTION("ill-formed: missing third byte") + { + for (int byte1 = 0xF4; byte1 <= 0xF4; ++byte1) + { + for (int byte2 = 0x80; byte2 <= 0x8F; ++byte2) + { + const auto json_string = create_string(byte1, byte2); + CAPTURE(byte1); + CAPTURE(byte2); + CAPTURE(json_string); + CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + } + } + } + + SECTION("ill-formed: missing fourth byte") + { + for (int byte1 = 0xF4; byte1 <= 0xF4; ++byte1) + { + for (int byte2 = 0x80; byte2 <= 0x8F; ++byte2) + { + for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) + { + const auto json_string = create_string(byte1, byte2, byte3); + CAPTURE(byte1); + CAPTURE(byte2); + CAPTURE(byte3); + CAPTURE(json_string); + CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + } + } + } + } + +#ifdef WRONG_SECOND + SECTION("ill-formed: wrong second byte") + { + for (int byte1 = 0xF4; byte1 <= 0xF4; ++byte1) + { + for (int byte2 = 0x00; byte2 <= 0xFF; ++byte2) + { + // skip correct second byte + if (0x80 <= byte2 and byte2 <= 0x8F) + { + continue; + } + + for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) + { + for (int byte4 = 0x80; byte4 <= 0xBF; ++byte4) + { + const auto json_string = create_string(byte1, byte2, byte3, byte4); + CAPTURE(byte1); + CAPTURE(byte2); + CAPTURE(byte3); + CAPTURE(byte4); + CAPTURE(json_string); + CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + } + } + } + } + } +#endif + +#ifdef WRONG_THIRD + SECTION("ill-formed: wrong third byte") + { + for (int byte1 = 0xF4; byte1 <= 0xF4; ++byte1) + { + for (int byte2 = 0x80; byte2 <= 0x8F; ++byte2) + { + for (int byte3 = 0x00; byte3 <= 0xFF; ++byte2) + { + // skip correct third byte + if (0x80 <= byte3 and byte3 <= 0xBF) + { + continue; + } + + for (int byte4 = 0x80; byte4 <= 0xBF; ++byte4) + { + const auto json_string = create_string(byte1, byte2, byte3, byte4); + CAPTURE(byte1); + CAPTURE(byte2); + CAPTURE(byte3); + CAPTURE(byte4); + CAPTURE(json_string); + CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + } + } + } + } + } +#endif + +#ifdef WRONG_FOURTH + SECTION("ill-formed: wrong fourth byte") + { + for (int byte1 = 0xF4; byte1 <= 0xF4; ++byte1) + { + for (int byte2 = 0x80; byte2 <= 0x8F; ++byte2) + { + for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) + { + for (int byte4 = 0x00; byte4 <= 0xFF; ++byte4) + { + // skip correct second byte + if (0x80 <= byte3 and byte3 <= 0xBF) + { + continue; + } + + const auto json_string = create_string(byte1, byte2, byte3, byte4); + CAPTURE(byte1); + CAPTURE(byte2); + CAPTURE(byte3); + CAPTURE(byte4); + CAPTURE(json_string); + CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + } + } + } + } + } +#endif + } +} + TEST_CASE("Unicode", "[hide]") { /* NOTE: to_unicode is not used any more From 5febd04a26a6afccf1c335e012b443503c9ef068 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 23 Apr 2017 18:22:35 +0200 Subject: [PATCH 171/530] :bug: fixed test suite --- test/src/unit-unicode.cpp | 16 ++++++++-------- 1 file changed, 8 insertions(+), 8 deletions(-) diff --git a/test/src/unit-unicode.cpp b/test/src/unit-unicode.cpp index d889a87b..a0eef991 100644 --- a/test/src/unit-unicode.cpp +++ b/test/src/unit-unicode.cpp @@ -34,7 +34,7 @@ using nlohmann::json; #include -TEST_CASE("RFC 3629") +TEST_CASE("RFC 3629", "[hide]") { /* RFC 3629 describes in Sect. 4 the syntax of UTF-8 byte sequences as @@ -264,7 +264,7 @@ TEST_CASE("RFC 3629") { for (int byte2 = 0xA0; byte2 <= 0xBF; ++byte2) { - for (int byte3 = 0x00; byte3 <= 0xFF; ++byte2) + for (int byte3 = 0x00; byte3 <= 0xFF; ++byte3) { // skip correct third byte if (0x80 <= byte3 and byte3 <= 0xBF) @@ -366,7 +366,7 @@ TEST_CASE("RFC 3629") { for (int byte2 = 0x80; byte2 <= 0xBF; ++byte2) { - for (int byte3 = 0x00; byte3 <= 0xFF; ++byte2) + for (int byte3 = 0x00; byte3 <= 0xFF; ++byte3) { // skip correct third byte if (0x80 <= byte3 and byte3 <= 0xBF) @@ -468,7 +468,7 @@ TEST_CASE("RFC 3629") { for (int byte2 = 0x80; byte2 <= 0x9F; ++byte2) { - for (int byte3 = 0x00; byte3 <= 0xFF; ++byte2) + for (int byte3 = 0x00; byte3 <= 0xFF; ++byte3) { // skip correct third byte if (0x80 <= byte3 and byte3 <= 0xBF) @@ -570,7 +570,7 @@ TEST_CASE("RFC 3629") { for (int byte2 = 0x80; byte2 <= 0xBF; ++byte2) { - for (int byte3 = 0x00; byte3 <= 0xFF; ++byte2) + for (int byte3 = 0x00; byte3 <= 0xFF; ++byte3) { // skip correct third byte if (0x80 <= byte3 and byte3 <= 0xBF) @@ -699,7 +699,7 @@ TEST_CASE("RFC 3629") { for (int byte2 = 0x90; byte2 <= 0xBF; ++byte2) { - for (int byte3 = 0x00; byte3 <= 0xFF; ++byte2) + for (int byte3 = 0x00; byte3 <= 0xFF; ++byte3) { // skip correct third byte if (0x80 <= byte3 and byte3 <= 0xBF) @@ -865,7 +865,7 @@ TEST_CASE("RFC 3629") { for (int byte2 = 0x80; byte2 <= 0xBF; ++byte2) { - for (int byte3 = 0x00; byte3 <= 0xFF; ++byte2) + for (int byte3 = 0x00; byte3 <= 0xFF; ++byte3) { // skip correct third byte if (0x80 <= byte3 and byte3 <= 0xBF) @@ -1029,7 +1029,7 @@ TEST_CASE("RFC 3629") { for (int byte2 = 0x80; byte2 <= 0x8F; ++byte2) { - for (int byte3 = 0x00; byte3 <= 0xFF; ++byte2) + for (int byte3 = 0x00; byte3 <= 0xFF; ++byte3) { // skip correct third byte if (0x80 <= byte3 and byte3 <= 0xBF) From b686cc6ad980dd5aa58978424b8554a368678fb6 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 23 Apr 2017 18:34:14 +0200 Subject: [PATCH 172/530] :hammer: removed #ifdefs --- test/src/unit-unicode.cpp | 36 ------------------------------------ 1 file changed, 36 deletions(-) diff --git a/test/src/unit-unicode.cpp b/test/src/unit-unicode.cpp index a0eef991..f2f2f83c 100644 --- a/test/src/unit-unicode.cpp +++ b/test/src/unit-unicode.cpp @@ -159,7 +159,6 @@ TEST_CASE("RFC 3629", "[hide]") } } -#ifdef WRONG_SECOND SECTION("ill-formed: wrong second byte") { for (int byte1 = 0xC2; byte1 <= 0xDF; ++byte1) @@ -180,7 +179,6 @@ TEST_CASE("RFC 3629", "[hide]") } } } -#endif } SECTION("UTF8-3 (xE0 xA0-BF UTF8-tail)") @@ -230,7 +228,6 @@ TEST_CASE("RFC 3629", "[hide]") } } -#ifdef WRONG_SECOND SECTION("ill-formed: wrong second byte") { for (int byte1 = 0xE0; byte1 <= 0xE0; ++byte1) @@ -255,9 +252,7 @@ TEST_CASE("RFC 3629", "[hide]") } } } -#endif -#ifdef WRONG_THIRD SECTION("ill-formed: wrong third byte") { for (int byte1 = 0xE0; byte1 <= 0xE0; ++byte1) @@ -282,7 +277,6 @@ TEST_CASE("RFC 3629", "[hide]") } } } -#endif } SECTION("UTF8-3 (xE1-xEC UTF8-tail UTF8-tail)") @@ -332,7 +326,6 @@ TEST_CASE("RFC 3629", "[hide]") } } -#ifdef WRONG_SECOND SECTION("ill-formed: wrong second byte") { for (int byte1 = 0xE1; byte1 <= 0xEC; ++byte1) @@ -357,9 +350,7 @@ TEST_CASE("RFC 3629", "[hide]") } } } -#endif -#ifdef WRONG_THIRD SECTION("ill-formed: wrong third byte") { for (int byte1 = 0xE1; byte1 <= 0xEC; ++byte1) @@ -384,7 +375,6 @@ TEST_CASE("RFC 3629", "[hide]") } } } -#endif } SECTION("UTF8-3 (xED x80-9F UTF8-tail)") @@ -434,7 +424,6 @@ TEST_CASE("RFC 3629", "[hide]") } } -#ifdef WRONG_SECOND SECTION("ill-formed: wrong second byte") { for (int byte1 = 0xED; byte1 <= 0xED; ++byte1) @@ -459,9 +448,7 @@ TEST_CASE("RFC 3629", "[hide]") } } } -#endif -#ifdef WRONG_THIRD SECTION("ill-formed: wrong third byte") { for (int byte1 = 0xED; byte1 <= 0xED; ++byte1) @@ -486,7 +473,6 @@ TEST_CASE("RFC 3629", "[hide]") } } } -#endif } SECTION("UTF8-3 (xEE-xEF UTF8-tail UTF8-tail)") @@ -536,7 +522,6 @@ TEST_CASE("RFC 3629", "[hide]") } } -#ifdef WRONG_SECOND SECTION("ill-formed: wrong second byte") { for (int byte1 = 0xEE; byte1 <= 0xEF; ++byte1) @@ -561,9 +546,7 @@ TEST_CASE("RFC 3629", "[hide]") } } } -#endif -#ifdef WRONG_THIRD SECTION("ill-formed: wrong third byte") { for (int byte1 = 0xEE; byte1 <= 0xEF; ++byte1) @@ -588,7 +571,6 @@ TEST_CASE("RFC 3629", "[hide]") } } } -#endif } SECTION("UTF8-4 (xF0 x90-BF UTF8-tail UTF8-tail)") @@ -661,7 +643,6 @@ TEST_CASE("RFC 3629", "[hide]") } } -#ifdef WRONG_SECOND SECTION("ill-formed: wrong second byte") { for (int byte1 = 0xF0; byte1 <= 0xF0; ++byte1) @@ -690,9 +671,7 @@ TEST_CASE("RFC 3629", "[hide]") } } } -#endif -#ifdef WRONG_THIRD SECTION("ill-formed: wrong third byte") { for (int byte1 = 0xF0; byte1 <= 0xF0; ++byte1) @@ -721,9 +700,7 @@ TEST_CASE("RFC 3629", "[hide]") } } } -#endif -#ifdef WRONG_FOURTH SECTION("ill-formed: wrong fourth byte") { for (int byte1 = 0xF0; byte1 <= 0xF0; ++byte1) @@ -754,7 +731,6 @@ TEST_CASE("RFC 3629", "[hide]") } } } -#endif } SECTION("UTF8-4 (xF1-F3 UTF8-tail UTF8-tail UTF8-tail)") @@ -827,7 +803,6 @@ TEST_CASE("RFC 3629", "[hide]") } } -#ifdef WRONG_SECOND SECTION("ill-formed: wrong second byte") { for (int byte1 = 0xF1; byte1 <= 0xF3; ++byte1) @@ -856,9 +831,7 @@ TEST_CASE("RFC 3629", "[hide]") } } } -#endif -#ifdef WRONG_THIRD SECTION("ill-formed: wrong third byte") { for (int byte1 = 0xF1; byte1 <= 0xF3; ++byte1) @@ -887,9 +860,7 @@ TEST_CASE("RFC 3629", "[hide]") } } } -#endif -#ifdef WRONG_FOURTH SECTION("ill-formed: wrong fourth byte") { for (int byte1 = 0xF1; byte1 <= 0xF3; ++byte1) @@ -918,7 +889,6 @@ TEST_CASE("RFC 3629", "[hide]") } } } -#endif } SECTION("UTF8-4 (xF4 x80-8F UTF8-tail UTF8-tail)") @@ -991,7 +961,6 @@ TEST_CASE("RFC 3629", "[hide]") } } -#ifdef WRONG_SECOND SECTION("ill-formed: wrong second byte") { for (int byte1 = 0xF4; byte1 <= 0xF4; ++byte1) @@ -1020,9 +989,7 @@ TEST_CASE("RFC 3629", "[hide]") } } } -#endif -#ifdef WRONG_THIRD SECTION("ill-formed: wrong third byte") { for (int byte1 = 0xF4; byte1 <= 0xF4; ++byte1) @@ -1051,9 +1018,7 @@ TEST_CASE("RFC 3629", "[hide]") } } } -#endif -#ifdef WRONG_FOURTH SECTION("ill-formed: wrong fourth byte") { for (int byte1 = 0xF4; byte1 <= 0xF4; ++byte1) @@ -1082,7 +1047,6 @@ TEST_CASE("RFC 3629", "[hide]") } } } -#endif } } From 01e05d89e9e15e37219ec40090b1e37f3ae75f90 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 23 Apr 2017 18:40:17 +0200 Subject: [PATCH 173/530] :hammer: fixed a compiler warning Default arguments are forbidden in lambdas. --- test/src/unit-unicode.cpp | 38 +++++++++++++++++++------------------- 1 file changed, 19 insertions(+), 19 deletions(-) diff --git a/test/src/unit-unicode.cpp b/test/src/unit-unicode.cpp index f2f2f83c..93df6808 100644 --- a/test/src/unit-unicode.cpp +++ b/test/src/unit-unicode.cpp @@ -34,6 +34,25 @@ using nlohmann::json; #include +std::string create_string(int byte1, int byte2 = -1, int byte3 = -1, int byte4 = -1) +{ + std::string result = "\"" + std::string(1, static_cast(byte1)); + if (byte2 != -1) + { + result += std::string(1, static_cast(byte2)); + } + if (byte3 != -1) + { + result += std::string(1, static_cast(byte3)); + } + if (byte4 != -1) + { + result += std::string(1, static_cast(byte4)); + } + result += "\""; + return result; +} + TEST_CASE("RFC 3629", "[hide]") { /* @@ -56,25 +75,6 @@ TEST_CASE("RFC 3629", "[hide]") UTF8-tail = %x80-BF */ - auto create_string = [](int byte1, int byte2 = -1, int byte3 = -1, int byte4 = -1) - { - std::string result = "\"" + std::string(1, static_cast(byte1)); - if (byte2 != -1) - { - result += std::string(1, static_cast(byte2)); - } - if (byte3 != -1) - { - result += std::string(1, static_cast(byte3)); - } - if (byte4 != -1) - { - result += std::string(1, static_cast(byte4)); - } - result += "\""; - return result; - }; - SECTION("ill-formed first byte") { for (int byte1 = 0x80; byte1 <= 0xC1; ++byte1) From 734297ff45bcc1474cfdc8e02ea1222f41d40890 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 23 Apr 2017 20:32:05 +0200 Subject: [PATCH 174/530] :hammer: cleanup --- README.md | 2 +- test/src/unit-unicode.cpp | 360 +++++++++----------------------------- 2 files changed, 82 insertions(+), 280 deletions(-) diff --git a/README.md b/README.md index 265ec06f..fbae1a79 100644 --- a/README.md +++ b/README.md @@ -897,7 +897,7 @@ $ make json_unit -Ctest $ ./test/json_unit "*" =============================================================================== -All tests passed (11203022 assertions in 48 test cases) +All tests passed (13391115 assertions in 49 test cases) ``` Alternatively, you can use [CMake](https://cmake.org) and run diff --git a/test/src/unit-unicode.cpp b/test/src/unit-unicode.cpp index 93df6808..7b94db67 100644 --- a/test/src/unit-unicode.cpp +++ b/test/src/unit-unicode.cpp @@ -34,23 +34,44 @@ using nlohmann::json; #include -std::string create_string(int byte1, int byte2 = -1, int byte3 = -1, int byte4 = -1) +// create and check a JSON string with up to four UTF-8 bytes +void check_utf8string(bool success_expected, int byte1, int byte2 = -1, int byte3 = -1, int byte4 = -1) { - std::string result = "\"" + std::string(1, static_cast(byte1)); + std::string json_string = "\""; + + CAPTURE(byte1); + json_string += std::string(1, static_cast(byte1)); + if (byte2 != -1) { - result += std::string(1, static_cast(byte2)); + CAPTURE(byte2); + json_string += std::string(1, static_cast(byte2)); } + if (byte3 != -1) { - result += std::string(1, static_cast(byte3)); + CAPTURE(byte3); + json_string += std::string(1, static_cast(byte3)); } + if (byte4 != -1) { - result += std::string(1, static_cast(byte4)); + CAPTURE(byte4); + json_string += std::string(1, static_cast(byte4)); + } + + json_string += "\""; + + CAPTURE(json_string); + + if (success_expected) + { + CHECK_NOTHROW(json::parse(json_string)); + } + else + { + CHECK_THROWS_AS(json::parse(json_string), json::parse_error); } - result += "\""; - return result; } TEST_CASE("RFC 3629", "[hide]") @@ -79,18 +100,12 @@ TEST_CASE("RFC 3629", "[hide]") { for (int byte1 = 0x80; byte1 <= 0xC1; ++byte1) { - const auto json_string = create_string(byte1); - CAPTURE(byte1); - CAPTURE(json_string); - CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + check_utf8string(false, byte1); } for (int byte1 = 0xF5; byte1 <= 0xFF; ++byte1) { - const auto json_string = create_string(byte1); - CAPTURE(byte1); - CAPTURE(json_string); - CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + check_utf8string(false, byte1); } } @@ -100,33 +115,29 @@ TEST_CASE("RFC 3629", "[hide]") { for (int byte1 = 0x00; byte1 <= 0x7F; ++byte1) { - const auto json_string = create_string(byte1); - CAPTURE(byte1); - CAPTURE(json_string); - // unescaped control characters are parse errors in JSON if (0x00 <= byte1 and byte1 <= 0x1F) { - CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + check_utf8string(false, byte1); continue; } // a single quote is a parse error in JSON if (byte1 == 0x22) { - CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + check_utf8string(false, byte1); continue; } // a single backslash is a parse error in JSON if (byte1 == 0x5C) { - CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + check_utf8string(false, byte1); continue; } // all other characters are OK - CHECK_NOTHROW(json::parse(json_string)); + check_utf8string(true, byte1); } } } @@ -139,11 +150,7 @@ TEST_CASE("RFC 3629", "[hide]") { for (int byte2 = 0x80; byte2 <= 0xBF; ++byte2) { - const auto json_string = create_string(byte1, byte2); - CAPTURE(byte1); - CAPTURE(byte2); - CAPTURE(json_string); - CHECK_NOTHROW(json::parse(json_string)); + check_utf8string(true, byte1, byte2); } } } @@ -152,10 +159,7 @@ TEST_CASE("RFC 3629", "[hide]") { for (int byte1 = 0xC2; byte1 <= 0xDF; ++byte1) { - const auto json_string = create_string(byte1); - CAPTURE(byte1); - CAPTURE(json_string); - CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + check_utf8string(false, byte1); } } @@ -171,11 +175,7 @@ TEST_CASE("RFC 3629", "[hide]") continue; } - const auto json_string = create_string(byte1, byte2); - CAPTURE(byte1); - CAPTURE(byte2); - CAPTURE(json_string); - CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + check_utf8string(false, byte1, byte2); } } } @@ -191,12 +191,7 @@ TEST_CASE("RFC 3629", "[hide]") { for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) { - const auto json_string = create_string(byte1, byte2, byte3); - CAPTURE(byte1); - CAPTURE(byte2); - CAPTURE(byte3); - CAPTURE(json_string); - CHECK_NOTHROW(json::parse(json_string)); + check_utf8string(true, byte1, byte2, byte3); } } } @@ -206,10 +201,7 @@ TEST_CASE("RFC 3629", "[hide]") { for (int byte1 = 0xE0; byte1 <= 0xE0; ++byte1) { - const auto json_string = create_string(byte1); - CAPTURE(byte1); - CAPTURE(json_string); - CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + check_utf8string(false, byte1); } } @@ -219,11 +211,7 @@ TEST_CASE("RFC 3629", "[hide]") { for (int byte2 = 0xA0; byte2 <= 0xBF; ++byte2) { - const auto json_string = create_string(byte1, byte2); - CAPTURE(byte1); - CAPTURE(byte2); - CAPTURE(json_string); - CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + check_utf8string(false, byte1, byte2); } } } @@ -242,12 +230,7 @@ TEST_CASE("RFC 3629", "[hide]") for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) { - const auto json_string = create_string(byte1, byte2, byte3); - CAPTURE(byte1); - CAPTURE(byte2); - CAPTURE(byte3); - CAPTURE(json_string); - CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + check_utf8string(false, byte1, byte2, byte3); } } } @@ -267,12 +250,7 @@ TEST_CASE("RFC 3629", "[hide]") continue; } - const auto json_string = create_string(byte1, byte2, byte3); - CAPTURE(byte1); - CAPTURE(byte2); - CAPTURE(byte3); - CAPTURE(json_string); - CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + check_utf8string(false, byte1, byte2, byte3); } } } @@ -289,12 +267,7 @@ TEST_CASE("RFC 3629", "[hide]") { for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) { - const auto json_string = create_string(byte1, byte2, byte3); - CAPTURE(byte1); - CAPTURE(byte2); - CAPTURE(byte3); - CAPTURE(json_string); - CHECK_NOTHROW(json::parse(json_string)); + check_utf8string(true, byte1, byte2, byte3); } } } @@ -304,10 +277,7 @@ TEST_CASE("RFC 3629", "[hide]") { for (int byte1 = 0xE1; byte1 <= 0xEC; ++byte1) { - const auto json_string = create_string(byte1); - CAPTURE(byte1); - CAPTURE(json_string); - CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + check_utf8string(false, byte1); } } @@ -317,11 +287,7 @@ TEST_CASE("RFC 3629", "[hide]") { for (int byte2 = 0x80; byte2 <= 0xBF; ++byte2) { - const auto json_string = create_string(byte1, byte2); - CAPTURE(byte1); - CAPTURE(byte2); - CAPTURE(json_string); - CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + check_utf8string(false, byte1, byte2); } } } @@ -340,12 +306,7 @@ TEST_CASE("RFC 3629", "[hide]") for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) { - const auto json_string = create_string(byte1, byte2, byte3); - CAPTURE(byte1); - CAPTURE(byte2); - CAPTURE(byte3); - CAPTURE(json_string); - CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + check_utf8string(false, byte1, byte2, byte3); } } } @@ -365,12 +326,7 @@ TEST_CASE("RFC 3629", "[hide]") continue; } - const auto json_string = create_string(byte1, byte2, byte3); - CAPTURE(byte1); - CAPTURE(byte2); - CAPTURE(byte3); - CAPTURE(json_string); - CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + check_utf8string(false, byte1, byte2, byte3); } } } @@ -387,12 +343,7 @@ TEST_CASE("RFC 3629", "[hide]") { for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) { - const auto json_string = create_string(byte1, byte2, byte3); - CAPTURE(byte1); - CAPTURE(byte2); - CAPTURE(byte3); - CAPTURE(json_string); - CHECK_NOTHROW(json::parse(json_string)); + check_utf8string(true, byte1, byte2, byte3); } } } @@ -402,10 +353,7 @@ TEST_CASE("RFC 3629", "[hide]") { for (int byte1 = 0xED; byte1 <= 0xED; ++byte1) { - const auto json_string = create_string(byte1); - CAPTURE(byte1); - CAPTURE(json_string); - CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + check_utf8string(false, byte1); } } @@ -415,11 +363,7 @@ TEST_CASE("RFC 3629", "[hide]") { for (int byte2 = 0x80; byte2 <= 0x9F; ++byte2) { - const auto json_string = create_string(byte1, byte2); - CAPTURE(byte1); - CAPTURE(byte2); - CAPTURE(json_string); - CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + check_utf8string(false, byte1, byte2); } } } @@ -438,12 +382,7 @@ TEST_CASE("RFC 3629", "[hide]") for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) { - const auto json_string = create_string(byte1, byte2, byte3); - CAPTURE(byte1); - CAPTURE(byte2); - CAPTURE(byte3); - CAPTURE(json_string); - CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + check_utf8string(false, byte1, byte2, byte3); } } } @@ -463,12 +402,7 @@ TEST_CASE("RFC 3629", "[hide]") continue; } - const auto json_string = create_string(byte1, byte2, byte3); - CAPTURE(byte1); - CAPTURE(byte2); - CAPTURE(byte3); - CAPTURE(json_string); - CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + check_utf8string(false, byte1, byte2, byte3); } } } @@ -485,12 +419,7 @@ TEST_CASE("RFC 3629", "[hide]") { for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) { - const auto json_string = create_string(byte1, byte2, byte3); - CAPTURE(byte1); - CAPTURE(byte2); - CAPTURE(byte3); - CAPTURE(json_string); - CHECK_NOTHROW(json::parse(json_string)); + check_utf8string(true, byte1, byte2, byte3); } } } @@ -500,10 +429,7 @@ TEST_CASE("RFC 3629", "[hide]") { for (int byte1 = 0xEE; byte1 <= 0xEF; ++byte1) { - const auto json_string = create_string(byte1); - CAPTURE(byte1); - CAPTURE(json_string); - CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + check_utf8string(false, byte1); } } @@ -513,11 +439,7 @@ TEST_CASE("RFC 3629", "[hide]") { for (int byte2 = 0x80; byte2 <= 0xBF; ++byte2) { - const auto json_string = create_string(byte1, byte2); - CAPTURE(byte1); - CAPTURE(byte2); - CAPTURE(json_string); - CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + check_utf8string(false, byte1, byte2); } } } @@ -536,12 +458,7 @@ TEST_CASE("RFC 3629", "[hide]") for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) { - const auto json_string = create_string(byte1, byte2, byte3); - CAPTURE(byte1); - CAPTURE(byte2); - CAPTURE(byte3); - CAPTURE(json_string); - CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + check_utf8string(false, byte1, byte2, byte3); } } } @@ -561,12 +478,7 @@ TEST_CASE("RFC 3629", "[hide]") continue; } - const auto json_string = create_string(byte1, byte2, byte3); - CAPTURE(byte1); - CAPTURE(byte2); - CAPTURE(byte3); - CAPTURE(json_string); - CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + check_utf8string(false, byte1, byte2, byte3); } } } @@ -585,13 +497,7 @@ TEST_CASE("RFC 3629", "[hide]") { for (int byte4 = 0x80; byte4 <= 0xBF; ++byte4) { - const auto json_string = create_string(byte1, byte2, byte3, byte4); - CAPTURE(byte1); - CAPTURE(byte2); - CAPTURE(byte3); - CAPTURE(byte4); - CAPTURE(json_string); - CHECK_NOTHROW(json::parse(json_string)); + check_utf8string(true, byte1, byte2, byte3, byte4); } } } @@ -602,10 +508,7 @@ TEST_CASE("RFC 3629", "[hide]") { for (int byte1 = 0xF0; byte1 <= 0xF0; ++byte1) { - const auto json_string = create_string(byte1); - CAPTURE(byte1); - CAPTURE(json_string); - CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + check_utf8string(false, byte1); } } @@ -615,11 +518,7 @@ TEST_CASE("RFC 3629", "[hide]") { for (int byte2 = 0x90; byte2 <= 0xBF; ++byte2) { - const auto json_string = create_string(byte1, byte2); - CAPTURE(byte1); - CAPTURE(byte2); - CAPTURE(json_string); - CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + check_utf8string(false, byte1, byte2); } } } @@ -632,12 +531,7 @@ TEST_CASE("RFC 3629", "[hide]") { for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) { - const auto json_string = create_string(byte1, byte2, byte3); - CAPTURE(byte1); - CAPTURE(byte2); - CAPTURE(byte3); - CAPTURE(json_string); - CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + check_utf8string(false, byte1, byte2, byte3); } } } @@ -659,13 +553,7 @@ TEST_CASE("RFC 3629", "[hide]") { for (int byte4 = 0x80; byte4 <= 0xBF; ++byte4) { - const auto json_string = create_string(byte1, byte2, byte3, byte4); - CAPTURE(byte1); - CAPTURE(byte2); - CAPTURE(byte3); - CAPTURE(byte4); - CAPTURE(json_string); - CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + check_utf8string(false, byte1, byte2, byte3, byte4); } } } @@ -688,13 +576,7 @@ TEST_CASE("RFC 3629", "[hide]") for (int byte4 = 0x80; byte4 <= 0xBF; ++byte4) { - const auto json_string = create_string(byte1, byte2, byte3, byte4); - CAPTURE(byte1); - CAPTURE(byte2); - CAPTURE(byte3); - CAPTURE(byte4); - CAPTURE(json_string); - CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + check_utf8string(false, byte1, byte2, byte3, byte4); } } } @@ -709,24 +591,16 @@ TEST_CASE("RFC 3629", "[hide]") { for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) { - { for (int byte4 = 0x00; byte4 <= 0xFF; ++byte4) { - // skip correct second byte + // skip fourth second byte if (0x80 <= byte3 and byte3 <= 0xBF) { continue; } - const auto json_string = create_string(byte1, byte2, byte3, byte4); - CAPTURE(byte1); - CAPTURE(byte2); - CAPTURE(byte3); - CAPTURE(byte4); - CAPTURE(json_string); - CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + check_utf8string(false, byte1, byte2, byte3, byte4); } - } } } } @@ -745,13 +619,7 @@ TEST_CASE("RFC 3629", "[hide]") { for (int byte4 = 0x80; byte4 <= 0xBF; ++byte4) { - const auto json_string = create_string(byte1, byte2, byte3, byte4); - CAPTURE(byte1); - CAPTURE(byte2); - CAPTURE(byte3); - CAPTURE(byte4); - CAPTURE(json_string); - CHECK_NOTHROW(json::parse(json_string)); + check_utf8string(true, byte1, byte2, byte3, byte4); } } } @@ -762,10 +630,7 @@ TEST_CASE("RFC 3629", "[hide]") { for (int byte1 = 0xF1; byte1 <= 0xF3; ++byte1) { - const auto json_string = create_string(byte1); - CAPTURE(byte1); - CAPTURE(json_string); - CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + check_utf8string(false, byte1); } } @@ -775,11 +640,7 @@ TEST_CASE("RFC 3629", "[hide]") { for (int byte2 = 0x80; byte2 <= 0xBF; ++byte2) { - const auto json_string = create_string(byte1, byte2); - CAPTURE(byte1); - CAPTURE(byte2); - CAPTURE(json_string); - CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + check_utf8string(false, byte1, byte2); } } } @@ -792,12 +653,7 @@ TEST_CASE("RFC 3629", "[hide]") { for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) { - const auto json_string = create_string(byte1, byte2, byte3); - CAPTURE(byte1); - CAPTURE(byte2); - CAPTURE(byte3); - CAPTURE(json_string); - CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + check_utf8string(false, byte1, byte2, byte3); } } } @@ -819,13 +675,7 @@ TEST_CASE("RFC 3629", "[hide]") { for (int byte4 = 0x80; byte4 <= 0xBF; ++byte4) { - const auto json_string = create_string(byte1, byte2, byte3, byte4); - CAPTURE(byte1); - CAPTURE(byte2); - CAPTURE(byte3); - CAPTURE(byte4); - CAPTURE(json_string); - CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + check_utf8string(false, byte1, byte2, byte3, byte4); } } } @@ -848,13 +698,7 @@ TEST_CASE("RFC 3629", "[hide]") for (int byte4 = 0x80; byte4 <= 0xBF; ++byte4) { - const auto json_string = create_string(byte1, byte2, byte3, byte4); - CAPTURE(byte1); - CAPTURE(byte2); - CAPTURE(byte3); - CAPTURE(byte4); - CAPTURE(json_string); - CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + check_utf8string(false, byte1, byte2, byte3, byte4); } } } @@ -871,19 +715,13 @@ TEST_CASE("RFC 3629", "[hide]") { for (int byte4 = 0x00; byte4 <= 0xFF; ++byte4) { - // skip correct second byte + // skip correct fourth byte if (0x80 <= byte3 and byte3 <= 0xBF) { continue; } - const auto json_string = create_string(byte1, byte2, byte3, byte4); - CAPTURE(byte1); - CAPTURE(byte2); - CAPTURE(byte3); - CAPTURE(byte4); - CAPTURE(json_string); - CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + check_utf8string(false, byte1, byte2, byte3, byte4); } } } @@ -903,13 +741,7 @@ TEST_CASE("RFC 3629", "[hide]") { for (int byte4 = 0x80; byte4 <= 0xBF; ++byte4) { - const auto json_string = create_string(byte1, byte2, byte3, byte4); - CAPTURE(byte1); - CAPTURE(byte2); - CAPTURE(byte3); - CAPTURE(byte4); - CAPTURE(json_string); - CHECK_NOTHROW(json::parse(json_string)); + check_utf8string(true, byte1, byte2, byte3, byte4); } } } @@ -920,10 +752,7 @@ TEST_CASE("RFC 3629", "[hide]") { for (int byte1 = 0xF4; byte1 <= 0xF4; ++byte1) { - const auto json_string = create_string(byte1); - CAPTURE(byte1); - CAPTURE(json_string); - CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + check_utf8string(false, byte1); } } @@ -933,11 +762,7 @@ TEST_CASE("RFC 3629", "[hide]") { for (int byte2 = 0x80; byte2 <= 0x8F; ++byte2) { - const auto json_string = create_string(byte1, byte2); - CAPTURE(byte1); - CAPTURE(byte2); - CAPTURE(json_string); - CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + check_utf8string(false, byte1, byte2); } } } @@ -950,12 +775,7 @@ TEST_CASE("RFC 3629", "[hide]") { for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) { - const auto json_string = create_string(byte1, byte2, byte3); - CAPTURE(byte1); - CAPTURE(byte2); - CAPTURE(byte3); - CAPTURE(json_string); - CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + check_utf8string(false, byte1, byte2, byte3); } } } @@ -977,13 +797,7 @@ TEST_CASE("RFC 3629", "[hide]") { for (int byte4 = 0x80; byte4 <= 0xBF; ++byte4) { - const auto json_string = create_string(byte1, byte2, byte3, byte4); - CAPTURE(byte1); - CAPTURE(byte2); - CAPTURE(byte3); - CAPTURE(byte4); - CAPTURE(json_string); - CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + check_utf8string(false, byte1, byte2, byte3, byte4); } } } @@ -1006,13 +820,7 @@ TEST_CASE("RFC 3629", "[hide]") for (int byte4 = 0x80; byte4 <= 0xBF; ++byte4) { - const auto json_string = create_string(byte1, byte2, byte3, byte4); - CAPTURE(byte1); - CAPTURE(byte2); - CAPTURE(byte3); - CAPTURE(byte4); - CAPTURE(json_string); - CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + check_utf8string(false, byte1, byte2, byte3, byte4); } } } @@ -1029,19 +837,13 @@ TEST_CASE("RFC 3629", "[hide]") { for (int byte4 = 0x00; byte4 <= 0xFF; ++byte4) { - // skip correct second byte + // skip correct fourth byte if (0x80 <= byte3 and byte3 <= 0xBF) { continue; } - const auto json_string = create_string(byte1, byte2, byte3, byte4); - CAPTURE(byte1); - CAPTURE(byte2); - CAPTURE(byte3); - CAPTURE(byte4); - CAPTURE(json_string); - CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + check_utf8string(false, byte1, byte2, byte3, byte4); } } } From 2df8f12c090106ad32df99480f7a19e936c5f143 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 23 Apr 2017 20:51:29 +0200 Subject: [PATCH 175/530] :checkered_flag: another fix for MSVC #464 needed for VS2015.3 with /std:c++latest --- src/json.hpp | 2 +- src/json.hpp.re2c | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 49dfd013..8a8b876a 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -3730,7 +3730,7 @@ class basic_json #ifndef _MSC_VER // fix for issue #167 operator<< ambiguity under VS2015 and not std::is_same>::value #endif -#if defined(_MSC_VER) && defined(_HAS_CXX17) && _HAS_CXX17 == 1 // fix for issue #464 +#if defined(_MSC_VER) && _MSC_VER >1900 && defined(_HAS_CXX17) && _HAS_CXX17 == 1 // fix for issue #464 and not std::is_same::value #endif , int >::type = 0 > diff --git a/src/json.hpp.re2c b/src/json.hpp.re2c index 48c8eb0d..5dc86d1e 100644 --- a/src/json.hpp.re2c +++ b/src/json.hpp.re2c @@ -3730,7 +3730,7 @@ class basic_json #ifndef _MSC_VER // fix for issue #167 operator<< ambiguity under VS2015 and not std::is_same>::value #endif -#if defined(_MSC_VER) && defined(_HAS_CXX17) && _HAS_CXX17 == 1 // fix for issue #464 +#if defined(_MSC_VER) && _MSC_VER >1900 && defined(_HAS_CXX17) && _HAS_CXX17 == 1 // fix for issue #464 and not std::is_same::value #endif , int >::type = 0 > From c42273d2a052545b34f1d10d6a78696c2e641b7b Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 23 Apr 2017 21:46:30 +0200 Subject: [PATCH 176/530] :memo: fixed documentation #565 The documentation mismatched the actual code. --- README.md | 1 + doc/index.md | 4 ++-- 2 files changed, 3 insertions(+), 2 deletions(-) diff --git a/README.md b/README.md index 265ec06f..fc6dde9b 100644 --- a/README.md +++ b/README.md @@ -835,6 +835,7 @@ I deeply appreciate the help of the following people. - [Krzysztof Woś](https://github.com/krzysztofwos) made exceptions more visible. - [ftillier](https://github.com/ftillier) fixed a compiler warning. - [tinloaf](https://github.com/tinloaf) made sure all pushed warnings are properly popped. +- [Fytch](https://github.com/Fytch) found a bug in the documentation. Thanks a lot for helping out! Please [let me know](mailto:mail@nlohmann.me) if I forgot someone. diff --git a/doc/index.md b/doc/index.md index 9493412e..5392f4d3 100644 --- a/doc/index.md +++ b/doc/index.md @@ -254,21 +254,21 @@ The container functions known from STL have been extended to support the differe lookup `find` - @link nlohmann::basic_json::find `find` @endlink (returns `end()`) @link nlohmann::basic_json::find `find` @endlink @link nlohmann::basic_json::find `find` @endlink (returns `end()`) @link nlohmann::basic_json::find `find` @endlink (returns `end()`) @link nlohmann::basic_json::find `find` @endlink (returns `end()`) @link nlohmann::basic_json::find `find` @endlink (returns `end()`) + @link nlohmann::basic_json::find `find` @endlink (returns `end()`) `count` - @link nlohmann::basic_json::count `count` @endlink (returns `0`) @link nlohmann::basic_json::count `count` @endlink @link nlohmann::basic_json::count `count` @endlink (returns `0`) @link nlohmann::basic_json::count `count` @endlink (returns `0`) @link nlohmann::basic_json::count `count` @endlink (returns `0`) @link nlohmann::basic_json::count `count` @endlink (returns `0`) + @link nlohmann::basic_json::count `count` @endlink (returns `0`) From 6d2c0a79287c8013f16c2674d37f52a4ccd93be6 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 23 Apr 2017 22:54:21 +0200 Subject: [PATCH 177/530] :white_check_mark: added more Unicode test cases --- test/src/unit-unicode.cpp | 1237 +++++++++++++++++++------------------ 1 file changed, 631 insertions(+), 606 deletions(-) diff --git a/test/src/unit-unicode.cpp b/test/src/unit-unicode.cpp index 7b94db67..67e97346 100644 --- a/test/src/unit-unicode.cpp +++ b/test/src/unit-unicode.cpp @@ -74,523 +74,525 @@ void check_utf8string(bool success_expected, int byte1, int byte2 = -1, int byte } } -TEST_CASE("RFC 3629", "[hide]") +TEST_CASE("Unicode", "[hide]") { - /* - RFC 3629 describes in Sect. 4 the syntax of UTF-8 byte sequences as - follows: - - A UTF-8 string is a sequence of octets representing a sequence of UCS - characters. An octet sequence is valid UTF-8 only if it matches the - following syntax, which is derived from the rules for encoding UTF-8 - and is expressed in the ABNF of [RFC2234]. - - UTF8-octets = *( UTF8-char ) - UTF8-char = UTF8-1 / UTF8-2 / UTF8-3 / UTF8-4 - UTF8-1 = %x00-7F - UTF8-2 = %xC2-DF UTF8-tail - UTF8-3 = %xE0 %xA0-BF UTF8-tail / %xE1-EC 2( UTF8-tail ) / - %xED %x80-9F UTF8-tail / %xEE-EF 2( UTF8-tail ) - UTF8-4 = %xF0 %x90-BF 2( UTF8-tail ) / %xF1-F3 3( UTF8-tail ) / - %xF4 %x80-8F 2( UTF8-tail ) - UTF8-tail = %x80-BF - */ - - SECTION("ill-formed first byte") + SECTION("RFC 3629") { - for (int byte1 = 0x80; byte1 <= 0xC1; ++byte1) - { - check_utf8string(false, byte1); - } + /* + RFC 3629 describes in Sect. 4 the syntax of UTF-8 byte sequences as + follows: - for (int byte1 = 0xF5; byte1 <= 0xFF; ++byte1) - { - check_utf8string(false, byte1); - } - } + A UTF-8 string is a sequence of octets representing a sequence of UCS + characters. An octet sequence is valid UTF-8 only if it matches the + following syntax, which is derived from the rules for encoding UTF-8 + and is expressed in the ABNF of [RFC2234]. - SECTION("UTF8-1 (x00-x7F)") - { - SECTION("well-formed") + UTF8-octets = *( UTF8-char ) + UTF8-char = UTF8-1 / UTF8-2 / UTF8-3 / UTF8-4 + UTF8-1 = %x00-7F + UTF8-2 = %xC2-DF UTF8-tail + UTF8-3 = %xE0 %xA0-BF UTF8-tail / %xE1-EC 2( UTF8-tail ) / + %xED %x80-9F UTF8-tail / %xEE-EF 2( UTF8-tail ) + UTF8-4 = %xF0 %x90-BF 2( UTF8-tail ) / %xF1-F3 3( UTF8-tail ) / + %xF4 %x80-8F 2( UTF8-tail ) + UTF8-tail = %x80-BF + */ + + SECTION("ill-formed first byte") { - for (int byte1 = 0x00; byte1 <= 0x7F; ++byte1) + for (int byte1 = 0x80; byte1 <= 0xC1; ++byte1) { - // unescaped control characters are parse errors in JSON - if (0x00 <= byte1 and byte1 <= 0x1F) + check_utf8string(false, byte1); + } + + for (int byte1 = 0xF5; byte1 <= 0xFF; ++byte1) + { + check_utf8string(false, byte1); + } + } + + SECTION("UTF8-1 (x00-x7F)") + { + SECTION("well-formed") + { + for (int byte1 = 0x00; byte1 <= 0x7F; ++byte1) + { + // unescaped control characters are parse errors in JSON + if (0x00 <= byte1 and byte1 <= 0x1F) + { + check_utf8string(false, byte1); + continue; + } + + // a single quote is a parse error in JSON + if (byte1 == 0x22) + { + check_utf8string(false, byte1); + continue; + } + + // a single backslash is a parse error in JSON + if (byte1 == 0x5C) + { + check_utf8string(false, byte1); + continue; + } + + // all other characters are OK + check_utf8string(true, byte1); + } + } + } + + SECTION("UTF8-2 (xC2-xDF UTF8-tail)") + { + SECTION("well-formed") + { + for (int byte1 = 0xC2; byte1 <= 0xDF; ++byte1) + { + for (int byte2 = 0x80; byte2 <= 0xBF; ++byte2) + { + check_utf8string(true, byte1, byte2); + } + } + } + + SECTION("ill-formed: missing second byte") + { + for (int byte1 = 0xC2; byte1 <= 0xDF; ++byte1) { check_utf8string(false, byte1); - continue; } + } - // a single quote is a parse error in JSON - if (byte1 == 0x22) + SECTION("ill-formed: wrong second byte") + { + for (int byte1 = 0xC2; byte1 <= 0xDF; ++byte1) + { + for (int byte2 = 0x00; byte2 <= 0xFF; ++byte2) + { + // skip correct second byte + if (0x80 <= byte2 and byte2 <= 0xBF) + { + continue; + } + + check_utf8string(false, byte1, byte2); + } + } + } + } + + SECTION("UTF8-3 (xE0 xA0-BF UTF8-tail)") + { + SECTION("well-formed") + { + for (int byte1 = 0xE0; byte1 <= 0xE0; ++byte1) + { + for (int byte2 = 0xA0; byte2 <= 0xBF; ++byte2) + { + for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) + { + check_utf8string(true, byte1, byte2, byte3); + } + } + } + } + + SECTION("ill-formed: missing second byte") + { + for (int byte1 = 0xE0; byte1 <= 0xE0; ++byte1) { check_utf8string(false, byte1); - continue; } + } - // a single backslash is a parse error in JSON - if (byte1 == 0x5C) + SECTION("ill-formed: missing third byte") + { + for (int byte1 = 0xE0; byte1 <= 0xE0; ++byte1) + { + for (int byte2 = 0xA0; byte2 <= 0xBF; ++byte2) + { + check_utf8string(false, byte1, byte2); + } + } + } + + SECTION("ill-formed: wrong second byte") + { + for (int byte1 = 0xE0; byte1 <= 0xE0; ++byte1) + { + for (int byte2 = 0x00; byte2 <= 0xFF; ++byte2) + { + // skip correct second byte + if (0xA0 <= byte2 and byte2 <= 0xBF) + { + continue; + } + + for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) + { + check_utf8string(false, byte1, byte2, byte3); + } + } + } + } + + SECTION("ill-formed: wrong third byte") + { + for (int byte1 = 0xE0; byte1 <= 0xE0; ++byte1) + { + for (int byte2 = 0xA0; byte2 <= 0xBF; ++byte2) + { + for (int byte3 = 0x00; byte3 <= 0xFF; ++byte3) + { + // skip correct third byte + if (0x80 <= byte3 and byte3 <= 0xBF) + { + continue; + } + + check_utf8string(false, byte1, byte2, byte3); + } + } + } + } + } + + SECTION("UTF8-3 (xE1-xEC UTF8-tail UTF8-tail)") + { + SECTION("well-formed") + { + for (int byte1 = 0xE1; byte1 <= 0xEC; ++byte1) + { + for (int byte2 = 0x80; byte2 <= 0xBF; ++byte2) + { + for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) + { + check_utf8string(true, byte1, byte2, byte3); + } + } + } + } + + SECTION("ill-formed: missing second byte") + { + for (int byte1 = 0xE1; byte1 <= 0xEC; ++byte1) { check_utf8string(false, byte1); - continue; - } - - // all other characters are OK - check_utf8string(true, byte1); - } - } - } - - SECTION("UTF8-2 (xC2-xDF UTF8-tail)") - { - SECTION("well-formed") - { - for (int byte1 = 0xC2; byte1 <= 0xDF; ++byte1) - { - for (int byte2 = 0x80; byte2 <= 0xBF; ++byte2) - { - check_utf8string(true, byte1, byte2); } } - } - SECTION("ill-formed: missing second byte") - { - for (int byte1 = 0xC2; byte1 <= 0xDF; ++byte1) + SECTION("ill-formed: missing third byte") { - check_utf8string(false, byte1); - } - } - - SECTION("ill-formed: wrong second byte") - { - for (int byte1 = 0xC2; byte1 <= 0xDF; ++byte1) - { - for (int byte2 = 0x00; byte2 <= 0xFF; ++byte2) + for (int byte1 = 0xE1; byte1 <= 0xEC; ++byte1) { - // skip correct second byte - if (0x80 <= byte2 and byte2 <= 0xBF) + for (int byte2 = 0x80; byte2 <= 0xBF; ++byte2) { - continue; - } - - check_utf8string(false, byte1, byte2); - } - } - } - } - - SECTION("UTF8-3 (xE0 xA0-BF UTF8-tail)") - { - SECTION("well-formed") - { - for (int byte1 = 0xE0; byte1 <= 0xE0; ++byte1) - { - for (int byte2 = 0xA0; byte2 <= 0xBF; ++byte2) - { - for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) - { - check_utf8string(true, byte1, byte2, byte3); + check_utf8string(false, byte1, byte2); } } } - } - SECTION("ill-formed: missing second byte") - { - for (int byte1 = 0xE0; byte1 <= 0xE0; ++byte1) + SECTION("ill-formed: wrong second byte") { - check_utf8string(false, byte1); - } - } - - SECTION("ill-formed: missing third byte") - { - for (int byte1 = 0xE0; byte1 <= 0xE0; ++byte1) - { - for (int byte2 = 0xA0; byte2 <= 0xBF; ++byte2) + for (int byte1 = 0xE1; byte1 <= 0xEC; ++byte1) { - check_utf8string(false, byte1, byte2); - } - } - } - - SECTION("ill-formed: wrong second byte") - { - for (int byte1 = 0xE0; byte1 <= 0xE0; ++byte1) - { - for (int byte2 = 0x00; byte2 <= 0xFF; ++byte2) - { - // skip correct second byte - if (0xA0 <= byte2 and byte2 <= 0xBF) + for (int byte2 = 0x00; byte2 <= 0xFF; ++byte2) { - continue; - } - - for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) - { - check_utf8string(false, byte1, byte2, byte3); - } - } - } - } - - SECTION("ill-formed: wrong third byte") - { - for (int byte1 = 0xE0; byte1 <= 0xE0; ++byte1) - { - for (int byte2 = 0xA0; byte2 <= 0xBF; ++byte2) - { - for (int byte3 = 0x00; byte3 <= 0xFF; ++byte3) - { - // skip correct third byte - if (0x80 <= byte3 and byte3 <= 0xBF) + // skip correct second byte + if (0x80 <= byte2 and byte2 <= 0xBF) { continue; } - check_utf8string(false, byte1, byte2, byte3); + for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) + { + check_utf8string(false, byte1, byte2, byte3); + } } } } - } - } - SECTION("UTF8-3 (xE1-xEC UTF8-tail UTF8-tail)") - { - SECTION("well-formed") - { - for (int byte1 = 0xE1; byte1 <= 0xEC; ++byte1) + SECTION("ill-formed: wrong third byte") { - for (int byte2 = 0x80; byte2 <= 0xBF; ++byte2) + for (int byte1 = 0xE1; byte1 <= 0xEC; ++byte1) { - for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) + for (int byte2 = 0x80; byte2 <= 0xBF; ++byte2) { - check_utf8string(true, byte1, byte2, byte3); + for (int byte3 = 0x00; byte3 <= 0xFF; ++byte3) + { + // skip correct third byte + if (0x80 <= byte3 and byte3 <= 0xBF) + { + continue; + } + + check_utf8string(false, byte1, byte2, byte3); + } } } } } - SECTION("ill-formed: missing second byte") + SECTION("UTF8-3 (xED x80-9F UTF8-tail)") { - for (int byte1 = 0xE1; byte1 <= 0xEC; ++byte1) + SECTION("well-formed") { - check_utf8string(false, byte1); - } - } - - SECTION("ill-formed: missing third byte") - { - for (int byte1 = 0xE1; byte1 <= 0xEC; ++byte1) - { - for (int byte2 = 0x80; byte2 <= 0xBF; ++byte2) + for (int byte1 = 0xED; byte1 <= 0xED; ++byte1) { - check_utf8string(false, byte1, byte2); - } - } - } - - SECTION("ill-formed: wrong second byte") - { - for (int byte1 = 0xE1; byte1 <= 0xEC; ++byte1) - { - for (int byte2 = 0x00; byte2 <= 0xFF; ++byte2) - { - // skip correct second byte - if (0x80 <= byte2 and byte2 <= 0xBF) + for (int byte2 = 0x80; byte2 <= 0x9F; ++byte2) { - continue; - } - - for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) - { - check_utf8string(false, byte1, byte2, byte3); + for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) + { + check_utf8string(true, byte1, byte2, byte3); + } } } } - } - SECTION("ill-formed: wrong third byte") - { - for (int byte1 = 0xE1; byte1 <= 0xEC; ++byte1) + SECTION("ill-formed: missing second byte") { - for (int byte2 = 0x80; byte2 <= 0xBF; ++byte2) + for (int byte1 = 0xED; byte1 <= 0xED; ++byte1) { - for (int byte3 = 0x00; byte3 <= 0xFF; ++byte3) + check_utf8string(false, byte1); + } + } + + SECTION("ill-formed: missing third byte") + { + for (int byte1 = 0xED; byte1 <= 0xED; ++byte1) + { + for (int byte2 = 0x80; byte2 <= 0x9F; ++byte2) { - // skip correct third byte - if (0x80 <= byte3 and byte3 <= 0xBF) + check_utf8string(false, byte1, byte2); + } + } + } + + SECTION("ill-formed: wrong second byte") + { + for (int byte1 = 0xED; byte1 <= 0xED; ++byte1) + { + for (int byte2 = 0x00; byte2 <= 0xFF; ++byte2) + { + // skip correct second byte + if (0x80 <= byte2 and byte2 <= 0x9F) { continue; } - check_utf8string(false, byte1, byte2, byte3); + for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) + { + check_utf8string(false, byte1, byte2, byte3); + } } } } - } - } - SECTION("UTF8-3 (xED x80-9F UTF8-tail)") - { - SECTION("well-formed") - { - for (int byte1 = 0xED; byte1 <= 0xED; ++byte1) + SECTION("ill-formed: wrong third byte") { - for (int byte2 = 0x80; byte2 <= 0x9F; ++byte2) + for (int byte1 = 0xED; byte1 <= 0xED; ++byte1) { - for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) + for (int byte2 = 0x80; byte2 <= 0x9F; ++byte2) { - check_utf8string(true, byte1, byte2, byte3); + for (int byte3 = 0x00; byte3 <= 0xFF; ++byte3) + { + // skip correct third byte + if (0x80 <= byte3 and byte3 <= 0xBF) + { + continue; + } + + check_utf8string(false, byte1, byte2, byte3); + } } } } } - SECTION("ill-formed: missing second byte") + SECTION("UTF8-3 (xEE-xEF UTF8-tail UTF8-tail)") { - for (int byte1 = 0xED; byte1 <= 0xED; ++byte1) + SECTION("well-formed") { - check_utf8string(false, byte1); - } - } - - SECTION("ill-formed: missing third byte") - { - for (int byte1 = 0xED; byte1 <= 0xED; ++byte1) - { - for (int byte2 = 0x80; byte2 <= 0x9F; ++byte2) + for (int byte1 = 0xEE; byte1 <= 0xEF; ++byte1) { - check_utf8string(false, byte1, byte2); - } - } - } - - SECTION("ill-formed: wrong second byte") - { - for (int byte1 = 0xED; byte1 <= 0xED; ++byte1) - { - for (int byte2 = 0x00; byte2 <= 0xFF; ++byte2) - { - // skip correct second byte - if (0x80 <= byte2 and byte2 <= 0x9F) + for (int byte2 = 0x80; byte2 <= 0xBF; ++byte2) { - continue; - } - - for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) - { - check_utf8string(false, byte1, byte2, byte3); + for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) + { + check_utf8string(true, byte1, byte2, byte3); + } } } } - } - SECTION("ill-formed: wrong third byte") - { - for (int byte1 = 0xED; byte1 <= 0xED; ++byte1) + SECTION("ill-formed: missing second byte") { - for (int byte2 = 0x80; byte2 <= 0x9F; ++byte2) + for (int byte1 = 0xEE; byte1 <= 0xEF; ++byte1) { - for (int byte3 = 0x00; byte3 <= 0xFF; ++byte3) + check_utf8string(false, byte1); + } + } + + SECTION("ill-formed: missing third byte") + { + for (int byte1 = 0xEE; byte1 <= 0xEF; ++byte1) + { + for (int byte2 = 0x80; byte2 <= 0xBF; ++byte2) { - // skip correct third byte - if (0x80 <= byte3 and byte3 <= 0xBF) + check_utf8string(false, byte1, byte2); + } + } + } + + SECTION("ill-formed: wrong second byte") + { + for (int byte1 = 0xEE; byte1 <= 0xEF; ++byte1) + { + for (int byte2 = 0x00; byte2 <= 0xFF; ++byte2) + { + // skip correct second byte + if (0x80 <= byte2 and byte2 <= 0xBF) { continue; } - check_utf8string(false, byte1, byte2, byte3); + for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) + { + check_utf8string(false, byte1, byte2, byte3); + } } } } - } - } - SECTION("UTF8-3 (xEE-xEF UTF8-tail UTF8-tail)") - { - SECTION("well-formed") - { - for (int byte1 = 0xEE; byte1 <= 0xEF; ++byte1) + SECTION("ill-formed: wrong third byte") { - for (int byte2 = 0x80; byte2 <= 0xBF; ++byte2) + for (int byte1 = 0xEE; byte1 <= 0xEF; ++byte1) { - for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) + for (int byte2 = 0x80; byte2 <= 0xBF; ++byte2) { - check_utf8string(true, byte1, byte2, byte3); + for (int byte3 = 0x00; byte3 <= 0xFF; ++byte3) + { + // skip correct third byte + if (0x80 <= byte3 and byte3 <= 0xBF) + { + continue; + } + + check_utf8string(false, byte1, byte2, byte3); + } } } } } - SECTION("ill-formed: missing second byte") + SECTION("UTF8-4 (xF0 x90-BF UTF8-tail UTF8-tail)") { - for (int byte1 = 0xEE; byte1 <= 0xEF; ++byte1) + SECTION("well-formed") { - check_utf8string(false, byte1); - } - } - - SECTION("ill-formed: missing third byte") - { - for (int byte1 = 0xEE; byte1 <= 0xEF; ++byte1) - { - for (int byte2 = 0x80; byte2 <= 0xBF; ++byte2) + for (int byte1 = 0xF0; byte1 <= 0xF0; ++byte1) { - check_utf8string(false, byte1, byte2); - } - } - } - - SECTION("ill-formed: wrong second byte") - { - for (int byte1 = 0xEE; byte1 <= 0xEF; ++byte1) - { - for (int byte2 = 0x00; byte2 <= 0xFF; ++byte2) - { - // skip correct second byte - if (0x80 <= byte2 and byte2 <= 0xBF) + for (int byte2 = 0x90; byte2 <= 0xBF; ++byte2) { - continue; - } - - for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) - { - check_utf8string(false, byte1, byte2, byte3); + for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) + { + for (int byte4 = 0x80; byte4 <= 0xBF; ++byte4) + { + check_utf8string(true, byte1, byte2, byte3, byte4); + } + } } } } - } - SECTION("ill-formed: wrong third byte") - { - for (int byte1 = 0xEE; byte1 <= 0xEF; ++byte1) + SECTION("ill-formed: missing second byte") { - for (int byte2 = 0x80; byte2 <= 0xBF; ++byte2) + for (int byte1 = 0xF0; byte1 <= 0xF0; ++byte1) { - for (int byte3 = 0x00; byte3 <= 0xFF; ++byte3) + check_utf8string(false, byte1); + } + } + + SECTION("ill-formed: missing third byte") + { + for (int byte1 = 0xF0; byte1 <= 0xF0; ++byte1) + { + for (int byte2 = 0x90; byte2 <= 0xBF; ++byte2) { - // skip correct third byte - if (0x80 <= byte3 and byte3 <= 0xBF) + check_utf8string(false, byte1, byte2); + } + } + } + + SECTION("ill-formed: missing fourth byte") + { + for (int byte1 = 0xF0; byte1 <= 0xF0; ++byte1) + { + for (int byte2 = 0x90; byte2 <= 0xBF; ++byte2) + { + for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) + { + check_utf8string(false, byte1, byte2, byte3); + } + } + } + } + + SECTION("ill-formed: wrong second byte") + { + for (int byte1 = 0xF0; byte1 <= 0xF0; ++byte1) + { + for (int byte2 = 0x00; byte2 <= 0xFF; ++byte2) + { + // skip correct second byte + if (0x90 <= byte2 and byte2 <= 0xBF) { continue; } - check_utf8string(false, byte1, byte2, byte3); - } - } - } - } - } - - SECTION("UTF8-4 (xF0 x90-BF UTF8-tail UTF8-tail)") - { - SECTION("well-formed") - { - for (int byte1 = 0xF0; byte1 <= 0xF0; ++byte1) - { - for (int byte2 = 0x90; byte2 <= 0xBF; ++byte2) - { - for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) - { - for (int byte4 = 0x80; byte4 <= 0xBF; ++byte4) + for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) { - check_utf8string(true, byte1, byte2, byte3, byte4); + for (int byte4 = 0x80; byte4 <= 0xBF; ++byte4) + { + check_utf8string(false, byte1, byte2, byte3, byte4); + } } } } } - } - SECTION("ill-formed: missing second byte") - { - for (int byte1 = 0xF0; byte1 <= 0xF0; ++byte1) + SECTION("ill-formed: wrong third byte") { - check_utf8string(false, byte1); - } - } - - SECTION("ill-formed: missing third byte") - { - for (int byte1 = 0xF0; byte1 <= 0xF0; ++byte1) - { - for (int byte2 = 0x90; byte2 <= 0xBF; ++byte2) + for (int byte1 = 0xF0; byte1 <= 0xF0; ++byte1) { - check_utf8string(false, byte1, byte2); - } - } - } - - SECTION("ill-formed: missing fourth byte") - { - for (int byte1 = 0xF0; byte1 <= 0xF0; ++byte1) - { - for (int byte2 = 0x90; byte2 <= 0xBF; ++byte2) - { - for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) + for (int byte2 = 0x90; byte2 <= 0xBF; ++byte2) { - check_utf8string(false, byte1, byte2, byte3); - } - } - } - } - - SECTION("ill-formed: wrong second byte") - { - for (int byte1 = 0xF0; byte1 <= 0xF0; ++byte1) - { - for (int byte2 = 0x00; byte2 <= 0xFF; ++byte2) - { - // skip correct second byte - if (0x90 <= byte2 and byte2 <= 0xBF) - { - continue; - } - - for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) - { - for (int byte4 = 0x80; byte4 <= 0xBF; ++byte4) + for (int byte3 = 0x00; byte3 <= 0xFF; ++byte3) { - check_utf8string(false, byte1, byte2, byte3, byte4); + // skip correct third byte + if (0x80 <= byte3 and byte3 <= 0xBF) + { + continue; + } + + for (int byte4 = 0x80; byte4 <= 0xBF; ++byte4) + { + check_utf8string(false, byte1, byte2, byte3, byte4); + } } } } } - } - SECTION("ill-formed: wrong third byte") - { - for (int byte1 = 0xF0; byte1 <= 0xF0; ++byte1) + SECTION("ill-formed: wrong fourth byte") { - for (int byte2 = 0x90; byte2 <= 0xBF; ++byte2) + for (int byte1 = 0xF0; byte1 <= 0xF0; ++byte1) { - for (int byte3 = 0x00; byte3 <= 0xFF; ++byte3) + for (int byte2 = 0x90; byte2 <= 0xBF; ++byte2) { - // skip correct third byte - if (0x80 <= byte3 and byte3 <= 0xBF) + for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) { - continue; - } - - for (int byte4 = 0x80; byte4 <= 0xBF; ++byte4) - { - check_utf8string(false, byte1, byte2, byte3, byte4); - } - } - } - } - } - - SECTION("ill-formed: wrong fourth byte") - { - for (int byte1 = 0xF0; byte1 <= 0xF0; ++byte1) - { - for (int byte2 = 0x90; byte2 <= 0xBF; ++byte2) - { - for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) - { for (int byte4 = 0x00; byte4 <= 0xFF; ++byte4) { // skip fourth second byte @@ -601,269 +603,263 @@ TEST_CASE("RFC 3629", "[hide]") check_utf8string(false, byte1, byte2, byte3, byte4); } - } - } - } - } - } - - SECTION("UTF8-4 (xF1-F3 UTF8-tail UTF8-tail UTF8-tail)") - { - SECTION("well-formed") - { - for (int byte1 = 0xF1; byte1 <= 0xF3; ++byte1) - { - for (int byte2 = 0x80; byte2 <= 0xBF; ++byte2) - { - for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) - { - for (int byte4 = 0x80; byte4 <= 0xBF; ++byte4) - { - check_utf8string(true, byte1, byte2, byte3, byte4); } } } } } - SECTION("ill-formed: missing second byte") + SECTION("UTF8-4 (xF1-F3 UTF8-tail UTF8-tail UTF8-tail)") { - for (int byte1 = 0xF1; byte1 <= 0xF3; ++byte1) + SECTION("well-formed") { - check_utf8string(false, byte1); - } - } - - SECTION("ill-formed: missing third byte") - { - for (int byte1 = 0xF1; byte1 <= 0xF3; ++byte1) - { - for (int byte2 = 0x80; byte2 <= 0xBF; ++byte2) + for (int byte1 = 0xF1; byte1 <= 0xF3; ++byte1) { - check_utf8string(false, byte1, byte2); - } - } - } - - SECTION("ill-formed: missing fourth byte") - { - for (int byte1 = 0xF1; byte1 <= 0xF3; ++byte1) - { - for (int byte2 = 0x80; byte2 <= 0xBF; ++byte2) - { - for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) + for (int byte2 = 0x80; byte2 <= 0xBF; ++byte2) { - check_utf8string(false, byte1, byte2, byte3); - } - } - } - } - - SECTION("ill-formed: wrong second byte") - { - for (int byte1 = 0xF1; byte1 <= 0xF3; ++byte1) - { - for (int byte2 = 0x00; byte2 <= 0xFF; ++byte2) - { - // skip correct second byte - if (0x80 <= byte2 and byte2 <= 0xBF) - { - continue; - } - - for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) - { - for (int byte4 = 0x80; byte4 <= 0xBF; ++byte4) + for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) { - check_utf8string(false, byte1, byte2, byte3, byte4); + for (int byte4 = 0x80; byte4 <= 0xBF; ++byte4) + { + check_utf8string(true, byte1, byte2, byte3, byte4); + } } } } } - } - SECTION("ill-formed: wrong third byte") - { - for (int byte1 = 0xF1; byte1 <= 0xF3; ++byte1) + SECTION("ill-formed: missing second byte") { - for (int byte2 = 0x80; byte2 <= 0xBF; ++byte2) + for (int byte1 = 0xF1; byte1 <= 0xF3; ++byte1) { - for (int byte3 = 0x00; byte3 <= 0xFF; ++byte3) + check_utf8string(false, byte1); + } + } + + SECTION("ill-formed: missing third byte") + { + for (int byte1 = 0xF1; byte1 <= 0xF3; ++byte1) + { + for (int byte2 = 0x80; byte2 <= 0xBF; ++byte2) { - // skip correct third byte - if (0x80 <= byte3 and byte3 <= 0xBF) + check_utf8string(false, byte1, byte2); + } + } + } + + SECTION("ill-formed: missing fourth byte") + { + for (int byte1 = 0xF1; byte1 <= 0xF3; ++byte1) + { + for (int byte2 = 0x80; byte2 <= 0xBF; ++byte2) + { + for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) + { + check_utf8string(false, byte1, byte2, byte3); + } + } + } + } + + SECTION("ill-formed: wrong second byte") + { + for (int byte1 = 0xF1; byte1 <= 0xF3; ++byte1) + { + for (int byte2 = 0x00; byte2 <= 0xFF; ++byte2) + { + // skip correct second byte + if (0x80 <= byte2 and byte2 <= 0xBF) { continue; } - for (int byte4 = 0x80; byte4 <= 0xBF; ++byte4) + for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) { - check_utf8string(false, byte1, byte2, byte3, byte4); + for (int byte4 = 0x80; byte4 <= 0xBF; ++byte4) + { + check_utf8string(false, byte1, byte2, byte3, byte4); + } } } } } - } - SECTION("ill-formed: wrong fourth byte") - { - for (int byte1 = 0xF1; byte1 <= 0xF3; ++byte1) + SECTION("ill-formed: wrong third byte") { - for (int byte2 = 0x80; byte2 <= 0xBF; ++byte2) + for (int byte1 = 0xF1; byte1 <= 0xF3; ++byte1) { - for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) + for (int byte2 = 0x80; byte2 <= 0xBF; ++byte2) { - for (int byte4 = 0x00; byte4 <= 0xFF; ++byte4) + for (int byte3 = 0x00; byte3 <= 0xFF; ++byte3) { - // skip correct fourth byte + // skip correct third byte if (0x80 <= byte3 and byte3 <= 0xBF) { continue; } - check_utf8string(false, byte1, byte2, byte3, byte4); + for (int byte4 = 0x80; byte4 <= 0xBF; ++byte4) + { + check_utf8string(false, byte1, byte2, byte3, byte4); + } } } } } - } - } - SECTION("UTF8-4 (xF4 x80-8F UTF8-tail UTF8-tail)") - { - SECTION("well-formed") - { - for (int byte1 = 0xF4; byte1 <= 0xF4; ++byte1) + SECTION("ill-formed: wrong fourth byte") { - for (int byte2 = 0x80; byte2 <= 0x8F; ++byte2) + for (int byte1 = 0xF1; byte1 <= 0xF3; ++byte1) { - for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) + for (int byte2 = 0x80; byte2 <= 0xBF; ++byte2) { - for (int byte4 = 0x80; byte4 <= 0xBF; ++byte4) + for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) { - check_utf8string(true, byte1, byte2, byte3, byte4); + for (int byte4 = 0x00; byte4 <= 0xFF; ++byte4) + { + // skip correct fourth byte + if (0x80 <= byte3 and byte3 <= 0xBF) + { + continue; + } + + check_utf8string(false, byte1, byte2, byte3, byte4); + } } } } } } - SECTION("ill-formed: missing second byte") + SECTION("UTF8-4 (xF4 x80-8F UTF8-tail UTF8-tail)") { - for (int byte1 = 0xF4; byte1 <= 0xF4; ++byte1) + SECTION("well-formed") { - check_utf8string(false, byte1); - } - } - - SECTION("ill-formed: missing third byte") - { - for (int byte1 = 0xF4; byte1 <= 0xF4; ++byte1) - { - for (int byte2 = 0x80; byte2 <= 0x8F; ++byte2) + for (int byte1 = 0xF4; byte1 <= 0xF4; ++byte1) { - check_utf8string(false, byte1, byte2); - } - } - } - - SECTION("ill-formed: missing fourth byte") - { - for (int byte1 = 0xF4; byte1 <= 0xF4; ++byte1) - { - for (int byte2 = 0x80; byte2 <= 0x8F; ++byte2) - { - for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) + for (int byte2 = 0x80; byte2 <= 0x8F; ++byte2) { - check_utf8string(false, byte1, byte2, byte3); - } - } - } - } - - SECTION("ill-formed: wrong second byte") - { - for (int byte1 = 0xF4; byte1 <= 0xF4; ++byte1) - { - for (int byte2 = 0x00; byte2 <= 0xFF; ++byte2) - { - // skip correct second byte - if (0x80 <= byte2 and byte2 <= 0x8F) - { - continue; - } - - for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) - { - for (int byte4 = 0x80; byte4 <= 0xBF; ++byte4) + for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) { - check_utf8string(false, byte1, byte2, byte3, byte4); + for (int byte4 = 0x80; byte4 <= 0xBF; ++byte4) + { + check_utf8string(true, byte1, byte2, byte3, byte4); + } } } } } - } - SECTION("ill-formed: wrong third byte") - { - for (int byte1 = 0xF4; byte1 <= 0xF4; ++byte1) + SECTION("ill-formed: missing second byte") { - for (int byte2 = 0x80; byte2 <= 0x8F; ++byte2) + for (int byte1 = 0xF4; byte1 <= 0xF4; ++byte1) { - for (int byte3 = 0x00; byte3 <= 0xFF; ++byte3) + check_utf8string(false, byte1); + } + } + + SECTION("ill-formed: missing third byte") + { + for (int byte1 = 0xF4; byte1 <= 0xF4; ++byte1) + { + for (int byte2 = 0x80; byte2 <= 0x8F; ++byte2) { - // skip correct third byte - if (0x80 <= byte3 and byte3 <= 0xBF) + check_utf8string(false, byte1, byte2); + } + } + } + + SECTION("ill-formed: missing fourth byte") + { + for (int byte1 = 0xF4; byte1 <= 0xF4; ++byte1) + { + for (int byte2 = 0x80; byte2 <= 0x8F; ++byte2) + { + for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) + { + check_utf8string(false, byte1, byte2, byte3); + } + } + } + } + + SECTION("ill-formed: wrong second byte") + { + for (int byte1 = 0xF4; byte1 <= 0xF4; ++byte1) + { + for (int byte2 = 0x00; byte2 <= 0xFF; ++byte2) + { + // skip correct second byte + if (0x80 <= byte2 and byte2 <= 0x8F) { continue; } - for (int byte4 = 0x80; byte4 <= 0xBF; ++byte4) + for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) { - check_utf8string(false, byte1, byte2, byte3, byte4); + for (int byte4 = 0x80; byte4 <= 0xBF; ++byte4) + { + check_utf8string(false, byte1, byte2, byte3, byte4); + } } } } } - } - SECTION("ill-formed: wrong fourth byte") - { - for (int byte1 = 0xF4; byte1 <= 0xF4; ++byte1) + SECTION("ill-formed: wrong third byte") { - for (int byte2 = 0x80; byte2 <= 0x8F; ++byte2) + for (int byte1 = 0xF4; byte1 <= 0xF4; ++byte1) { - for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) + for (int byte2 = 0x80; byte2 <= 0x8F; ++byte2) { - for (int byte4 = 0x00; byte4 <= 0xFF; ++byte4) + for (int byte3 = 0x00; byte3 <= 0xFF; ++byte3) { - // skip correct fourth byte + // skip correct third byte if (0x80 <= byte3 and byte3 <= 0xBF) { continue; } - check_utf8string(false, byte1, byte2, byte3, byte4); + for (int byte4 = 0x80; byte4 <= 0xBF; ++byte4) + { + check_utf8string(false, byte1, byte2, byte3, byte4); + } + } + } + } + } + + SECTION("ill-formed: wrong fourth byte") + { + for (int byte1 = 0xF4; byte1 <= 0xF4; ++byte1) + { + for (int byte2 = 0x80; byte2 <= 0x8F; ++byte2) + { + for (int byte3 = 0x80; byte3 <= 0xBF; ++byte3) + { + for (int byte4 = 0x00; byte4 <= 0xFF; ++byte4) + { + // skip correct fourth byte + if (0x80 <= byte3 and byte3 <= 0xBF) + { + continue; + } + + check_utf8string(false, byte1, byte2, byte3, byte4); + } } } } } } } -} -TEST_CASE("Unicode", "[hide]") -{ - /* NOTE: to_unicode is not used any more - SECTION("full enumeration of Unicode code points") + SECTION("\\uxxxx sequences") { - // lexer to call to_unicode on - json::lexer dummy_lexer("", 0); - // create an escaped string from a code point const auto codepoint_to_unicode = [](std::size_t cp) { - // copd points are represented as a six-character sequence: a + // code points are represented as a six-character sequence: a // reverse solidus, followed by the lowercase letter u, followed // by four hexadecimal digits that encode the character's code // point @@ -872,72 +868,101 @@ TEST_CASE("Unicode", "[hide]") return ss.str(); }; - // generate all UTF-8 code points; in total, 1112064 code points are - // generated: 0x1FFFFF code points - 2048 invalid values between - // 0xD800 and 0xDFFF. - for (std::size_t cp = 0; cp <= 0x10FFFFu; ++cp) + SECTION("correct sequences") { - // The Unicode standard permanently reserves these code point - // values for UTF-16 encoding of the high and low surrogates, and - // they will never be assigned a character, so there should be no - // reason to encode them. The official Unicode standard says that - // no UTF forms, including UTF-16, can encode these code points. - if (cp >= 0xD800u and cp <= 0xDFFFu) + // generate all UTF-8 code points; in total, 1112064 code points are + // generated: 0x1FFFFF code points - 2048 invalid values between + // 0xD800 and 0xDFFF. + for (std::size_t cp = 0; cp <= 0x10FFFFu; ++cp) { - // if we would not skip these code points, we would get a - // "missing low surrogate" exception - continue; - } + // string to store the code point as in \uxxxx format + std::string json_text = "\""; - // string to store the code point as in \uxxxx format - std::string escaped_string; - // string to store the code point as unescaped character sequence - std::string unescaped_string; - - if (cp < 0x10000u) - { - // code points in the Basic Multilingual Plane can be - // represented with one \\uxxxx sequence - escaped_string = codepoint_to_unicode(cp); - - // All Unicode characters may be placed within the quotation - // marks, except for the characters that must be escaped: - // quotation mark, reverse solidus, and the control characters - // (U+0000 through U+001F); we ignore these code points as - // they are checked with codepoint_to_unicode. - if (cp > 0x1f and cp != 0x22 and cp != 0x5c) + // decide whether to use one or two \uxxxx sequences + if (cp < 0x10000u) { - unescaped_string = dummy_lexer.to_unicode(cp); + // The Unicode standard permanently reserves these code point + // values for UTF-16 encoding of the high and low surrogates, and + // they will never be assigned a character, so there should be no + // reason to encode them. The official Unicode standard says that + // no UTF forms, including UTF-16, can encode these code points. + if (cp >= 0xD800u and cp <= 0xDFFFu) + { + // if we would not skip these code points, we would get a + // "missing low surrogate" exception + continue; + } + + // code points in the Basic Multilingual Plane can be + // represented with one \uxxxx sequence + json_text += codepoint_to_unicode(cp); + } + else + { + // To escape an extended character that is not in the Basic + // Multilingual Plane, the character is represented as a + // 12-character sequence, encoding the UTF-16 surrogate pair + const auto codepoint1 = 0xd800u + (((cp - 0x10000u) >> 10) & 0x3ffu); + const auto codepoint2 = 0xdc00u + ((cp - 0x10000u) & 0x3ffu); + json_text += codepoint_to_unicode(codepoint1) + codepoint_to_unicode(codepoint2); + } + + json_text += "\""; + CAPTURE(json_text); + CHECK_NOTHROW(json::parse(json_text)); + } + } + + SECTION("incorrect sequences") + { + SECTION("high surrogate without low surrogate") + { + // D800..DBFF are high surrogates and must be followed by low + // surrogates DC00..DFFF; here, nothing follows + for (std::size_t cp = 0xD800u; cp <= 0xDBFFu; ++cp) + { + std::string json_text = "\"" + codepoint_to_unicode(cp) + "\""; + CAPTURE(json_text); + CHECK_THROWS_AS(json::parse(json_text), json::parse_error); } } - else + +#if 0 + SECTION("high surrogate with wrong low surrogate") { - // To escape an extended character that is not in the Basic - // Multilingual Plane, the character is represented as a - // 12-character sequence, encoding the UTF-16 surrogate pair - const auto codepoint1 = 0xd800u + (((cp - 0x10000u) >> 10) & 0x3ffu); - const auto codepoint2 = 0xdc00u + ((cp - 0x10000u) & 0x3ffu); - escaped_string = codepoint_to_unicode(codepoint1); - escaped_string += codepoint_to_unicode(codepoint2); - unescaped_string += dummy_lexer.to_unicode(codepoint1, codepoint2); + // D800..DBFF are high surrogates and must be followed by low + // surrogates DC00..DFFF; here a different sequence follows + for (std::size_t cp1 = 0xD800u; cp1 <= 0xDBFFu; ++cp1) + { + for (std::size_t cp2 = 0x0000u; cp2 <= 0xFFFFu; ++cp2) + { + if (0xDC00u <= cp2 and cp2 <= 0xDFFFu) + { + continue; + } + + std::string json_text = "\"" + codepoint_to_unicode(cp1) + codepoint_to_unicode(cp2) + "\""; + CAPTURE(json_text); + CHECK_THROWS_AS(json::parse(json_text), json::parse_error); + } + } + } +#endif + + SECTION("low surrogate without high surrogate") + { + // low surrogates DC00..DFFF must follow high surrogates; here, + // they occur alone + for (std::size_t cp = 0xDC00u; cp <= 0xDFFFu; ++cp) + { + std::string json_text = "\"" + codepoint_to_unicode(cp) + "\""; + CAPTURE(json_text); + CHECK_THROWS_AS(json::parse(json_text), json::parse_error); + } } - // all other code points are valid and must not yield parse errors - CAPTURE(cp); - CAPTURE(escaped_string); - CAPTURE(unescaped_string); - - json j1, j2, j3, j4; - CHECK_NOTHROW(j1 = json::parse("\"" + escaped_string + "\"")); - CHECK_NOTHROW(j2 = json::parse(j1.dump())); - CHECK(j1 == j2); - - CHECK_NOTHROW(j3 = json::parse("\"" + unescaped_string + "\"")); - CHECK_NOTHROW(j4 = json::parse(j3.dump())); - CHECK(j3 == j4); } } - */ SECTION("read all unicode characters") { From cfc2e8391cf6cfe098455f9b77eb9c3695bb41c6 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Mon, 24 Apr 2017 15:07:43 +0200 Subject: [PATCH 178/530] :hammer: removed too long running tests --- test/src/unit-unicode.cpp | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/test/src/unit-unicode.cpp b/test/src/unit-unicode.cpp index 67e97346..120941ee 100644 --- a/test/src/unit-unicode.cpp +++ b/test/src/unit-unicode.cpp @@ -913,6 +913,7 @@ TEST_CASE("Unicode", "[hide]") } } +#if 0 SECTION("incorrect sequences") { SECTION("high surrogate without low surrogate") @@ -927,7 +928,6 @@ TEST_CASE("Unicode", "[hide]") } } -#if 0 SECTION("high surrogate with wrong low surrogate") { // D800..DBFF are high surrogates and must be followed by low @@ -947,7 +947,6 @@ TEST_CASE("Unicode", "[hide]") } } } -#endif SECTION("low surrogate without high surrogate") { @@ -962,6 +961,7 @@ TEST_CASE("Unicode", "[hide]") } } +#endif } SECTION("read all unicode characters") From 8b9f51179e57c758da4a7042831e68901606d5c7 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Mon, 24 Apr 2017 17:46:21 +0200 Subject: [PATCH 179/530] :sparkles: started working on #458 a simple acceptor function --- src/json.hpp | 144 +++++++++++ test/src/unit-class_parser.cpp | 445 +++++++++++++++++++++++++++++++++ 2 files changed, 589 insertions(+) diff --git a/src/json.hpp b/src/json.hpp index 99a6e854..a0fe5e8e 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -12501,6 +12501,7 @@ scan_number_done: @brief public parser interface @param[in] strict whether to expect the last token to be EOF + @return parsed JSON value @throw parse_error.101 in case of an unexpected token @throw parse_error.102 if to_unicode fails or surrogate error @@ -12524,6 +12525,30 @@ scan_number_done: return result.is_discarded() ? basic_json() : std::move(result); } + /*! + @brief public accept interface + + @param[in] strict whether to expect the last token to be EOF + @return whether the input is a proper JSON text + */ + bool accept(const bool strict = true) + { + // read first token + get_token(); + + if (not accept_internal()) + { + return false; + } + + if (strict and last_token != lexer::token_type::end_of_input) + { + return false; + } + + return true; + } + private: /*! @brief the actual parser @@ -12745,6 +12770,125 @@ scan_number_done: return result; } + /*! + @brief the acutal acceptor + */ + bool accept_internal() + { + switch (last_token) + { + case lexer::token_type::begin_object: + { + // read next token + get_token(); + + // closing } -> we are done + if (last_token == lexer::token_type::end_object) + { + get_token(); + return true; + } + + // parse values + while (true) + { + // parse key + if (last_token != lexer::token_type::value_string) + { + return false; + } + + // parse separator (:) + get_token(); + if (last_token != lexer::token_type::name_separator) + { + return false; + } + + // parse value + get_token(); + if (not accept_internal()) + { + return false; + } + + // comma -> next value + if (last_token == lexer::token_type::value_separator) + { + get_token(); + continue; + } + + // closing } + if (last_token != lexer::token_type::end_object) + { + return false; + } + + get_token(); + return true; + } + } + + case lexer::token_type::begin_array: + { + // read next token + get_token(); + + // closing ] -> we are done + if (last_token == lexer::token_type::end_array) + { + get_token(); + return true; + } + + // parse values + while (true) + { + // parse value + if (not accept_internal()) + { + return false; + } + + // comma -> next value + if (last_token == lexer::token_type::value_separator) + { + get_token(); + continue; + } + + // closing ] + if (last_token != lexer::token_type::end_array) + { + return false; + } + + get_token(); + return true; + } + } + + case lexer::token_type::literal_null: + case lexer::token_type::value_string: + case lexer::token_type::literal_true: + case lexer::token_type::literal_false: + case lexer::token_type::value_unsigned: + case lexer::token_type::value_integer: + case lexer::token_type::value_float: + { + get_token(); + return true; + } + + default: + { + // the last token was unexpected + return false; + } + } + } + /// get next token from lexer typename lexer::token_type get_token() { diff --git a/test/src/unit-class_parser.cpp b/test/src/unit-class_parser.cpp index b631a978..5866b901 100644 --- a/test/src/unit-class_parser.cpp +++ b/test/src/unit-class_parser.cpp @@ -350,6 +350,268 @@ TEST_CASE("parser class") } } + SECTION("accept") + { + SECTION("null") + { + CHECK(parse_string("null").accept()); + } + + SECTION("true") + { + CHECK(parse_string("true").accept()); + } + + SECTION("false") + { + CHECK(parse_string("false").accept()); + } + + SECTION("array") + { + SECTION("empty array") + { + CHECK(parse_string("[]").accept()); + CHECK(parse_string("[ ]").accept()); + } + + SECTION("nonempty array") + { + CHECK(parse_string("[true, false, null]").accept()); + } + } + + SECTION("object") + { + SECTION("empty object") + { + CHECK(parse_string("{}").accept()); + CHECK(parse_string("{ }").accept()); + } + + SECTION("nonempty object") + { + CHECK(parse_string("{\"\": true, \"one\": 1, \"two\": null}").accept()); + } + } + + SECTION("string") + { + // empty string + CHECK(parse_string("\"\"").accept()); + + SECTION("errors") + { + // error: tab in string + CHECK(parse_string("\"\t\"").accept() == false); + // error: newline in string + CHECK(parse_string("\"\n\"").accept() == false); + CHECK(parse_string("\"\r\"").accept() == false); + // error: backspace in string + CHECK(parse_string("\"\b\"").accept() == false); + // improve code coverage + CHECK(parse_string("\uFF01").accept() == false); + CHECK(parse_string("[-4:1,]").accept() == false); + // unescaped control characters + CHECK(parse_string("\"\x00\"").accept() == false); + CHECK(parse_string("\"\x01\"").accept() == false); + CHECK(parse_string("\"\x02\"").accept() == false); + CHECK(parse_string("\"\x03\"").accept() == false); + CHECK(parse_string("\"\x04\"").accept() == false); + CHECK(parse_string("\"\x05\"").accept() == false); + CHECK(parse_string("\"\x06\"").accept() == false); + CHECK(parse_string("\"\x07\"").accept() == false); + CHECK(parse_string("\"\x08\"").accept() == false); + CHECK(parse_string("\"\x09\"").accept() == false); + CHECK(parse_string("\"\x0a\"").accept() == false); + CHECK(parse_string("\"\x0b\"").accept() == false); + CHECK(parse_string("\"\x0c\"").accept() == false); + CHECK(parse_string("\"\x0d\"").accept() == false); + CHECK(parse_string("\"\x0e\"").accept() == false); + CHECK(parse_string("\"\x0f\"").accept() == false); + CHECK(parse_string("\"\x10\"").accept() == false); + CHECK(parse_string("\"\x11\"").accept() == false); + CHECK(parse_string("\"\x12\"").accept() == false); + CHECK(parse_string("\"\x13\"").accept() == false); + CHECK(parse_string("\"\x14\"").accept() == false); + CHECK(parse_string("\"\x15\"").accept() == false); + CHECK(parse_string("\"\x16\"").accept() == false); + CHECK(parse_string("\"\x17\"").accept() == false); + CHECK(parse_string("\"\x18\"").accept() == false); + CHECK(parse_string("\"\x19\"").accept() == false); + CHECK(parse_string("\"\x1a\"").accept() == false); + CHECK(parse_string("\"\x1b\"").accept() == false); + CHECK(parse_string("\"\x1c\"").accept() == false); + CHECK(parse_string("\"\x1d\"").accept() == false); + CHECK(parse_string("\"\x1e\"").accept() == false); + CHECK(parse_string("\"\x1f\"").accept() == false); + } + + SECTION("escaped") + { + // quotation mark "\"" + auto r1 = R"("\"")"_json; + CHECK(parse_string("\"\\\"\"").accept()); + // reverse solidus "\\" + auto r2 = R"("\\")"_json; + CHECK(parse_string("\"\\\\\"").accept()); + // solidus + CHECK(parse_string("\"\\/\"").accept()); + // backspace + CHECK(parse_string("\"\\b\"").accept()); + // formfeed + CHECK(parse_string("\"\\f\"").accept()); + // newline + CHECK(parse_string("\"\\n\"").accept()); + // carriage return + CHECK(parse_string("\"\\r\"").accept()); + // horizontal tab + CHECK(parse_string("\"\\t\"").accept()); + + CHECK(parse_string("\"\\u0001\"").accept()); + CHECK(parse_string("\"\\u000a\"").accept()); + CHECK(parse_string("\"\\u00b0\"").accept()); + CHECK(parse_string("\"\\u0c00\"").accept()); + CHECK(parse_string("\"\\ud000\"").accept()); + CHECK(parse_string("\"\\u000E\"").accept()); + CHECK(parse_string("\"\\u00F0\"").accept()); + CHECK(parse_string("\"\\u0100\"").accept()); + CHECK(parse_string("\"\\u2000\"").accept()); + CHECK(parse_string("\"\\uFFFF\"").accept()); + CHECK(parse_string("\"\\u20AC\"").accept()); + CHECK(parse_string("\"€\"").accept()); + CHECK(parse_string("\"🎈\"").accept()); + + CHECK(parse_string("\"\\ud80c\\udc60\"").accept()); + CHECK(parse_string("\"\\ud83c\\udf1e\"").accept()); + } + } + + SECTION("number") + { + SECTION("integers") + { + SECTION("without exponent") + { + CHECK(parse_string("-128").accept()); + CHECK(parse_string("-0").accept()); + CHECK(parse_string("0").accept()); + CHECK(parse_string("128").accept()); + } + + SECTION("with exponent") + { + CHECK(parse_string("0e1").accept()); + CHECK(parse_string("0E1").accept()); + + CHECK(parse_string("10000E-4").accept()); + CHECK(parse_string("10000E-3").accept()); + CHECK(parse_string("10000E-2").accept()); + CHECK(parse_string("10000E-1").accept()); + CHECK(parse_string("10000E0").accept()); + CHECK(parse_string("10000E1").accept()); + CHECK(parse_string("10000E2").accept()); + CHECK(parse_string("10000E3").accept()); + CHECK(parse_string("10000E4").accept()); + + CHECK(parse_string("10000e-4").accept()); + CHECK(parse_string("10000e-3").accept()); + CHECK(parse_string("10000e-2").accept()); + CHECK(parse_string("10000e-1").accept()); + CHECK(parse_string("10000e0").accept()); + CHECK(parse_string("10000e1").accept()); + CHECK(parse_string("10000e2").accept()); + CHECK(parse_string("10000e3").accept()); + CHECK(parse_string("10000e4").accept()); + + CHECK(parse_string("-0e1").accept()); + CHECK(parse_string("-0E1").accept()); + CHECK(parse_string("-0E123").accept()); + } + + SECTION("edge cases") + { + // From RFC7159, Section 6: + // Note that when such software is used, numbers that are + // integers and are in the range [-(2**53)+1, (2**53)-1] + // are interoperable in the sense that implementations will + // agree exactly on their numeric values. + + // -(2**53)+1 + CHECK(parse_string("-9007199254740991").accept()); + // (2**53)-1 + CHECK(parse_string("9007199254740991").accept()); + } + + SECTION("over the edge cases") // issue #178 - Integer conversion to unsigned (incorrect handling of 64 bit integers) + { + // While RFC7159, Section 6 specifies a preference for support + // for ranges in range of IEEE 754-2008 binary64 (double precision) + // this does not accommodate 64 bit integers without loss of accuracy. + // As 64 bit integers are now widely used in software, it is desirable + // to expand support to to the full 64 bit (signed and unsigned) range + // i.e. -(2**63) -> (2**64)-1. + + // -(2**63) ** Note: compilers see negative literals as negated positive numbers (hence the -1)) + CHECK(parse_string("-9223372036854775808").accept()); + // (2**63)-1 + CHECK(parse_string("9223372036854775807").accept()); + // (2**64)-1 + CHECK(parse_string("18446744073709551615").accept()); + } + } + + SECTION("floating-point") + { + SECTION("without exponent") + { + CHECK(parse_string("-128.5").accept()); + CHECK(parse_string("0.999").accept()); + CHECK(parse_string("128.5").accept()); + CHECK(parse_string("-0.0").accept()); + } + + SECTION("with exponent") + { + CHECK(parse_string("-128.5E3").accept()); + CHECK(parse_string("-128.5E-3").accept()); + CHECK(parse_string("-0.0e1").accept()); + CHECK(parse_string("-0.0E1").accept()); + } + } + + SECTION("overflow") + { + // overflows during parsing yield an exception, but is accepted anyway + CHECK(parse_string("1.18973e+4932").accept()); + } + + SECTION("invalid numbers") + { + CHECK(parse_string("01").accept() == false); + CHECK(parse_string("--1").accept() == false); + CHECK(parse_string("1.").accept() == false); + CHECK(parse_string("1E").accept() == false); + CHECK(parse_string("1E-").accept() == false); + CHECK(parse_string("1.E1").accept() == false); + CHECK(parse_string("-1E").accept() == false); + CHECK(parse_string("-0E#").accept() == false); + CHECK(parse_string("-0E-#").accept() == false); + CHECK(parse_string("-0#").accept() == false); + CHECK(parse_string("-0.0:").accept() == false); + CHECK(parse_string("-0.0Z").accept() == false); + CHECK(parse_string("-0E123:").accept() == false); + CHECK(parse_string("-0e0-:").accept() == false); + CHECK(parse_string("-0e-:").accept() == false); + CHECK(parse_string("-0f").accept() == false); + + // numbers must not begin with "+" + CHECK(parse_string("+1").accept() == false); + CHECK(parse_string("+0").accept() == false); + } + } + } + SECTION("parse errors") { // unexpected end of number @@ -661,6 +923,189 @@ TEST_CASE("parser class") "[json.exception.parse_error.101] parse error at 13: syntax error - invalid string: surrogate U+D80C must be followed by U+DC00..U+DFFF instead of U+FFFF; last read '\"\\uD80C\\uFFFF'"); } + SECTION("parse errors (accept)") + { + // unexpected end of number + CHECK(parse_string("0.").accept() == false); + CHECK(parse_string("-").accept() == false); + CHECK(parse_string("--").accept() == false); + CHECK(parse_string("-0.").accept() == false); + CHECK(parse_string("-.").accept() == false); + CHECK(parse_string("-:").accept() == false); + CHECK(parse_string("0.:").accept() == false); + CHECK(parse_string("e.").accept() == false); + CHECK(parse_string("1e.").accept() == false); + CHECK(parse_string("1e/").accept() == false); + CHECK(parse_string("1e:").accept() == false); + CHECK(parse_string("1E.").accept() == false); + CHECK(parse_string("1E/").accept() == false); + CHECK(parse_string("1E:").accept() == false); + + // unexpected end of null + CHECK(parse_string("n").accept() == false); + CHECK(parse_string("nu").accept() == false); + CHECK(parse_string("nul").accept() == false); + + // unexpected end of true + CHECK(parse_string("t").accept() == false); + CHECK(parse_string("tr").accept() == false); + CHECK(parse_string("tru").accept() == false); + + // unexpected end of false + CHECK(parse_string("f").accept() == false); + CHECK(parse_string("fa").accept() == false); + CHECK(parse_string("fal").accept() == false); + CHECK(parse_string("fals").accept() == false); + + // missing/unexpected end of array + CHECK(parse_string("[").accept() == false); + CHECK(parse_string("[1").accept() == false); + CHECK(parse_string("[1,").accept() == false); + CHECK(parse_string("[1,]").accept() == false); + CHECK(parse_string("]").accept() == false); + + // missing/unexpected end of object + CHECK(parse_string("{").accept() == false); + CHECK(parse_string("{\"foo\"").accept() == false); + CHECK(parse_string("{\"foo\":").accept() == false); + CHECK(parse_string("{\"foo\":}").accept() == false); + CHECK(parse_string("{\"foo\":1,}").accept() == false); + CHECK(parse_string("}").accept() == false); + + // missing/unexpected end of string + CHECK(parse_string("\"").accept() == false); + CHECK(parse_string("\"\\\"").accept() == false); + CHECK(parse_string("\"\\u\"").accept() == false); + CHECK(parse_string("\"\\u0\"").accept() == false); + CHECK(parse_string("\"\\u01\"").accept() == false); + CHECK(parse_string("\"\\u012\"").accept() == false); + CHECK(parse_string("\"\\u").accept() == false); + CHECK(parse_string("\"\\u0").accept() == false); + CHECK(parse_string("\"\\u01").accept() == false); + CHECK(parse_string("\"\\u012").accept() == false); + + // invalid escapes + for (int c = 1; c < 128; ++c) + { + auto s = std::string("\"\\") + std::string(1, static_cast(c)) + "\""; + + switch (c) + { + // valid escapes + case ('"'): + case ('\\'): + case ('/'): + case ('b'): + case ('f'): + case ('n'): + case ('r'): + case ('t'): + { + CHECK(parse_string(s.c_str()).accept()); + break; + } + + // \u must be followed with four numbers, so we skip it here + case ('u'): + { + break; + } + + // any other combination of backslash and character is invalid + default: + { + CHECK(parse_string(s.c_str()).accept() == false); + break; + } + } + } + + // invalid \uxxxx escapes + { + // check whether character is a valid hex character + const auto valid = [](int c) + { + switch (c) + { + case ('0'): + case ('1'): + case ('2'): + case ('3'): + case ('4'): + case ('5'): + case ('6'): + case ('7'): + case ('8'): + case ('9'): + case ('a'): + case ('b'): + case ('c'): + case ('d'): + case ('e'): + case ('f'): + case ('A'): + case ('B'): + case ('C'): + case ('D'): + case ('E'): + case ('F'): + { + return true; + } + + default: + { + return false; + } + } + }; + + for (int c = 1; c < 128; ++c) + { + std::string s = "\"\\u"; + + // create a string with the iterated character at each position + auto s1 = s + "000" + std::string(1, static_cast(c)) + "\""; + auto s2 = s + "00" + std::string(1, static_cast(c)) + "0\""; + auto s3 = s + "0" + std::string(1, static_cast(c)) + "00\""; + auto s4 = s + std::string(1, static_cast(c)) + "000\""; + + if (valid(c)) + { + CAPTURE(s1); + CHECK(parse_string(s1.c_str()).accept()); + CAPTURE(s2); + CHECK(parse_string(s2.c_str()).accept()); + CAPTURE(s3); + CHECK(parse_string(s3.c_str()).accept()); + CAPTURE(s4); + CHECK(parse_string(s4.c_str()).accept()); + } + else + { + CAPTURE(s1); + CHECK(parse_string(s1.c_str()).accept() == false); + + CAPTURE(s2); + CHECK(parse_string(s2.c_str()).accept() == false); + + CAPTURE(s3); + CHECK(parse_string(s3.c_str()).accept() == false); + + CAPTURE(s4); + CHECK(parse_string(s4.c_str()).accept() == false); + } + } + } + + // missing part of a surrogate pair + CHECK(parse_string("\"\\uD80C\"").accept() == false); + // invalid surrogate pair + CHECK(parse_string("\"\\uD80C\\uD80C\"").accept() == false); + CHECK(parse_string("\"\\uD80C\\u0000\"").accept() == false); + CHECK(parse_string("\"\\uD80C\\uFFFF\"").accept() == false); + } + SECTION("tests found by mutate++") { // test case to make sure no comma preceeds the first key From 19d2dec8b6d1005665d1c5ae003297c1c2d95963 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Wed, 3 May 2017 22:49:21 +0200 Subject: [PATCH 180/530] :memo: added badge from isitmaintained.com --- README.md | 1 + 1 file changed, 1 insertion(+) diff --git a/README.md b/README.md index fc6dde9b..6e345eba 100644 --- a/README.md +++ b/README.md @@ -10,6 +10,7 @@ [![GitHub license](https://img.shields.io/badge/license-MIT-blue.svg)](https://raw.githubusercontent.com/nlohmann/json/master/LICENSE.MIT) [![Github Releases](https://img.shields.io/github/release/nlohmann/json.svg)](https://github.com/nlohmann/json/releases) [![Github Issues](https://img.shields.io/github/issues/nlohmann/json.svg)](http://github.com/nlohmann/json/issues) +[![Average time to resolve an issue](http://isitmaintained.com/badge/resolution/nlohmann/json.svg)](http://isitmaintained.com/project/nlohmann/json "Average time to resolve an issue") [![CII Best Practices](https://bestpractices.coreinfrastructure.org/projects/289/badge)](https://bestpractices.coreinfrastructure.org/projects/289) - [Design goals](#design-goals) From ecf895f2d13ed315ad3a7b64367745da1bd8e4cb Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Fri, 5 May 2017 16:42:22 +0200 Subject: [PATCH 181/530] :memo: mentioned value function on main doc page #570 --- doc/index.md | 1 + 1 file changed, 1 insertion(+) diff --git a/doc/index.md b/doc/index.md index 5392f4d3..0a7b45c8 100644 --- a/doc/index.md +++ b/doc/index.md @@ -27,6 +27,7 @@ These pages contain the API documentation of JSON for Modern C++, a C++11 header - @link nlohmann::basic_json::get_ptr get_ptr @endlink -- get a value pointer - @link nlohmann::basic_json::get_ref get_ref @endlink -- get a value reference - @link nlohmann::basic_json::operator ValueType() const operator ValueType @endlink -- get a value (implicit conversion) + - @link nlohmann::basic_json::value value @endlink -- get a value from an object and return default value if key is not present - lexicographical comparison operators - serialization - deserialization From fba1bcdd0b109c217564088219eb44e2a2a8e077 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 7 May 2017 13:41:48 +0200 Subject: [PATCH 182/530] :bug: fixing #575 I forgot to consider the offset. --- src/json.hpp | 2 +- test/src/unit-regression.cpp | 6 ++++++ 2 files changed, 7 insertions(+), 1 deletion(-) diff --git a/src/json.hpp b/src/json.hpp index f2919b12..e16f2e99 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -8915,7 +8915,7 @@ class basic_json { // avoid reading too many characters const size_t max_length = static_cast(limit - start); - return std::string(start + offset, std::min({length, max_length})); + return std::string(start + offset, std::min({length, max_length - offset})); } private: diff --git a/test/src/unit-regression.cpp b/test/src/unit-regression.cpp index ab9fa935..1ff8f987 100644 --- a/test/src/unit-regression.cpp +++ b/test/src/unit-regression.cpp @@ -1010,4 +1010,10 @@ TEST_CASE("regression tests") CHECK(not(6 <= j["a"])); CHECK(not(6 < j["a"])); } + + SECTION("issue #575 - heap-buffer-overflow (OSS-Fuzz 1400)") + { + std::vector vec = {'"', '\\', '"', 'X', '"', '"'}; + CHECK_THROWS_AS(json::parse(vec), json::parse_error); + } } From 962da001714bf6cff2baaa49046e06c0601e564f Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 7 May 2017 19:27:40 +0200 Subject: [PATCH 183/530] :sparkles: implemented an indentation character #520 An optional parameter for dump() allows to set the character to use for indentation (default: space). In case a JSON value is serialized to an output stream, its fill character is used (and can be set with std::setfill). --- doc/examples/dump.cpp | 2 ++ doc/examples/dump.link | 2 +- doc/examples/dump.output | 13 +++++++++ doc/examples/operator_serialize.cpp | 1 + doc/examples/operator_serialize.link | 2 +- doc/examples/operator_serialize.output | 5 ++++ src/json.hpp | 39 +++++++++++++++++--------- test/src/unit-convenience.cpp | 2 +- test/src/unit-inspection.cpp | 6 ++++ test/src/unit-serialization.cpp | 20 +++++++++++++ 10 files changed, 76 insertions(+), 16 deletions(-) diff --git a/doc/examples/dump.cpp b/doc/examples/dump.cpp index 8190ae49..d1359684 100644 --- a/doc/examples/dump.cpp +++ b/doc/examples/dump.cpp @@ -13,8 +13,10 @@ int main() std::cout << j_object.dump(-1) << "\n\n"; std::cout << j_object.dump(0) << "\n\n"; std::cout << j_object.dump(4) << "\n\n"; + std::cout << j_object.dump(1, '\t') << "\n\n"; std::cout << j_array.dump() << "\n\n"; std::cout << j_array.dump(-1) << "\n\n"; std::cout << j_array.dump(0) << "\n\n"; std::cout << j_array.dump(4) << "\n\n"; + std::cout << j_array.dump(1, '\t') << "\n\n"; } diff --git a/doc/examples/dump.link b/doc/examples/dump.link index 3ced4448..ae5a4aba 100644 --- a/doc/examples/dump.link +++ b/doc/examples/dump.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/dump.output b/doc/examples/dump.output index 9b462eab..bc38c5ea 100644 --- a/doc/examples/dump.output +++ b/doc/examples/dump.output @@ -12,6 +12,11 @@ "two": 2 } +{ + "one": 1, + "two": 2 +} + [1,2,4,8,16] [1,2,4,8,16] @@ -32,3 +37,11 @@ 16 ] +[ + 1, + 2, + 4, + 8, + 16 +] + diff --git a/doc/examples/operator_serialize.cpp b/doc/examples/operator_serialize.cpp index 889bc108..1d817816 100644 --- a/doc/examples/operator_serialize.cpp +++ b/doc/examples/operator_serialize.cpp @@ -16,4 +16,5 @@ int main() // serialize with indentation std::cout << std::setw(4) << j_object << "\n\n"; std::cout << std::setw(2) << j_array << "\n\n"; + std::cout << std::setw(1) << std::setfill('\t') << j_object << "\n\n"; } diff --git a/doc/examples/operator_serialize.link b/doc/examples/operator_serialize.link index a77519c2..104ce0fd 100644 --- a/doc/examples/operator_serialize.link +++ b/doc/examples/operator_serialize.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/operator_serialize.output b/doc/examples/operator_serialize.output index 712a58da..7e86bfa2 100644 --- a/doc/examples/operator_serialize.output +++ b/doc/examples/operator_serialize.output @@ -15,3 +15,8 @@ 16 ] +{ + "one": 1, + "two": 2 +} + diff --git a/src/json.hpp b/src/json.hpp index e16f2e99..14e687d9 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -2883,6 +2883,8 @@ class basic_json members will be pretty-printed with that indent level. An indent level of `0` will only insert newlines. `-1` (the default) selects the most compact representation. + @param[in] indent_char The character to use for indentation of @a indent is + greate than `0`. The default is ` ` (space). @return string containing the serialization of the JSON value @@ -2893,12 +2895,12 @@ class basic_json @see https://docs.python.org/2/library/json.html#json.dump - @since version 1.0.0 + @since version 1.0.0; indentaction character added in version 3.0.0 */ - string_t dump(const int indent = -1) const + string_t dump(const int indent = -1, const char indent_char = ' ') const { string_t result; - serializer s(output_adapter::create(result)); + serializer s(output_adapter::create(result), indent_char); if (indent >= 0) { @@ -6724,11 +6726,13 @@ class basic_json public: /*! @param[in] s output stream to serialize to + @param[in] ichar indentation character to use */ - serializer(output_adapter_t s) + serializer(output_adapter_t s, const char ichar) : o(s), loc(std::localeconv()), thousands_sep(!loc->thousands_sep ? '\0' : loc->thousands_sep[0]), - decimal_point(!loc->decimal_point ? '\0' : loc->decimal_point[0]) + decimal_point(!loc->decimal_point ? '\0' : loc->decimal_point[0]), + indent_char(ichar), indent_string(512, indent_char) {} /*! @@ -7293,8 +7297,11 @@ class basic_json /// the locale's decimal point character const char decimal_point = '\0'; + /// the indentation character + const char indent_char; + /// the indentation string - string_t indent_string = string_t(512, ' '); + string_t indent_string; }; public: @@ -7302,11 +7309,17 @@ class basic_json @brief serialize to stream Serialize the given JSON value @a j to the output stream @a o. The JSON - value will be serialized using the @ref dump member function. The - indentation of the output can be controlled with the member variable - `width` of the output stream @a o. For instance, using the manipulator - `std::setw(4)` on @a o sets the indentation level to `4` and the - serialization result is the same as calling `dump(4)`. + value will be serialized using the @ref dump member function. + + - The indentation of the output can be controlled with the member variable + `width` of the output stream @a o. For instance, using the manipulator + `std::setw(4)` on @a o sets the indentation level to `4` and the + serialization result is the same as calling `dump(4)`. + + - The indentation characrer can be controlled with the member variable + `fill` of the output stream @a o. For instance, the manipulator + `std::setfill('\\t')` sets indentation to use a tab character rather than + the default space character. @param[in,out] o stream to serialize to @param[in] j JSON value to serialize @@ -7318,7 +7331,7 @@ class basic_json @liveexample{The example below shows the serialization with different parameters to `width` to adjust the indentation level.,operator_serialize} - @since version 1.0.0 + @since version 1.0.0; indentaction character added in version 3.0.0 */ friend std::ostream& operator<<(std::ostream& o, const basic_json& j) { @@ -7330,7 +7343,7 @@ class basic_json o.width(0); // do the actual serialization - serializer s(output_adapter::create(o)); + serializer s(output_adapter::create(o), o.fill()); s.dump(j, pretty_print, static_cast(indentation)); return o; } diff --git a/test/src/unit-convenience.cpp b/test/src/unit-convenience.cpp index 5e16962d..f757541d 100644 --- a/test/src/unit-convenience.cpp +++ b/test/src/unit-convenience.cpp @@ -53,7 +53,7 @@ TEST_CASE("convenience functions") const char* escaped) { std::stringstream ss; - json::serializer s(json::output_adapter::create(ss)); + json::serializer s(json::output_adapter::create(ss), ' '); s.dump_escaped(original); CHECK(ss.str() == escaped); }; diff --git a/test/src/unit-inspection.cpp b/test/src/unit-inspection.cpp index 4900e425..8b294bb8 100644 --- a/test/src/unit-inspection.cpp +++ b/test/src/unit-inspection.cpp @@ -207,6 +207,12 @@ TEST_CASE("object inspection") "{\n\"array\": [\n1,\n2,\n3,\n4\n],\n\"boolean\": false,\n\"null\": null,\n\"number\": 42,\n\"object\": {},\n\"string\": \"Hello world\"\n}"); } + SECTION("indent=1, space='\t'") + { + CHECK(j.dump(1, '\t') == + "{\n\t\"array\": [\n\t\t1,\n\t\t2,\n\t\t3,\n\t\t4\n\t],\n\t\"boolean\": false,\n\t\"null\": null,\n\t\"number\": 42,\n\t\"object\": {},\n\t\"string\": \"Hello world\"\n}"); + } + SECTION("indent=4") { CHECK(j.dump(4) == diff --git a/test/src/unit-serialization.cpp b/test/src/unit-serialization.cpp index 3df56cdc..fb500ed9 100644 --- a/test/src/unit-serialization.cpp +++ b/test/src/unit-serialization.cpp @@ -51,6 +51,15 @@ TEST_CASE("serialization") CHECK(ss.str() == "[\n \"foo\",\n 1,\n 2,\n 3,\n false,\n {\n \"one\": 1\n }\n]"); } + + SECTION("given fill") + { + std::stringstream ss; + json j = {"foo", 1, 2, 3, false, {{"one", 1}}}; + ss << std::setw(1) << std::setfill('\t') << j; + CHECK(ss.str() == + "[\n\t\"foo\",\n\t1,\n\t2,\n\t3,\n\tfalse,\n\t{\n\t\t\"one\": 1\n\t}\n]"); + } } SECTION("operator>>") @@ -72,5 +81,16 @@ TEST_CASE("serialization") CHECK(ss.str() == "[\n \"foo\",\n 1,\n 2,\n 3,\n false,\n {\n \"one\": 1\n }\n]"); } + + SECTION("given fill") + { + std::stringstream ss; + json j = {"foo", 1, 2, 3, false, {{"one", 1}}}; + ss.width(1); + ss.fill('\t'); + j >> ss; + CHECK(ss.str() == + "[\n\t\"foo\",\n\t1,\n\t2,\n\t3,\n\tfalse,\n\t{\n\t\t\"one\": 1\n\t}\n]"); + } } } From 31fe2f5825d2268138060a91b7d0305293149078 Mon Sep 17 00:00:00 2001 From: Jay Sistar Date: Mon, 8 May 2017 13:35:00 +0000 Subject: [PATCH 184/530] Adding support for Meson. --- README.md | 24 ++++++++++++++++++++++++ meson.build | 7 +++++++ 2 files changed, 31 insertions(+) create mode 100644 meson.build diff --git a/README.md b/README.md index fe92f9f1..1bf3d67b 100644 --- a/README.md +++ b/README.md @@ -66,6 +66,30 @@ to the files you want to use JSON objects. That's it. Do not forget to set the n :beer: If you are using OS X and [Homebrew](http://brew.sh), just type `brew tap nlohmann/json` and `brew install nlohmann_json` and you're set. If you want the bleeding edge rather than the latest release, use `brew install nlohmann_json --HEAD`. +If you are using the Meson Build System, then add a file called `nlohmann_json.wrap` to a file to your `subprojects` directory with the following 4 lines: + +``` +[wrap-git] +directory=nlohmann_json +url=https://github.com/nlohmann/json.git +revision=head +``` + +You can then create a reference to it in your `meson.build` file by adding the line: + +``` +nlohmann_json_sp = subproject('nlohmann_json') +``` + +Here's an example of an executable called `awesomeness` that uses this library through the reference created above. + +``` +executable('awesomeness', + 'awesomeness.cpp', + dependencies: nlohmann_json_sp.get_variable('nlohmann_json_dep') +) +``` + :warning: [Version 3.0.0](https://github.com/nlohmann/json/wiki/Road-toward-3.0.0) is currently under development. Branch `develop` is used for the ongoing work and is probably **unstable**. Please use the `master` branch for the last stable version 2.1.1. diff --git a/meson.build b/meson.build new file mode 100644 index 00000000..2cfc9628 --- /dev/null +++ b/meson.build @@ -0,0 +1,7 @@ +project('nlohmann_json', 'cpp') + +nlohmann_json_inc = include_directories('src') + +nlohmann_json_dep = declare_dependency( + include_directories : nlohmann_json_inc +) From 0719287e4474b9d0ddc6ba5be5533d3317a6a85e Mon Sep 17 00:00:00 2001 From: Jay Sistar Date: Tue, 9 May 2017 02:01:55 +0000 Subject: [PATCH 185/530] Reducing chatter about Meson. --- README.md | 23 +---------------------- 1 file changed, 1 insertion(+), 22 deletions(-) diff --git a/README.md b/README.md index 1bf3d67b..0637136e 100644 --- a/README.md +++ b/README.md @@ -66,30 +66,9 @@ to the files you want to use JSON objects. That's it. Do not forget to set the n :beer: If you are using OS X and [Homebrew](http://brew.sh), just type `brew tap nlohmann/json` and `brew install nlohmann_json` and you're set. If you want the bleeding edge rather than the latest release, use `brew install nlohmann_json --HEAD`. -If you are using the Meson Build System, then add a file called `nlohmann_json.wrap` to a file to your `subprojects` directory with the following 4 lines: - -``` -[wrap-git] -directory=nlohmann_json -url=https://github.com/nlohmann/json.git -revision=head -``` - +If you are using the Meson Build System, then you can wrap this repo as a subproject. You can then create a reference to it in your `meson.build` file by adding the line: -``` -nlohmann_json_sp = subproject('nlohmann_json') -``` - -Here's an example of an executable called `awesomeness` that uses this library through the reference created above. - -``` -executable('awesomeness', - 'awesomeness.cpp', - dependencies: nlohmann_json_sp.get_variable('nlohmann_json_dep') -) -``` - :warning: [Version 3.0.0](https://github.com/nlohmann/json/wiki/Road-toward-3.0.0) is currently under development. Branch `develop` is used for the ongoing work and is probably **unstable**. Please use the `master` branch for the last stable version 2.1.1. From 87744fd594cb838c8d3eb26226a0f9c76e2ca85d Mon Sep 17 00:00:00 2001 From: Jay Sistar Date: Tue, 9 May 2017 02:04:02 +0000 Subject: [PATCH 186/530] Missed one. --- README.md | 1 - 1 file changed, 1 deletion(-) diff --git a/README.md b/README.md index 0637136e..6ef08a56 100644 --- a/README.md +++ b/README.md @@ -67,7 +67,6 @@ to the files you want to use JSON objects. That's it. Do not forget to set the n :beer: If you are using OS X and [Homebrew](http://brew.sh), just type `brew tap nlohmann/json` and `brew install nlohmann_json` and you're set. If you want the bleeding edge rather than the latest release, use `brew install nlohmann_json --HEAD`. If you are using the Meson Build System, then you can wrap this repo as a subproject. -You can then create a reference to it in your `meson.build` file by adding the line: :warning: [Version 3.0.0](https://github.com/nlohmann/json/wiki/Road-toward-3.0.0) is currently under development. Branch `develop` is used for the ongoing work and is probably **unstable**. Please use the `master` branch for the last stable version 2.1.1. From 2afbd33472a2b64ccf57767e90427bba6b469be2 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Wed, 10 May 2017 12:06:24 +0200 Subject: [PATCH 187/530] :hammer: working on #367 Test cases succeed as expected, but the example in #367 is not fully realized yet. --- src/json.hpp | 4 +-- test/src/unit-testsuites.cpp | 54 ++++++++++++++++++++++++++++++++++-- 2 files changed, 54 insertions(+), 4 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 14e687d9..62b4cfb7 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -7618,7 +7618,7 @@ class basic_json JSON_DEPRECATED friend std::istream& operator<<(basic_json& j, std::istream& i) { - j = parser(input_adapter::create(i)).parse(true); + j = parser(input_adapter::create(i)).parse(false); return i; } @@ -7650,7 +7650,7 @@ class basic_json */ friend std::istream& operator>>(std::istream& i, basic_json& j) { - j = parser(input_adapter::create(i)).parse(true); + j = parser(input_adapter::create(i)).parse(false); return i; } diff --git a/test/src/unit-testsuites.cpp b/test/src/unit-testsuites.cpp index 798bc5c7..d3dbaf4b 100644 --- a/test/src/unit-testsuites.cpp +++ b/test/src/unit-testsuites.cpp @@ -75,11 +75,29 @@ TEST_CASE("compliance tests from json.org") "test/data/json_tests/fail32.json", "test/data/json_tests/fail33.json" }) + { + CAPTURE(filename); + std::ifstream f(filename); + CHECK_THROWS_AS(json::parse(f), json::parse_error); + } + } + + SECTION("no failures with trailing literals (relaxed)") + { + // these tests fail above, because the parser does not end on EOF; + // they succeed when the operator>> is used, because it does not + // have this constraint + for (auto filename : + { + "test/data/json_tests/fail7.json", + "test/data/json_tests/fail8.json", + "test/data/json_tests/fail10.json", + }) { CAPTURE(filename); std::ifstream f(filename); json j; - CHECK_THROWS_AS(f >> j, json::parse_error); + CHECK_NOTHROW(f >> j); } } @@ -751,11 +769,43 @@ TEST_CASE("nst's JSONTestSuite") "test/data/nst_json_testsuite/test_parsing/n_structure_whitespace_formfeed.json" } ) + { + CAPTURE(filename); + std::ifstream f(filename); + CHECK_THROWS_AS(json::parse(f), json::parse_error); + } + } + + SECTION("n -> y (relaxed)") + { + // these tests fail above, because the parser does not end on EOF; + // they succeed when the operator>> is used, because it does not + // have this constraint + for (auto filename : + { + "test/data/nst_json_testsuite/test_parsing/n_array_comma_after_close.json", + "test/data/nst_json_testsuite/test_parsing/n_array_extra_close.json", + "test/data/nst_json_testsuite/test_parsing/n_object_trailing_comment.json", + "test/data/nst_json_testsuite/test_parsing/n_object_trailing_comment_open.json", + "test/data/nst_json_testsuite/test_parsing/n_object_trailing_comment_slash_open.json", + "test/data/nst_json_testsuite/test_parsing/n_object_trailing_comment_slash_open_incomplete.json", + "test/data/nst_json_testsuite/test_parsing/n_object_with_trailing_garbage.json", + "test/data/nst_json_testsuite/test_parsing/n_string_with_trailing_garbage.json", + "test/data/nst_json_testsuite/test_parsing/n_structure_array_trailing_garbage.json", + "test/data/nst_json_testsuite/test_parsing/n_structure_array_with_extra_array_close.json", + "test/data/nst_json_testsuite/test_parsing/n_structure_close_unopened_array.json", + "test/data/nst_json_testsuite/test_parsing/n_structure_double_array.json", + "test/data/nst_json_testsuite/test_parsing/n_structure_number_with_trailing_garbage.json", + "test/data/nst_json_testsuite/test_parsing/n_structure_object_followed_by_closing_object.json", + "test/data/nst_json_testsuite/test_parsing/n_structure_object_with_trailing_garbage.json", + "test/data/nst_json_testsuite/test_parsing/n_structure_trailing_#.json" + } + ) { CAPTURE(filename); std::ifstream f(filename); json j; - CHECK_THROWS_AS(f >> j, json::parse_error); + CHECK_NOTHROW(f >> j); } } From 6a656ed375d96b64ddbb0d7d364fb03627f9b3ac Mon Sep 17 00:00:00 2001 From: Jay Sistar Date: Wed, 10 May 2017 13:20:52 +0000 Subject: [PATCH 188/530] Adding first and second properties to iteration_proxy_internal --- src/json.hpp | 43 ++++++++++++++++++++++++++++++++++++++++++- 1 file changed, 42 insertions(+), 1 deletion(-) diff --git a/src/json.hpp b/src/json.hpp index 62b4cfb7..8e2d8749 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -7877,6 +7877,44 @@ class basic_json class iteration_proxy { private: + /// helper class for first "property" + template + class iterator_key_property + { + private: + /// the reference to the proxy + ProxyType& proxy; + + public: + explicit iterator_key_property(ProxyType& proxyRef) noexcept + : proxy(proxyRef) {} + + /// conversion operator (calls key()) + operator typename basic_json::string_t() const + { + return proxy.key(); + } + }; + + /// helper class for second "property" + template + class iterator_value_property + { + private: + /// the reference to the proxy + ProxyType& proxy; + + public: + explicit iterator_value_property(ProxyType& proxyRef) noexcept + : proxy(proxyRef) {} + + /// conversion operator (calls value()) + operator typename IteratorType::reference() const + { + return proxy.value(); + } + }; + /// helper class for iteration class iteration_proxy_internal { @@ -7887,8 +7925,11 @@ class basic_json size_t array_index = 0; public: + iterator_key_property first; + iterator_value_property second; + explicit iteration_proxy_internal(IteratorType it) noexcept - : anchor(it) + : anchor(it), first(*this), second(*this) {} /// dereference operator (needed for range-based for) From d6c78dacd8f2d7461073f0956a2b68e4145016f7 Mon Sep 17 00:00:00 2001 From: Jay Sistar Date: Wed, 10 May 2017 17:53:27 +0000 Subject: [PATCH 189/530] Fixing assignement for iterator wrapper second, and adding unit test --- src/json.hpp | 8 + test/CMakeLists.txt | 1 + .../unit-iterator_wrapper_first_second.cpp | 729 ++++++++++++++++++ 3 files changed, 738 insertions(+) create mode 100644 test/src/unit-iterator_wrapper_first_second.cpp diff --git a/src/json.hpp b/src/json.hpp index 8e2d8749..a4e1275e 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -7913,6 +7913,14 @@ class basic_json { return proxy.value(); } + + /// assignment operator (calls value()) + template + iterator_value_property& operator=(const ValueType& value) + { + proxy.value() = value; + return *this; + } }; /// helper class for iteration diff --git a/test/CMakeLists.txt b/test/CMakeLists.txt index 25e345c2..36cc52b5 100644 --- a/test/CMakeLists.txt +++ b/test/CMakeLists.txt @@ -30,6 +30,7 @@ add_executable(${JSON_UNITTEST_TARGET_NAME} "src/unit-element_access2.cpp" "src/unit-inspection.cpp" "src/unit-iterator_wrapper.cpp" + "src/unit-iterator_wrapper_first_second.cpp" "src/unit-iterators1.cpp" "src/unit-iterators2.cpp" "src/unit-json_patch.cpp" diff --git a/test/src/unit-iterator_wrapper_first_second.cpp b/test/src/unit-iterator_wrapper_first_second.cpp new file mode 100644 index 00000000..6dceef8b --- /dev/null +++ b/test/src/unit-iterator_wrapper_first_second.cpp @@ -0,0 +1,729 @@ +/* + __ _____ _____ _____ + __| | __| | | | JSON for Modern C++ (test suite) +| | |__ | | | | | | version 2.1.1 +|_____|_____|_____|_|___| https://github.com/nlohmann/json + +Licensed under the MIT License . +Copyright (c) 2013-2017 Niels Lohmann . + +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. +*/ + +#include "catch.hpp" + +#include "json.hpp" +using nlohmann::json; + +TEST_CASE("iterator_wrapper_first_second") +{ + SECTION("object") + { + SECTION("value") + { + json j = {{"A", 1}, {"B", 2}}; + int counter = 1; + + for (auto i : json::iterator_wrapper(j)) + { + switch (counter++) + { + case 1: + { + CHECK(i.first == "A"); + CHECK(i.second == json(1)); + break; + } + + case 2: + { + CHECK(i.first == "B"); + CHECK(i.second == json(2)); + break; + } + + default: + { + break; + } + } + } + + CHECK(counter == 3); + } + + SECTION("reference") + { + json j = {{"A", 1}, {"B", 2}}; + int counter = 1; + + for (auto& i : json::iterator_wrapper(j)) + { + switch (counter++) + { + case 1: + { + CHECK(i.first == "A"); + CHECK(i.second == json(1)); + + // change the value + i.second = json(11); + CHECK(i.second == json(11)); + break; + } + + case 2: + { + CHECK(i.first == "B"); + CHECK(i.second == json(2)); + + // change the value + i.second = json(22); + CHECK(i.second == json(22)); + break; + } + + default: + { + break; + } + } + } + + CHECK(counter == 3); + + // check if values where changed + CHECK(j == json({{"A", 11}, {"B", 22}})); + } + + SECTION("const value") + { + json j = {{"A", 1}, {"B", 2}}; + int counter = 1; + + for (const auto i : json::iterator_wrapper(j)) + { + switch (counter++) + { + case 1: + { + CHECK(i.first == "A"); + CHECK(i.second == json(1)); + break; + } + + case 2: + { + CHECK(i.first == "B"); + CHECK(i.second == json(2)); + break; + } + + default: + { + break; + } + } + } + + CHECK(counter == 3); + } + + SECTION("const reference") + { + json j = {{"A", 1}, {"B", 2}}; + int counter = 1; + + for (const auto& i : json::iterator_wrapper(j)) + { + switch (counter++) + { + case 1: + { + CHECK(i.first == "A"); + CHECK(i.second == json(1)); + break; + } + + case 2: + { + CHECK(i.first == "B"); + CHECK(i.second == json(2)); + break; + } + + default: + { + break; + } + } + } + + CHECK(counter == 3); + } + } + + SECTION("const object") + { + SECTION("value") + { + const json j = {{"A", 1}, {"B", 2}}; + int counter = 1; + + for (auto i : json::iterator_wrapper(j)) + { + switch (counter++) + { + case 1: + { + CHECK(i.first == "A"); + CHECK(i.second == json(1)); + break; + } + + case 2: + { + CHECK(i.first == "B"); + CHECK(i.second == json(2)); + break; + } + + default: + { + break; + } + } + } + + CHECK(counter == 3); + } + + SECTION("reference") + { + const json j = {{"A", 1}, {"B", 2}}; + int counter = 1; + + for (auto& i : json::iterator_wrapper(j)) + { + switch (counter++) + { + case 1: + { + CHECK(i.first == "A"); + CHECK(i.second == json(1)); + break; + } + + case 2: + { + CHECK(i.first == "B"); + CHECK(i.second == json(2)); + break; + } + + default: + { + break; + } + } + } + + CHECK(counter == 3); + } + + SECTION("const value") + { + const json j = {{"A", 1}, {"B", 2}}; + int counter = 1; + + for (const auto i : json::iterator_wrapper(j)) + { + switch (counter++) + { + case 1: + { + CHECK(i.first == "A"); + CHECK(i.second == json(1)); + break; + } + + case 2: + { + CHECK(i.first == "B"); + CHECK(i.second == json(2)); + break; + } + + default: + { + break; + } + } + } + + CHECK(counter == 3); + } + + SECTION("const reference") + { + const json j = {{"A", 1}, {"B", 2}}; + int counter = 1; + + for (const auto& i : json::iterator_wrapper(j)) + { + switch (counter++) + { + case 1: + { + CHECK(i.first == "A"); + CHECK(i.second == json(1)); + break; + } + + case 2: + { + CHECK(i.first == "B"); + CHECK(i.second == json(2)); + break; + } + + default: + { + break; + } + } + } + + CHECK(counter == 3); + } + } + + SECTION("array") + { + SECTION("value") + { + json j = {"A", "B"}; + int counter = 1; + + for (auto i : json::iterator_wrapper(j)) + { + switch (counter++) + { + case 1: + { + CHECK(i.first == "0"); + CHECK(i.second == "A"); + break; + } + + case 2: + { + CHECK(i.first == "1"); + CHECK(i.second == "B"); + break; + } + + default: + { + break; + } + } + } + + CHECK(counter == 3); + } + + SECTION("reference") + { + json j = {"A", "B"}; + int counter = 1; + + for (auto& i : json::iterator_wrapper(j)) + { + switch (counter++) + { + case 1: + { + CHECK(i.first == "0"); + CHECK(i.second == "A"); + + // change the value + i.second = "AA"; + CHECK(i.second == "AA"); + break; + } + + case 2: + { + CHECK(i.first == "1"); + CHECK(i.second == "B"); + + // change the value + i.second = "BB"; + CHECK(i.second == "BB"); + break; + } + + default: + { + break; + } + } + } + + CHECK(counter == 3); + + // check if values where changed + CHECK(j == json({"AA", "BB"})); + } + + SECTION("const value") + { + json j = {"A", "B"}; + int counter = 1; + + for (const auto i : json::iterator_wrapper(j)) + { + switch (counter++) + { + case 1: + { + CHECK(i.first == "0"); + CHECK(i.second == "A"); + break; + } + + case 2: + { + CHECK(i.first == "1"); + CHECK(i.second == "B"); + break; + } + + default: + { + break; + } + } + } + + CHECK(counter == 3); + } + + SECTION("const reference") + { + json j = {"A", "B"}; + int counter = 1; + + for (const auto& i : json::iterator_wrapper(j)) + { + switch (counter++) + { + case 1: + { + CHECK(i.first == "0"); + CHECK(i.second == "A"); + break; + } + + case 2: + { + CHECK(i.first == "1"); + CHECK(i.second == "B"); + break; + } + + default: + { + break; + } + } + } + + CHECK(counter == 3); + } + } + + SECTION("const array") + { + SECTION("value") + { + const json j = {"A", "B"}; + int counter = 1; + + for (auto i : json::iterator_wrapper(j)) + { + switch (counter++) + { + case 1: + { + CHECK(i.first == "0"); + CHECK(i.second == "A"); + break; + } + + case 2: + { + CHECK(i.first == "1"); + CHECK(i.second == "B"); + break; + } + + default: + { + break; + } + } + } + + CHECK(counter == 3); + } + + SECTION("reference") + { + const json j = {"A", "B"}; + int counter = 1; + + for (auto& i : json::iterator_wrapper(j)) + { + switch (counter++) + { + case 1: + { + CHECK(i.first == "0"); + CHECK(i.second == "A"); + break; + } + + case 2: + { + CHECK(i.first == "1"); + CHECK(i.second == "B"); + break; + } + + default: + { + break; + } + } + } + + CHECK(counter == 3); + } + + SECTION("const value") + { + const json j = {"A", "B"}; + int counter = 1; + + for (const auto i : json::iterator_wrapper(j)) + { + switch (counter++) + { + case 1: + { + CHECK(i.first == "0"); + CHECK(i.second == "A"); + break; + } + + case 2: + { + CHECK(i.first == "1"); + CHECK(i.second == "B"); + break; + } + + default: + { + break; + } + } + } + + CHECK(counter == 3); + } + + SECTION("const reference") + { + const json j = {"A", "B"}; + int counter = 1; + + for (const auto& i : json::iterator_wrapper(j)) + { + switch (counter++) + { + case 1: + { + CHECK(i.first == "0"); + CHECK(i.second == "A"); + break; + } + + case 2: + { + CHECK(i.first == "1"); + CHECK(i.second == "B"); + break; + } + + default: + { + break; + } + } + } + + CHECK(counter == 3); + } + } + + SECTION("primitive") + { + SECTION("value") + { + json j = 1; + int counter = 1; + + for (auto i : json::iterator_wrapper(j)) + { + ++counter; + CHECK(i.first == ""); + CHECK(i.second == json(1)); + } + + CHECK(counter == 2); + } + + SECTION("reference") + { + json j = 1; + int counter = 1; + + for (auto& i : json::iterator_wrapper(j)) + { + ++counter; + CHECK(i.first == ""); + CHECK(i.second == json(1)); + + // change value + i.second = json(2); + } + + CHECK(counter == 2); + + // check if value has changed + CHECK(j == json(2)); + } + + SECTION("const value") + { + json j = 1; + int counter = 1; + + for (const auto i : json::iterator_wrapper(j)) + { + ++counter; + CHECK(i.first == ""); + CHECK(i.second == json(1)); + } + + CHECK(counter == 2); + } + + SECTION("const reference") + { + json j = 1; + int counter = 1; + + for (const auto& i : json::iterator_wrapper(j)) + { + ++counter; + CHECK(i.first == ""); + CHECK(i.second == json(1)); + } + + CHECK(counter == 2); + } + } + + SECTION("const primitive") + { + SECTION("value") + { + const json j = 1; + int counter = 1; + + for (auto i : json::iterator_wrapper(j)) + { + ++counter; + CHECK(i.first == ""); + CHECK(i.second == json(1)); + } + + CHECK(counter == 2); + } + + SECTION("reference") + { + const json j = 1; + int counter = 1; + + for (auto& i : json::iterator_wrapper(j)) + { + ++counter; + CHECK(i.first == ""); + CHECK(i.second == json(1)); + } + + CHECK(counter == 2); + } + + SECTION("const value") + { + const json j = 1; + int counter = 1; + + for (const auto i : json::iterator_wrapper(j)) + { + ++counter; + CHECK(i.first == ""); + CHECK(i.second == json(1)); + } + + CHECK(counter == 2); + } + + SECTION("const reference") + { + const json j = 1; + int counter = 1; + + for (const auto& i : json::iterator_wrapper(j)) + { + ++counter; + CHECK(i.first == ""); + CHECK(i.second == json(1)); + } + + CHECK(counter == 2); + } + } +} From 00d841bfda5dea2d21261bdefa62c9274554bd05 Mon Sep 17 00:00:00 2001 From: Jay Sistar Date: Thu, 11 May 2017 14:50:59 +0000 Subject: [PATCH 190/530] Adding equal and not equal operators to proxys. --- src/json.hpp | 28 ++++++++++++++++++++++++++++ 1 file changed, 28 insertions(+) diff --git a/src/json.hpp b/src/json.hpp index a4e1275e..b799f63b 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -7894,6 +7894,20 @@ class basic_json { return proxy.key(); } + + /// equal operator (calls key()) + template + bool operator==(const KeyType& key) + { + return proxy.key() == key; + } + + /// not equal operator (calls key()) + template + bool operator!=(const KeyType& key) + { + return proxy.key() != key; + } }; /// helper class for second "property" @@ -7914,6 +7928,20 @@ class basic_json return proxy.value(); } + /// equal operator (calls value()) + template + bool operator==(const ValueType& value) + { + return proxy.value() == value; + } + + /// not equal operator (calls value()) + template + bool operator!=(const ValueType& value) + { + return proxy.value() != value; + } + /// assignment operator (calls value()) template iterator_value_property& operator=(const ValueType& value) From b78457b792501803eadb7b48e04178a1ec570b1e Mon Sep 17 00:00:00 2001 From: Jay Sistar Date: Thu, 11 May 2017 13:22:47 -0400 Subject: [PATCH 191/530] Making comparison operators const. --- src/json.hpp | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index b799f63b..6e5803ca 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -7897,14 +7897,14 @@ class basic_json /// equal operator (calls key()) template - bool operator==(const KeyType& key) + bool operator==(const KeyType& key) const { return proxy.key() == key; } /// not equal operator (calls key()) template - bool operator!=(const KeyType& key) + bool operator!=(const KeyType& key) const { return proxy.key() != key; } @@ -7930,14 +7930,14 @@ class basic_json /// equal operator (calls value()) template - bool operator==(const ValueType& value) + bool operator==(const ValueType& value) const { return proxy.value() == value; } /// not equal operator (calls value()) template - bool operator!=(const ValueType& value) + bool operator!=(const ValueType& value) const { return proxy.value() != value; } From 978255da8c887dace2a67e96320e54736d4ec582 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sat, 13 May 2017 11:57:15 +0200 Subject: [PATCH 192/530] :memo: updated contribution guidelines --- .github/CONTRIBUTING.md | 16 +--------------- .github/PULL_REQUEST_TEMPLATE.md | 24 +++--------------------- 2 files changed, 4 insertions(+), 36 deletions(-) diff --git a/.github/CONTRIBUTING.md b/.github/CONTRIBUTING.md index b11f19fd..aae10000 100644 --- a/.github/CONTRIBUTING.md +++ b/.github/CONTRIBUTING.md @@ -14,8 +14,6 @@ Usually, all issues are tracked publicly on [Github](https://github.com/nlohmann Please [create an issue](https://github.com/nlohmann/json/issues/new), assuming one does not already exist, and describe your concern. Note you need a [GitHub account](https://github.com/signup/free) for this. -If you want propose changes to the code, you need to download the [`re2c`](http://re2c.org) tool. - ## Describe your issue Clearly describe the issue: @@ -28,16 +26,7 @@ Clearly describe the issue: There are currently two files which need to be edited: -1. [`src/json.hpp.re2c`](https://github.com/nlohmann/json/blob/master/src/json.hpp.re2c) (note the `.re2c` suffix) - This file contains a comment section which describes the JSON lexic. This section is translated by [`re2c`](http://re2c.org) into file [`src/json.hpp`](https://github.com/nlohmann/json/blob/master/src/json.hpp) which is plain "vanilla" C++11 code. (In fact, the generated lexer consists of some hundred lines of `goto`s, which is a hint you never want to edit this file...). - - If you only edit file `src/json.hpp` (without the `.re2c` suffix), your changes will be overwritten as soon as the lexer is touched again. To generate the `src/json.hpp` file which is actually used during compilation of the tests and all other code, please execute - - ```sh - make re2c - ``` - - To run [`re2c`](http://re2c.org) and generate/overwrite file `src/json.hpp` with your changes in file `src/json.hpp.re2c`. We currently use re2c version 0.16. Please also use this version, because other re2c versions tend to create code that differs a lot from which makes diffs unusable. - +1. [`src/json.hpp`](https://github.com/nlohmann/json/blob/master/src/json.hpp) 2. [`test/src/unit.cpp`](https://github.com/nlohmann/json/blob/master/test/unit.cpp) - This contains the [Catch](https://github.com/philsquared/Catch) unit tests which currently cover [100 %](https://coveralls.io/github/nlohmann/json) of the library's code. @@ -49,8 +38,6 @@ There are currently two files which need to be edited: The test cases are also executed with several different compilers on [Travis](https://travis-ci.org/nlohmann/json) once you open a pull request. -Please understand that I cannot accept pull requests changing only file `src/json.hpp`. - ## Note @@ -59,7 +46,6 @@ Please understand that I cannot accept pull requests changing only file `src/jso ## Please don't -- Please do not only make changes to file `src/json.hpp` -- please read the paragraph above and understand why `src/json.hpp.re2c` exists. - The C++11 support varies between different **compilers** and versions. Please note the [list of supported compilers](https://github.com/nlohmann/json/blob/master/README.md#supported-compilers). Some compilers like GCC 4.8 (and earlier), Clang 3.3 (and earlier), or Microsoft Visual Studio 13.0 and earlier are known not to work due to missing or incomplete C++11 support. Please refrain from proposing changes that work around these compiler's limitations with `#ifdef`s or other means. - Specifically, I am aware of compilation problems with **Microsoft Visual Studio** (there even is an [issue label](https://github.com/nlohmann/json/issues?utf8=✓&q=label%3A%22visual+studio%22+) for these kind of bugs). I understand that even in 2016, complete C++11 support isn't there yet. But please also understand that I do not want to drop features or uglify the code just to make Microsoft's sub-standard compiler happy. The past has shown that there are ways to express the functionality such that the code compiles with the most recent MSVC - unfortunately, this is not the main objective of the project. - Please refrain from proposing changes that would **break [JSON](http://json.org) conformance**. If you propose a conformant extension of JSON to be supported by the library, please motivate this extension. diff --git a/.github/PULL_REQUEST_TEMPLATE.md b/.github/PULL_REQUEST_TEMPLATE.md index 49d16659..59c2c51d 100644 --- a/.github/PULL_REQUEST_TEMPLATE.md +++ b/.github/PULL_REQUEST_TEMPLATE.md @@ -3,35 +3,18 @@ There are currently two files which need to be edited: -1. [`src/json.hpp.re2c`](https://github.com/nlohmann/json/blob/master/src/json.hpp.re2c) (note the `.re2c` suffix) - This file contains a comment section which describes the JSON lexic. This section is translated by [`re2c`](http://re2c.org) into file [`src/json.hpp`](https://github.com/nlohmann/json/blob/master/src/json.hpp) which is plain "vanilla" C++11 code. (In fact, the generated lexer consists of some hundred lines of `goto`s, which is a hint you never want to edit this file...). - - If you only edit file `src/json.hpp` (without the `.re2c` suffix), your changes will be overwritten as soon as the lexer is touched again. To generate the `src/json.hpp` file which is actually used during compilation of the tests and all other code, please execute - - ```sh - make re2c - ``` - - To run [`re2c`](http://re2c.org) and generate/overwrite file `src/json.hpp` with your changes in file `src/json.hpp.re2c`. - +1. [`src/json.hpp`](https://github.com/nlohmann/json/blob/master/src/json.hpp) 2. [`test/src/unit.cpp`](https://github.com/nlohmann/json/blob/master/test/unit.cpp) - This contains the [Catch](https://github.com/philsquared/Catch) unit tests which currently cover [100 %](https://coveralls.io/github/nlohmann/json) of the library's code. - If you add or change a feature, please also add a unit test to this file. The unit tests can be compiled with + If you add or change a feature, please also add a unit test to this file. The unit tests can be compiled and executed with ```sh - make - ``` - - and can be executed with - - ```sh - ./json_unit + make check ``` The test cases are also executed with several different compilers on [Travis](https://travis-ci.org/nlohmann/json) once you open a pull request. -Please understand that I cannot accept pull requests changing only file `src/json.hpp`. - ## Note @@ -39,7 +22,6 @@ Please understand that I cannot accept pull requests changing only file `src/jso ## Please don't -- Only make changes to file `src/json.hpp` -- please read the paragraph above and understand why `src/json.hpp.re2c` exists. - The C++11 support varies between different **compilers** and versions. Please note the [list of supported compilers](https://github.com/nlohmann/json/blob/master/README.md#supported-compilers). Some compilers like GCC 4.8 (and earlier), Clang 3.3 (and earlier), or Microsoft Visual Studio 13.0 and earlier are known not to work due to missing or incomplete C++11 support. Please refrain from proposing changes that work around these compiler's limitations with `#ifdef`s or other means. - Specifically, I am aware of compilation problems with **Microsoft Visual Studio** (there even is an [issue label](https://github.com/nlohmann/json/issues?utf8=✓&q=label%3A%22visual+studio%22+) for these kind of bugs). I understand that even in 2016, complete C++11 support isn't there yet. But please also understand that I do not want to drop features or uglify the code just to make Microsoft's sub-standard compiler happy. The past has shown that there are ways to express the functionality such that the code compiles with the most recent MSVC - unfortunately, this is not the main objective of the project. - Please refrain from proposing changes that would **break [JSON](http://json.org) conformance**. If you propose a conformant extension of JSON to be supported by the library, please motivate this extension. From 9b764ee5d671b41255d390ab9089f12036b2d38a Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sat, 13 May 2017 12:27:37 +0200 Subject: [PATCH 193/530] :checkered_flag: fix for #581 --- src/json.hpp | 18 +++++++++--------- 1 file changed, 9 insertions(+), 9 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 6e5803ca..7f3b30bb 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -7892,21 +7892,21 @@ class basic_json /// conversion operator (calls key()) operator typename basic_json::string_t() const { - return proxy.key(); + return proxy.key(); } /// equal operator (calls key()) template bool operator==(const KeyType& key) const { - return proxy.key() == key; + return proxy.key() == key; } /// not equal operator (calls key()) template bool operator!=(const KeyType& key) const { - return proxy.key() != key; + return proxy.key() != key; } }; @@ -7925,29 +7925,29 @@ class basic_json /// conversion operator (calls value()) operator typename IteratorType::reference() const { - return proxy.value(); + return proxy.value(); } /// equal operator (calls value()) template bool operator==(const ValueType& value) const { - return proxy.value() == value; + return proxy.value() == value; } /// not equal operator (calls value()) template bool operator!=(const ValueType& value) const { - return proxy.value() != value; + return proxy.value() != value; } /// assignment operator (calls value()) template iterator_value_property& operator=(const ValueType& value) { - proxy.value() = value; - return *this; + proxy.value() = value; + return *this; } }; @@ -9005,7 +9005,7 @@ class basic_json { // avoid reading too many characters const size_t max_length = static_cast(limit - start); - return std::string(start + offset, std::min({length, max_length - offset})); + return std::string(start + offset, std::min(length, max_length - offset)); } private: From dfa371c436fa14926544eb11f3207d0439cd2c43 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E9=83=AD=E8=8D=A3=E9=A3=9E?= Date: Sat, 13 May 2017 23:13:04 +0800 Subject: [PATCH 194/530] fix doxygen error of basic_json::get() --- src/json.hpp | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 7f3b30bb..2ed14964 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -3447,9 +3447,9 @@ class basic_json This overloads is chosen if: - @a ValueType is not @ref basic_json, - @ref json_serializer has a `from_json()` method of the form - `void from_json(const @ref basic_json&, ValueType&)`, and + `void from_json(const ` @ref basic_json`&, ValueType&)`, and - @ref json_serializer does not have a `from_json()` method of - the form `ValueType from_json(const @ref basic_json&)` + the form `ValueType from_json(const ` @ref basic_json`&)` @tparam ValueTypeCV the provided value type @tparam ValueType the returned value type @@ -3508,7 +3508,7 @@ class basic_json This overloads is chosen if: - @a ValueType is not @ref basic_json and - @ref json_serializer has a `from_json()` method of the form - `ValueType from_json(const @ref basic_json&)` + `ValueType from_json(const ` @ref basic_json`&)` @note If @ref json_serializer has both overloads of `from_json()`, this one is chosen. From 723c87560459eaacb3e3b3f0d4666bd2d02be317 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sat, 13 May 2017 19:56:38 +0200 Subject: [PATCH 195/530] :speech_balloon: replaced "backspace" with "backslash" #509 --- src/json.hpp | 2 +- test/src/unit-class_parser.cpp | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 7f3b30bb..36d74ef0 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -11586,7 +11586,7 @@ class basic_json // other characters after escape default: - error_message = "invalid string: forbidden character after backspace"; + error_message = "invalid string: forbidden character after backslash"; return token_type::parse_error; } diff --git a/test/src/unit-class_parser.cpp b/test/src/unit-class_parser.cpp index 5866b901..5a2ff350 100644 --- a/test/src/unit-class_parser.cpp +++ b/test/src/unit-class_parser.cpp @@ -798,7 +798,7 @@ TEST_CASE("parser class") if (c > 0x1f) { CHECK_THROWS_WITH(parse_string(s.c_str()).parse(), - "[json.exception.parse_error.101] parse error at 3: syntax error - invalid string: forbidden character after backspace; last read '\"\\" + std::string(1, static_cast(c)) + "'"); + "[json.exception.parse_error.101] parse error at 3: syntax error - invalid string: forbidden character after backslash; last read '\"\\" + std::string(1, static_cast(c)) + "'"); } break; } From b8dff3bc1674acb6a30cd733ecae545b93b1baeb Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E9=83=AD=E8=8D=A3=E9=A3=9E?= Date: Sun, 14 May 2017 07:50:26 +0800 Subject: [PATCH 196/530] remove the redundant @ref command --- src/json.hpp | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 2ed14964..f1f59b9a 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -3447,9 +3447,9 @@ class basic_json This overloads is chosen if: - @a ValueType is not @ref basic_json, - @ref json_serializer has a `from_json()` method of the form - `void from_json(const ` @ref basic_json`&, ValueType&)`, and + `void from_json(const basic_json&, ValueType&)`, and - @ref json_serializer does not have a `from_json()` method of - the form `ValueType from_json(const ` @ref basic_json`&)` + the form `ValueType from_json(const basic_json&)` @tparam ValueTypeCV the provided value type @tparam ValueType the returned value type @@ -3508,7 +3508,7 @@ class basic_json This overloads is chosen if: - @a ValueType is not @ref basic_json and - @ref json_serializer has a `from_json()` method of the form - `ValueType from_json(const ` @ref basic_json`&)` + `ValueType from_json(const basic_json&)` @note If @ref json_serializer has both overloads of `from_json()`, this one is chosen. From 0c3ffe12cdd9a28bbe467f1d567de41e08c3f984 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Mon, 15 May 2017 21:07:58 +0200 Subject: [PATCH 197/530] :rewind: reverting first/second experiment #350 --- README.md | 4 +- doc/examples/iterator_wrapper.cpp | 22 + doc/examples/iterator_wrapper.link | 1 + doc/examples/iterator_wrapper.output | 7 + src/json.hpp | 81 +- test/CMakeLists.txt | 1 - .../unit-iterator_wrapper_first_second.cpp | 729 ------------------ test/src/unit-unicode.cpp | 2 + 8 files changed, 37 insertions(+), 810 deletions(-) create mode 100644 doc/examples/iterator_wrapper.cpp create mode 100644 doc/examples/iterator_wrapper.link create mode 100644 doc/examples/iterator_wrapper.output delete mode 100644 test/src/unit-iterator_wrapper_first_second.cpp diff --git a/README.md b/README.md index 6ef08a56..5d84c3f2 100644 --- a/README.md +++ b/README.md @@ -66,7 +66,7 @@ to the files you want to use JSON objects. That's it. Do not forget to set the n :beer: If you are using OS X and [Homebrew](http://brew.sh), just type `brew tap nlohmann/json` and `brew install nlohmann_json` and you're set. If you want the bleeding edge rather than the latest release, use `brew install nlohmann_json --HEAD`. -If you are using the Meson Build System, then you can wrap this repo as a subproject. +If you are using the [Meson Build System](http://mesonbuild.com), then you can wrap this repo as a subproject. :warning: [Version 3.0.0](https://github.com/nlohmann/json/wiki/Road-toward-3.0.0) is currently under development. Branch `develop` is used for the ongoing work and is probably **unstable**. Please use the `master` branch for the last stable version 2.1.1. @@ -839,6 +839,7 @@ I deeply appreciate the help of the following people. - [ftillier](https://github.com/ftillier) fixed a compiler warning. - [tinloaf](https://github.com/tinloaf) made sure all pushed warnings are properly popped. - [Fytch](https://github.com/Fytch) found a bug in the documentation. +- [Jay Sistar](https://github.com/Type1J) implemented a Meson build description Thanks a lot for helping out! Please [let me know](mailto:mail@nlohmann.me) if I forgot someone. @@ -865,7 +866,6 @@ The library itself contains of a single header file licensed under the MIT licen - [**Github Changelog Generator**](https://github.com/skywinder/github-changelog-generator) to generate the [ChangeLog](https://github.com/nlohmann/json/blob/develop/ChangeLog.md) - [**libFuzzer**](http://llvm.org/docs/LibFuzzer.html) to implement fuzz testing for OSS-Fuzz - [**OSS-Fuzz**](https://github.com/google/oss-fuzz) for continuous fuzz testing of the library -- [**re2c**](http://re2c.org) to generate an automaton for the lexical analysis - [**send_to_wandbox**](https://github.com/nlohmann/json/blob/develop/doc/scripts/send_to_wandbox.py) to send code examples to [Wandbox](http://melpon.org/wandbox) - [**Travis**](https://travis-ci.org) for [continuous integration](https://travis-ci.org/nlohmann/json) on Linux and macOS - [**Valgrind**](http://valgrind.org) to check for correct memory management diff --git a/doc/examples/iterator_wrapper.cpp b/doc/examples/iterator_wrapper.cpp new file mode 100644 index 00000000..208bb705 --- /dev/null +++ b/doc/examples/iterator_wrapper.cpp @@ -0,0 +1,22 @@ +#include "json.hpp" + +using json = nlohmann::json; + +int main() +{ + // create JSON values + json j_object = {{"one", 1}, {"two", 2}}; + json j_array = {1, 2, 4, 8, 16}; + + // example for an object + for (auto& x : json::iterator_wrapper(j_object)) + { + std::cout << "key: " << x.key() << ", value: " << x.value() << '\n'; + } + + // example for an array + for (auto& x : json::iterator_wrapper(j_array)) + { + std::cout << "key: " << x.key() << ", value: " << x.value() << '\n'; + } +} diff --git a/doc/examples/iterator_wrapper.link b/doc/examples/iterator_wrapper.link new file mode 100644 index 00000000..c8f696c3 --- /dev/null +++ b/doc/examples/iterator_wrapper.link @@ -0,0 +1 @@ +online \ No newline at end of file diff --git a/doc/examples/iterator_wrapper.output b/doc/examples/iterator_wrapper.output new file mode 100644 index 00000000..89b09f52 --- /dev/null +++ b/doc/examples/iterator_wrapper.output @@ -0,0 +1,7 @@ +key: one, value: 1 +key: two, value: 2 +key: 0, value: 1 +key: 1, value: 2 +key: 2, value: 4 +key: 3, value: 8 +key: 4, value: 16 diff --git a/src/json.hpp b/src/json.hpp index 36d74ef0..3bfd1234 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -5204,6 +5204,8 @@ class basic_json reference to the JSON values is returned, so there is no access to the underlying iterator. + @liveexample{The following code shows how the wrapper is used,iterator_wrapper} + @note The name of this function is not yet final and may change in the future. */ @@ -7877,80 +7879,6 @@ class basic_json class iteration_proxy { private: - /// helper class for first "property" - template - class iterator_key_property - { - private: - /// the reference to the proxy - ProxyType& proxy; - - public: - explicit iterator_key_property(ProxyType& proxyRef) noexcept - : proxy(proxyRef) {} - - /// conversion operator (calls key()) - operator typename basic_json::string_t() const - { - return proxy.key(); - } - - /// equal operator (calls key()) - template - bool operator==(const KeyType& key) const - { - return proxy.key() == key; - } - - /// not equal operator (calls key()) - template - bool operator!=(const KeyType& key) const - { - return proxy.key() != key; - } - }; - - /// helper class for second "property" - template - class iterator_value_property - { - private: - /// the reference to the proxy - ProxyType& proxy; - - public: - explicit iterator_value_property(ProxyType& proxyRef) noexcept - : proxy(proxyRef) {} - - /// conversion operator (calls value()) - operator typename IteratorType::reference() const - { - return proxy.value(); - } - - /// equal operator (calls value()) - template - bool operator==(const ValueType& value) const - { - return proxy.value() == value; - } - - /// not equal operator (calls value()) - template - bool operator!=(const ValueType& value) const - { - return proxy.value() != value; - } - - /// assignment operator (calls value()) - template - iterator_value_property& operator=(const ValueType& value) - { - proxy.value() = value; - return *this; - } - }; - /// helper class for iteration class iteration_proxy_internal { @@ -7961,11 +7889,8 @@ class basic_json size_t array_index = 0; public: - iterator_key_property first; - iterator_value_property second; - explicit iteration_proxy_internal(IteratorType it) noexcept - : anchor(it), first(*this), second(*this) + : anchor(it) {} /// dereference operator (needed for range-based for) diff --git a/test/CMakeLists.txt b/test/CMakeLists.txt index 36cc52b5..25e345c2 100644 --- a/test/CMakeLists.txt +++ b/test/CMakeLists.txt @@ -30,7 +30,6 @@ add_executable(${JSON_UNITTEST_TARGET_NAME} "src/unit-element_access2.cpp" "src/unit-inspection.cpp" "src/unit-iterator_wrapper.cpp" - "src/unit-iterator_wrapper_first_second.cpp" "src/unit-iterators1.cpp" "src/unit-iterators2.cpp" "src/unit-json_patch.cpp" diff --git a/test/src/unit-iterator_wrapper_first_second.cpp b/test/src/unit-iterator_wrapper_first_second.cpp deleted file mode 100644 index 6dceef8b..00000000 --- a/test/src/unit-iterator_wrapper_first_second.cpp +++ /dev/null @@ -1,729 +0,0 @@ -/* - __ _____ _____ _____ - __| | __| | | | JSON for Modern C++ (test suite) -| | |__ | | | | | | version 2.1.1 -|_____|_____|_____|_|___| https://github.com/nlohmann/json - -Licensed under the MIT License . -Copyright (c) 2013-2017 Niels Lohmann . - -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. -*/ - -#include "catch.hpp" - -#include "json.hpp" -using nlohmann::json; - -TEST_CASE("iterator_wrapper_first_second") -{ - SECTION("object") - { - SECTION("value") - { - json j = {{"A", 1}, {"B", 2}}; - int counter = 1; - - for (auto i : json::iterator_wrapper(j)) - { - switch (counter++) - { - case 1: - { - CHECK(i.first == "A"); - CHECK(i.second == json(1)); - break; - } - - case 2: - { - CHECK(i.first == "B"); - CHECK(i.second == json(2)); - break; - } - - default: - { - break; - } - } - } - - CHECK(counter == 3); - } - - SECTION("reference") - { - json j = {{"A", 1}, {"B", 2}}; - int counter = 1; - - for (auto& i : json::iterator_wrapper(j)) - { - switch (counter++) - { - case 1: - { - CHECK(i.first == "A"); - CHECK(i.second == json(1)); - - // change the value - i.second = json(11); - CHECK(i.second == json(11)); - break; - } - - case 2: - { - CHECK(i.first == "B"); - CHECK(i.second == json(2)); - - // change the value - i.second = json(22); - CHECK(i.second == json(22)); - break; - } - - default: - { - break; - } - } - } - - CHECK(counter == 3); - - // check if values where changed - CHECK(j == json({{"A", 11}, {"B", 22}})); - } - - SECTION("const value") - { - json j = {{"A", 1}, {"B", 2}}; - int counter = 1; - - for (const auto i : json::iterator_wrapper(j)) - { - switch (counter++) - { - case 1: - { - CHECK(i.first == "A"); - CHECK(i.second == json(1)); - break; - } - - case 2: - { - CHECK(i.first == "B"); - CHECK(i.second == json(2)); - break; - } - - default: - { - break; - } - } - } - - CHECK(counter == 3); - } - - SECTION("const reference") - { - json j = {{"A", 1}, {"B", 2}}; - int counter = 1; - - for (const auto& i : json::iterator_wrapper(j)) - { - switch (counter++) - { - case 1: - { - CHECK(i.first == "A"); - CHECK(i.second == json(1)); - break; - } - - case 2: - { - CHECK(i.first == "B"); - CHECK(i.second == json(2)); - break; - } - - default: - { - break; - } - } - } - - CHECK(counter == 3); - } - } - - SECTION("const object") - { - SECTION("value") - { - const json j = {{"A", 1}, {"B", 2}}; - int counter = 1; - - for (auto i : json::iterator_wrapper(j)) - { - switch (counter++) - { - case 1: - { - CHECK(i.first == "A"); - CHECK(i.second == json(1)); - break; - } - - case 2: - { - CHECK(i.first == "B"); - CHECK(i.second == json(2)); - break; - } - - default: - { - break; - } - } - } - - CHECK(counter == 3); - } - - SECTION("reference") - { - const json j = {{"A", 1}, {"B", 2}}; - int counter = 1; - - for (auto& i : json::iterator_wrapper(j)) - { - switch (counter++) - { - case 1: - { - CHECK(i.first == "A"); - CHECK(i.second == json(1)); - break; - } - - case 2: - { - CHECK(i.first == "B"); - CHECK(i.second == json(2)); - break; - } - - default: - { - break; - } - } - } - - CHECK(counter == 3); - } - - SECTION("const value") - { - const json j = {{"A", 1}, {"B", 2}}; - int counter = 1; - - for (const auto i : json::iterator_wrapper(j)) - { - switch (counter++) - { - case 1: - { - CHECK(i.first == "A"); - CHECK(i.second == json(1)); - break; - } - - case 2: - { - CHECK(i.first == "B"); - CHECK(i.second == json(2)); - break; - } - - default: - { - break; - } - } - } - - CHECK(counter == 3); - } - - SECTION("const reference") - { - const json j = {{"A", 1}, {"B", 2}}; - int counter = 1; - - for (const auto& i : json::iterator_wrapper(j)) - { - switch (counter++) - { - case 1: - { - CHECK(i.first == "A"); - CHECK(i.second == json(1)); - break; - } - - case 2: - { - CHECK(i.first == "B"); - CHECK(i.second == json(2)); - break; - } - - default: - { - break; - } - } - } - - CHECK(counter == 3); - } - } - - SECTION("array") - { - SECTION("value") - { - json j = {"A", "B"}; - int counter = 1; - - for (auto i : json::iterator_wrapper(j)) - { - switch (counter++) - { - case 1: - { - CHECK(i.first == "0"); - CHECK(i.second == "A"); - break; - } - - case 2: - { - CHECK(i.first == "1"); - CHECK(i.second == "B"); - break; - } - - default: - { - break; - } - } - } - - CHECK(counter == 3); - } - - SECTION("reference") - { - json j = {"A", "B"}; - int counter = 1; - - for (auto& i : json::iterator_wrapper(j)) - { - switch (counter++) - { - case 1: - { - CHECK(i.first == "0"); - CHECK(i.second == "A"); - - // change the value - i.second = "AA"; - CHECK(i.second == "AA"); - break; - } - - case 2: - { - CHECK(i.first == "1"); - CHECK(i.second == "B"); - - // change the value - i.second = "BB"; - CHECK(i.second == "BB"); - break; - } - - default: - { - break; - } - } - } - - CHECK(counter == 3); - - // check if values where changed - CHECK(j == json({"AA", "BB"})); - } - - SECTION("const value") - { - json j = {"A", "B"}; - int counter = 1; - - for (const auto i : json::iterator_wrapper(j)) - { - switch (counter++) - { - case 1: - { - CHECK(i.first == "0"); - CHECK(i.second == "A"); - break; - } - - case 2: - { - CHECK(i.first == "1"); - CHECK(i.second == "B"); - break; - } - - default: - { - break; - } - } - } - - CHECK(counter == 3); - } - - SECTION("const reference") - { - json j = {"A", "B"}; - int counter = 1; - - for (const auto& i : json::iterator_wrapper(j)) - { - switch (counter++) - { - case 1: - { - CHECK(i.first == "0"); - CHECK(i.second == "A"); - break; - } - - case 2: - { - CHECK(i.first == "1"); - CHECK(i.second == "B"); - break; - } - - default: - { - break; - } - } - } - - CHECK(counter == 3); - } - } - - SECTION("const array") - { - SECTION("value") - { - const json j = {"A", "B"}; - int counter = 1; - - for (auto i : json::iterator_wrapper(j)) - { - switch (counter++) - { - case 1: - { - CHECK(i.first == "0"); - CHECK(i.second == "A"); - break; - } - - case 2: - { - CHECK(i.first == "1"); - CHECK(i.second == "B"); - break; - } - - default: - { - break; - } - } - } - - CHECK(counter == 3); - } - - SECTION("reference") - { - const json j = {"A", "B"}; - int counter = 1; - - for (auto& i : json::iterator_wrapper(j)) - { - switch (counter++) - { - case 1: - { - CHECK(i.first == "0"); - CHECK(i.second == "A"); - break; - } - - case 2: - { - CHECK(i.first == "1"); - CHECK(i.second == "B"); - break; - } - - default: - { - break; - } - } - } - - CHECK(counter == 3); - } - - SECTION("const value") - { - const json j = {"A", "B"}; - int counter = 1; - - for (const auto i : json::iterator_wrapper(j)) - { - switch (counter++) - { - case 1: - { - CHECK(i.first == "0"); - CHECK(i.second == "A"); - break; - } - - case 2: - { - CHECK(i.first == "1"); - CHECK(i.second == "B"); - break; - } - - default: - { - break; - } - } - } - - CHECK(counter == 3); - } - - SECTION("const reference") - { - const json j = {"A", "B"}; - int counter = 1; - - for (const auto& i : json::iterator_wrapper(j)) - { - switch (counter++) - { - case 1: - { - CHECK(i.first == "0"); - CHECK(i.second == "A"); - break; - } - - case 2: - { - CHECK(i.first == "1"); - CHECK(i.second == "B"); - break; - } - - default: - { - break; - } - } - } - - CHECK(counter == 3); - } - } - - SECTION("primitive") - { - SECTION("value") - { - json j = 1; - int counter = 1; - - for (auto i : json::iterator_wrapper(j)) - { - ++counter; - CHECK(i.first == ""); - CHECK(i.second == json(1)); - } - - CHECK(counter == 2); - } - - SECTION("reference") - { - json j = 1; - int counter = 1; - - for (auto& i : json::iterator_wrapper(j)) - { - ++counter; - CHECK(i.first == ""); - CHECK(i.second == json(1)); - - // change value - i.second = json(2); - } - - CHECK(counter == 2); - - // check if value has changed - CHECK(j == json(2)); - } - - SECTION("const value") - { - json j = 1; - int counter = 1; - - for (const auto i : json::iterator_wrapper(j)) - { - ++counter; - CHECK(i.first == ""); - CHECK(i.second == json(1)); - } - - CHECK(counter == 2); - } - - SECTION("const reference") - { - json j = 1; - int counter = 1; - - for (const auto& i : json::iterator_wrapper(j)) - { - ++counter; - CHECK(i.first == ""); - CHECK(i.second == json(1)); - } - - CHECK(counter == 2); - } - } - - SECTION("const primitive") - { - SECTION("value") - { - const json j = 1; - int counter = 1; - - for (auto i : json::iterator_wrapper(j)) - { - ++counter; - CHECK(i.first == ""); - CHECK(i.second == json(1)); - } - - CHECK(counter == 2); - } - - SECTION("reference") - { - const json j = 1; - int counter = 1; - - for (auto& i : json::iterator_wrapper(j)) - { - ++counter; - CHECK(i.first == ""); - CHECK(i.second == json(1)); - } - - CHECK(counter == 2); - } - - SECTION("const value") - { - const json j = 1; - int counter = 1; - - for (const auto i : json::iterator_wrapper(j)) - { - ++counter; - CHECK(i.first == ""); - CHECK(i.second == json(1)); - } - - CHECK(counter == 2); - } - - SECTION("const reference") - { - const json j = 1; - int counter = 1; - - for (const auto& i : json::iterator_wrapper(j)) - { - ++counter; - CHECK(i.first == ""); - CHECK(i.second == json(1)); - } - - CHECK(counter == 2); - } - } -} diff --git a/test/src/unit-unicode.cpp b/test/src/unit-unicode.cpp index 120941ee..17cd9aac 100644 --- a/test/src/unit-unicode.cpp +++ b/test/src/unit-unicode.cpp @@ -34,6 +34,8 @@ using nlohmann::json; #include +void check_utf8string(bool success_expected, int byte1, int byte2, int byte3, int byte4); + // create and check a JSON string with up to four UTF-8 bytes void check_utf8string(bool success_expected, int byte1, int byte2 = -1, int byte3 = -1, int byte4 = -1) { From 99cb095e89b9bb0c982e4691d418cfac188bad97 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sat, 20 May 2017 10:34:49 +0200 Subject: [PATCH 198/530] :bug: fix for #586 Also added a Travis builder with -std=c++1z --- .travis.yml | 10 ++++++++++ src/json.hpp | 2 +- 2 files changed, 11 insertions(+), 1 deletion(-) diff --git a/.travis.yml b/.travis.yml index 0d7026b0..089524f4 100644 --- a/.travis.yml +++ b/.travis.yml @@ -177,6 +177,16 @@ matrix: sources: ['ubuntu-toolchain-r-test'] packages: g++-6 + - os: linux + compiler: gcc + env: + - COMPILER=g++-6 + - CXXFLAGS=-std=c++1z + addons: + apt: + sources: ['ubuntu-toolchain-r-test'] + packages: g++-6 + # Linux / Clang - os: linux diff --git a/src/json.hpp b/src/json.hpp index 85d559d7..7bfb7359 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -3742,7 +3742,7 @@ class basic_json #ifndef _MSC_VER // fix for issue #167 operator<< ambiguity under VS2015 and not std::is_same>::value #endif -#if defined(_MSC_VER) && _MSC_VER >1900 && defined(_HAS_CXX17) && _HAS_CXX17 == 1 // fix for issue #464 +#if (defined(__cplusplus) && __cplusplus == 201703L) || (defined(_MSC_VER) && _MSC_VER >1900 && defined(_HAS_CXX17) && _HAS_CXX17 == 1) // fix for issue #464 and not std::is_same::value #endif , int >::type = 0 > From 9ddc69f3a9a3b6579830dbc3a22dfdcc07a65e9b Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sat, 20 May 2017 10:49:06 +0200 Subject: [PATCH 199/530] :bug: adjusting fix for future versions #586 --- src/json.hpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/json.hpp b/src/json.hpp index 7bfb7359..f522dea5 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -3742,7 +3742,7 @@ class basic_json #ifndef _MSC_VER // fix for issue #167 operator<< ambiguity under VS2015 and not std::is_same>::value #endif -#if (defined(__cplusplus) && __cplusplus == 201703L) || (defined(_MSC_VER) && _MSC_VER >1900 && defined(_HAS_CXX17) && _HAS_CXX17 == 1) // fix for issue #464 +#if (defined(__cplusplus) && __cplusplus >= 201703L) || (defined(_MSC_VER) && _MSC_VER >1900 && defined(_HAS_CXX17) && _HAS_CXX17 == 1) // fix for issue #464 and not std::is_same::value #endif , int >::type = 0 > From 8a35fdd8f549ba0e777ca49ce3444fd709ef445b Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sat, 20 May 2017 12:24:13 +0200 Subject: [PATCH 200/530] :hammer: overworked accept() function Fixed to make sure no token is "overlooked". --- src/json.hpp | 27 ++++++++++++++++----------- 1 file changed, 16 insertions(+), 11 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index f522dea5..02f0e7b5 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -12556,7 +12556,7 @@ scan_number_done: return false; } - if (strict and last_token != lexer::token_type::end_of_input) + if (strict and get_token() != lexer::token_type::end_of_input) { return false; } @@ -12787,6 +12787,14 @@ scan_number_done: /*! @brief the acutal acceptor + + @invariant 1. The last token is not yet processed. Therefore, the + caller of this function must make sure a token has + been read. + 2. When this function returns, the last token is processed. + That is, the last read character was already considered. + + This invariant makes sure that no token needs to be "unput". */ bool accept_internal() { @@ -12800,7 +12808,6 @@ scan_number_done: // closing } -> we are done if (last_token == lexer::token_type::end_object) { - get_token(); return true; } @@ -12828,6 +12835,7 @@ scan_number_done: } // comma -> next value + get_token(); if (last_token == lexer::token_type::value_separator) { get_token(); @@ -12840,7 +12848,6 @@ scan_number_done: return false; } - get_token(); return true; } } @@ -12853,7 +12860,6 @@ scan_number_done: // closing ] -> we are done if (last_token == lexer::token_type::end_array) { - get_token(); return true; } @@ -12867,6 +12873,7 @@ scan_number_done: } // comma -> next value + get_token(); if (last_token == lexer::token_type::value_separator) { get_token(); @@ -12879,20 +12886,18 @@ scan_number_done: return false; } - get_token(); return true; } } - case lexer::token_type::literal_null: - case lexer::token_type::value_string: - case lexer::token_type::literal_true: case lexer::token_type::literal_false: - case lexer::token_type::value_unsigned: - case lexer::token_type::value_integer: + case lexer::token_type::literal_null: + case lexer::token_type::literal_true: case lexer::token_type::value_float: + case lexer::token_type::value_integer: + case lexer::token_type::value_string: + case lexer::token_type::value_unsigned: { - get_token(); return true; } From 16b63d31975f283187cd4352fe9e9a7e754c49d7 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sat, 20 May 2017 12:44:13 +0200 Subject: [PATCH 201/530] :hammer: working toward #367 --- src/json.hpp | 14 +++----------- 1 file changed, 3 insertions(+), 11 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 02f0e7b5..9a201153 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -12532,6 +12532,7 @@ scan_number_done: if (strict) { + get_token(); expect(lexer::token_type::end_of_input); } @@ -12593,7 +12594,6 @@ scan_number_done: // closing } -> we are done if (last_token == lexer::token_type::end_object) { - get_token(); if (keep and callback and not callback(--depth, parse_event_t::object_end, result)) { result = basic_json(value_t::discarded); @@ -12635,6 +12635,7 @@ scan_number_done: } // comma -> next value + get_token(); if (last_token == lexer::token_type::value_separator) { get_token(); @@ -12643,7 +12644,6 @@ scan_number_done: // closing } expect(lexer::token_type::end_object); - get_token(); break; } @@ -12671,7 +12671,6 @@ scan_number_done: // closing ] -> we are done if (last_token == lexer::token_type::end_array) { - get_token(); if (callback and not callback(--depth, parse_event_t::array_end, result)) { result = basic_json(value_t::discarded); @@ -12690,6 +12689,7 @@ scan_number_done: } // comma -> next value + get_token(); if (last_token == lexer::token_type::value_separator) { get_token(); @@ -12698,7 +12698,6 @@ scan_number_done: // closing ] expect(lexer::token_type::end_array); - get_token(); break; } @@ -12713,14 +12712,12 @@ scan_number_done: case lexer::token_type::literal_null: { result.m_type = value_t::null; - get_token(); break; } case lexer::token_type::value_string: { result = basic_json(m_lexer.get_string()); - get_token(); break; } @@ -12728,7 +12725,6 @@ scan_number_done: { result.m_type = value_t::boolean; result.m_value = true; - get_token(); break; } @@ -12736,7 +12732,6 @@ scan_number_done: { result.m_type = value_t::boolean; result.m_value = false; - get_token(); break; } @@ -12744,7 +12739,6 @@ scan_number_done: { result.m_type = value_t::number_unsigned; result.m_value = m_lexer.get_number_unsigned(); - get_token(); break; } @@ -12752,7 +12746,6 @@ scan_number_done: { result.m_type = value_t::number_integer; result.m_value = m_lexer.get_number_integer(); - get_token(); break; } @@ -12767,7 +12760,6 @@ scan_number_done: JSON_THROW(out_of_range::create(406, "number overflow parsing '" + m_lexer.get_token_string() + "'")); } - get_token(); break; } From 9e507dfad59214483a20ffe34fde28b9b66ce6bb Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 21 May 2017 16:36:51 +0200 Subject: [PATCH 202/530] :white_check_mark: tests and fix for #367 operator>> now works as expected. --- src/json.hpp | 17 +++--- test/src/unit-regression.cpp | 112 +++++++++++++++++++++++++++++++++++ 2 files changed, 121 insertions(+), 8 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 9a201153..fad9d87a 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -8826,8 +8826,10 @@ class basic_json { // clear stream flags is.clear(); - // set stream after last processed char - is.seekg(start_position + static_cast(processed_chars - 1)); + // We initially read a lot of characters into the buffer, and we + // may not have processed all of them. Therefore, we need to + // "rewind" the stream after the last processed char. + is.seekg(start_position + static_cast(processed_chars)); } int get_character() override @@ -8840,20 +8842,19 @@ class basic_json // store number of bytes in the buffer fill_size = static_cast(is.gcount()); + // the buffer is ready + buffer_pos = 0; + // remember that filling did not yield new input if (fill_size == 0) { eof = true; + return std::char_traits::eof(); } - - // the buffer is ready - buffer_pos = 0; } ++processed_chars; - return eof - ? std::char_traits::eof() - : buffer[buffer_pos++] & 0xFF; + return buffer[buffer_pos++] & 0xFF;; } std::string read(size_t offset, size_t length) override diff --git a/test/src/unit-regression.cpp b/test/src/unit-regression.cpp index 1ff8f987..db281ac4 100644 --- a/test/src/unit-regression.cpp +++ b/test/src/unit-regression.cpp @@ -599,6 +599,118 @@ TEST_CASE("regression tests") "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected end of input"); } + SECTION("issue #367 - behavior of operator>> should more closely resemble that of built-in overloads") + { + SECTION("(empty)") + { + std::stringstream ss; + json j; + CHECK_THROWS_AS(ss >> j, json::parse_error); + CHECK_THROWS_WITH(ss >> j, + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected end of input"); + } + + SECTION("(whitespace)") + { + std::stringstream ss; + ss << " "; + json j; + CHECK_THROWS_AS(ss >> j, json::parse_error); + CHECK_THROWS_WITH(ss >> j, + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected end of input"); + } + + SECTION("one value") + { + std::stringstream ss; + ss << "111"; + json j; + CHECK_NOTHROW(ss >> j); + CHECK(j == 111); + + CHECK_THROWS_AS(ss >> j, json::parse_error); + CHECK_THROWS_WITH(ss >> j, + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected end of input"); + } + + SECTION("one value + whitespace") + { + std::stringstream ss; + ss << "222 \t\n"; + json j; + CHECK_NOTHROW(ss >> j); + CHECK(j == 222); + + CHECK_THROWS_AS(ss >> j, json::parse_error); + CHECK_THROWS_WITH(ss >> j, + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected end of input"); + } + + SECTION("whitespace + one value") + { + std::stringstream ss; + ss << "\n\t 333"; + json j; + CHECK_NOTHROW(ss >> j); + CHECK(j == 333); + + CHECK_THROWS_AS(ss >> j, json::parse_error); + CHECK_THROWS_WITH(ss >> j, + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected end of input"); + } + + SECTION("three values") + { + std::stringstream ss; + ss << " 111 \n222\n \n 333"; + json j; + CHECK_NOTHROW(ss >> j); + CHECK(j == 111); + CHECK_NOTHROW(ss >> j); + CHECK(j == 222); + CHECK_NOTHROW(ss >> j); + CHECK(j == 333); + + CHECK_THROWS_AS(ss >> j, json::parse_error); + CHECK_THROWS_WITH(ss >> j, + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected end of input"); + } + + SECTION("literals without whitespace") + { + std::stringstream ss; + ss << "truefalsenull\"\""; + json j; + CHECK_NOTHROW(ss >> j); + CHECK(j == true); + CHECK_NOTHROW(ss >> j); + CHECK(j == false); + CHECK_NOTHROW(ss >> j); + CHECK(j == nullptr); + CHECK_NOTHROW(ss >> j); + CHECK(j == ""); + + CHECK_THROWS_AS(ss >> j, json::parse_error); + CHECK_THROWS_WITH(ss >> j, + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected end of input"); + } + + SECTION("example from #529") + { + std::stringstream ss; + ss << "{\n \"one\" : 1,\n \"two\" : 2\n}\n{\n \"three\" : 3\n}"; + json j; + CHECK_NOTHROW(ss >> j); + CHECK(j == json({{"one", 1}, {"two", 2}})); + CHECK_NOTHROW(ss >> j); + CHECK(j == json({{"three", 3}})); + + CHECK_THROWS_AS(ss >> j, json::parse_error); + CHECK_THROWS_WITH(ss >> j, + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected end of input"); + } + } + SECTION("issue #389 - Integer-overflow (OSS-Fuzz issue 267)") { // original test case From c7bd01edf216da4a52daf07c6c25300e34e1ec34 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Mon, 22 May 2017 17:06:46 +0200 Subject: [PATCH 203/530] :hammer: trying something #367 --- src/json.hpp | 2 ++ 1 file changed, 2 insertions(+) diff --git a/src/json.hpp b/src/json.hpp index fad9d87a..486b9c6c 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -8830,6 +8830,8 @@ class basic_json // may not have processed all of them. Therefore, we need to // "rewind" the stream after the last processed char. is.seekg(start_position + static_cast(processed_chars)); + // clear stream flags + is.clear(); } int get_character() override From 9a576fe1d9e148c626f0616c46b826295ba1cb64 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Mon, 22 May 2017 22:49:39 +0200 Subject: [PATCH 204/530] :white_check_mark: added test for #367 --- test/src/unit-regression.cpp | 39 ++++++++++++++++++++++++++++++++++++ 1 file changed, 39 insertions(+) diff --git a/test/src/unit-regression.cpp b/test/src/unit-regression.cpp index db281ac4..a5e79ab6 100644 --- a/test/src/unit-regression.cpp +++ b/test/src/unit-regression.cpp @@ -34,6 +34,7 @@ using nlohmann::json; #include #include +#include TEST_CASE("regression tests") { @@ -709,6 +710,44 @@ TEST_CASE("regression tests") CHECK_THROWS_WITH(ss >> j, "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected end of input"); } + + SECTION("second example from #529") + { + std::string str = "{\n\"one\" : 1,\n\"two\" : 2\n}\n{\n\"three\" : 3\n}"; + + { + std::ofstream file("test.json"); + file << str; + } + + std::ifstream stream("test.json", std::ifstream::in); + json val; + + size_t i = 0; + while (stream.peek() != EOF) + { + CAPTURE(i); + CHECK_NOTHROW(stream >> val); + + CHECK(i < 2); + + if (i == 0) + { + CHECK(val == json({{"one", 1}, {"two", 2}})); + CHECK(stream.tellg() == 28); + } + + if (i == 1) + { + CHECK(val == json({{"three", 3}})); + CHECK(stream.tellg() == 44); + } + + ++i; + } + + std::remove("test.json"); + } } SECTION("issue #389 - Integer-overflow (OSS-Fuzz issue 267)") From 52adf3fd5b0b941d4a55e982ae803d968296e5fa Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Mon, 22 May 2017 22:59:57 +0200 Subject: [PATCH 205/530] :hammer: fixed compilation --- test/src/unit-regression.cpp | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/test/src/unit-regression.cpp b/test/src/unit-regression.cpp index a5e79ab6..655e6b19 100644 --- a/test/src/unit-regression.cpp +++ b/test/src/unit-regression.cpp @@ -734,13 +734,13 @@ TEST_CASE("regression tests") if (i == 0) { CHECK(val == json({{"one", 1}, {"two", 2}})); - CHECK(stream.tellg() == 28); + CHECK(static_cast(stream.tellg()) == 28); } if (i == 1) { CHECK(val == json({{"three", 3}})); - CHECK(stream.tellg() == 44); + CHECK(static_cast(stream.tellg()) == 44); } ++i; From ed62129f8e2d2388e4c05d8de9d34ac197d09a2c Mon Sep 17 00:00:00 2001 From: HenryLee Date: Sat, 27 May 2017 13:44:14 +1000 Subject: [PATCH 206/530] Override n + iterator operator in the iterator --- src/json.hpp | 15 +++++++++++++-- test/src/unit-iterators2.cpp | 20 ++++++++++++++++++++ 2 files changed, 33 insertions(+), 2 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 486b9c6c..902d80f6 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -8474,18 +8474,29 @@ class basic_json @brief add to iterator @pre The iterator is initialized; i.e. `m_object != nullptr`. */ - iter_impl operator+(difference_type i) + iter_impl operator+(difference_type i) const { auto result = *this; result += i; return result; } + /*! + @brief addition of distance and iterator + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + friend iter_impl operator+(difference_type i, const iter_impl& it) + { + auto result = it; + result += i; + return result; + } + /*! @brief subtract from iterator @pre The iterator is initialized; i.e. `m_object != nullptr`. */ - iter_impl operator-(difference_type i) + iter_impl operator-(difference_type i) const { auto result = *this; result -= i; diff --git a/test/src/unit-iterators2.cpp b/test/src/unit-iterators2.cpp index e20417a1..d9f43e2f 100644 --- a/test/src/unit-iterators2.cpp +++ b/test/src/unit-iterators2.cpp @@ -269,6 +269,16 @@ TEST_CASE("iterators 2") CHECK_THROWS_AS(it + 1, json::invalid_iterator); CHECK_THROWS_WITH(it + 1, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } + { + auto it = j_object.begin(); + CHECK_THROWS_AS(1 + it, json::invalid_iterator); + CHECK_THROWS_WITH(1 + it, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); + } + { + auto it = j_object.cbegin(); + CHECK_THROWS_AS(1 + it, json::invalid_iterator); + CHECK_THROWS_WITH(1 + it, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); + } { auto it = j_object.begin(); CHECK_THROWS_AS(it -= 1, json::invalid_iterator); @@ -688,6 +698,16 @@ TEST_CASE("iterators 2") CHECK_THROWS_AS(it + 1, json::invalid_iterator); CHECK_THROWS_WITH(it + 1, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } + { + auto it = j_object.rbegin(); + CHECK_THROWS_AS(1 + it, json::invalid_iterator); + CHECK_THROWS_WITH(1 + it, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); + } + { + auto it = j_object.crbegin(); + CHECK_THROWS_AS(1 + it, json::invalid_iterator); + CHECK_THROWS_WITH(1 + it, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); + } { auto it = j_object.rbegin(); CHECK_THROWS_AS(it -= 1, json::invalid_iterator); From adc73d6d5f3354da5b95283279e06e615e848738 Mon Sep 17 00:00:00 2001 From: HenryLee Date: Mon, 29 May 2017 18:52:53 +1000 Subject: [PATCH 207/530] Fix the arithmetic mistakes in reverse iterator --- src/json.hpp | 6 +++--- test/src/unit-iterators2.cpp | 28 ++++++++++++++-------------- 2 files changed, 17 insertions(+), 17 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 902d80f6..fb2740ab 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -8672,7 +8672,7 @@ class basic_json json_reverse_iterator operator+(difference_type i) const { auto result = *this; - result += i; + result -= i; return result; } @@ -8680,14 +8680,14 @@ class basic_json json_reverse_iterator operator-(difference_type i) const { auto result = *this; - result -= i; + result += i; return result; } /// return difference difference_type operator-(const json_reverse_iterator& other) const { - return this->base() - other.base(); + return other.base() - this->base(); } /// access to successor diff --git a/test/src/unit-iterators2.cpp b/test/src/unit-iterators2.cpp index d9f43e2f..3b767211 100644 --- a/test/src/unit-iterators2.cpp +++ b/test/src/unit-iterators2.cpp @@ -747,7 +747,7 @@ TEST_CASE("iterators 2") it += 3; CHECK((j_array.rbegin() + 3) == it); CHECK((it - 3) == j_array.rbegin()); - CHECK((j_array.rbegin() - it) == 3); + CHECK((it - j_array.rbegin()) == 3); CHECK(*it == json(3)); it -= 2; CHECK(*it == json(5)); @@ -757,7 +757,7 @@ TEST_CASE("iterators 2") it += 3; CHECK((j_array.crbegin() + 3) == it); CHECK((it - 3) == j_array.crbegin()); - CHECK((j_array.crbegin() - it) == 3); + CHECK((it - j_array.crbegin()) == 3); CHECK(*it == json(3)); it -= 2; CHECK(*it == json(5)); @@ -769,9 +769,9 @@ TEST_CASE("iterators 2") { auto it = j_null.rbegin(); it += 3; - CHECK((j_null.rbegin() + 3) == it); - CHECK((it - 3) == j_null.rbegin()); - CHECK((j_null.rbegin() - it) == 3); + CHECK((j_null.rbegin() - 3) == it); + CHECK((it + 3) == j_null.rbegin()); + CHECK((it - j_null.rbegin()) == 3); CHECK(it != j_null.rend()); it -= 3; CHECK(it == j_null.rend()); @@ -779,9 +779,9 @@ TEST_CASE("iterators 2") { auto it = j_null.crbegin(); it += 3; - CHECK((j_null.crbegin() + 3) == it); - CHECK((it - 3) == j_null.crbegin()); - CHECK((j_null.crbegin() - it) == 3); + CHECK((j_null.crbegin() - 3) == it); + CHECK((it + 3) == j_null.crbegin()); + CHECK((it - j_null.crbegin()) == 3); CHECK(it != j_null.crend()); it -= 3; CHECK(it == j_null.crend()); @@ -793,9 +793,9 @@ TEST_CASE("iterators 2") { auto it = j_value.rbegin(); it += 3; - CHECK((j_value.rbegin() + 3) == it); - CHECK((it - 3) == j_value.rbegin()); - CHECK((j_value.rbegin() - it) == 3); + CHECK((j_value.rbegin() - 3) == it); + CHECK((it + 3) == j_value.rbegin()); + CHECK((it - j_value.rbegin()) == 3); CHECK(it != j_value.rend()); it -= 3; CHECK(*it == json(42)); @@ -803,9 +803,9 @@ TEST_CASE("iterators 2") { auto it = j_value.crbegin(); it += 3; - CHECK((j_value.crbegin() + 3) == it); - CHECK((it - 3) == j_value.crbegin()); - CHECK((j_value.crbegin() - it) == 3); + CHECK((j_value.crbegin() - 3) == it); + CHECK((it + 3) == j_value.crbegin()); + CHECK((it - j_value.crbegin()) == 3); CHECK(it != j_value.crend()); it -= 3; CHECK(*it == json(42)); From a3bf0131c412d660ade37c08204909d979ca689e Mon Sep 17 00:00:00 2001 From: HenryLee Date: Mon, 29 May 2017 19:43:45 +1000 Subject: [PATCH 208/530] Fix some more reverse iterator arithmetic test codes --- test/src/unit-iterators2.cpp | 16 ++++++++-------- 1 file changed, 8 insertions(+), 8 deletions(-) diff --git a/test/src/unit-iterators2.cpp b/test/src/unit-iterators2.cpp index 3b767211..2fcca263 100644 --- a/test/src/unit-iterators2.cpp +++ b/test/src/unit-iterators2.cpp @@ -769,8 +769,8 @@ TEST_CASE("iterators 2") { auto it = j_null.rbegin(); it += 3; - CHECK((j_null.rbegin() - 3) == it); - CHECK((it + 3) == j_null.rbegin()); + CHECK((j_null.rbegin() + 3) == it); + CHECK((it - 3) == j_null.rbegin()); CHECK((it - j_null.rbegin()) == 3); CHECK(it != j_null.rend()); it -= 3; @@ -779,8 +779,8 @@ TEST_CASE("iterators 2") { auto it = j_null.crbegin(); it += 3; - CHECK((j_null.crbegin() - 3) == it); - CHECK((it + 3) == j_null.crbegin()); + CHECK((j_null.crbegin() + 3) == it); + CHECK((it - 3) == j_null.crbegin()); CHECK((it - j_null.crbegin()) == 3); CHECK(it != j_null.crend()); it -= 3; @@ -793,8 +793,8 @@ TEST_CASE("iterators 2") { auto it = j_value.rbegin(); it += 3; - CHECK((j_value.rbegin() - 3) == it); - CHECK((it + 3) == j_value.rbegin()); + CHECK((j_value.rbegin() + 3) == it); + CHECK((it - 3) == j_value.rbegin()); CHECK((it - j_value.rbegin()) == 3); CHECK(it != j_value.rend()); it -= 3; @@ -803,8 +803,8 @@ TEST_CASE("iterators 2") { auto it = j_value.crbegin(); it += 3; - CHECK((j_value.crbegin() - 3) == it); - CHECK((it + 3) == j_value.crbegin()); + CHECK((j_value.crbegin() + 3) == it); + CHECK((it - 3) == j_value.crbegin()); CHECK((it - j_value.crbegin()) == 3); CHECK(it != j_value.crend()); it -= 3; From 3aef1a582e929dfae7c8ca4da2dc03439eb6f26e Mon Sep 17 00:00:00 2001 From: HenryLee Date: Mon, 29 May 2017 19:45:34 +1000 Subject: [PATCH 209/530] Change the definition of the operator override of reverse iterator to using the result of the base class directly --- src/json.hpp | 14 +++++--------- 1 file changed, 5 insertions(+), 9 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index fb2740ab..86fe8f86 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -8626,7 +8626,7 @@ class basic_json using reference = typename Base::reference; /// create reverse iterator from iterator - json_reverse_iterator(const typename base_iterator::iterator_type& it) noexcept + json_reverse_iterator(const typename base_iterator::iterator_type& it) noexcept : base_iterator(it) {} @@ -8671,29 +8671,25 @@ class basic_json /// add to iterator json_reverse_iterator operator+(difference_type i) const { - auto result = *this; - result -= i; - return result; + return json_reverse_iterator(base_iterator::operator+(i)); } /// subtract from iterator json_reverse_iterator operator-(difference_type i) const { - auto result = *this; - result += i; - return result; + return json_reverse_iterator(base_iterator::operator-(i)); } /// return difference difference_type operator-(const json_reverse_iterator& other) const { - return other.base() - this->base(); + return base_iterator(*this) - base_iterator(other); } /// access to successor reference operator[](difference_type n) const { - return *(this->operator+(n)); + return base_iterator::operator[](n); } /// return the key of an object iterator From e42db27d6d90d5ab470ad392befa595277e0dd97 Mon Sep 17 00:00:00 2001 From: HenryLee Date: Mon, 29 May 2017 20:02:58 +1000 Subject: [PATCH 210/530] Fix the indentation from the latest commit --- src/json.hpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/json.hpp b/src/json.hpp index 86fe8f86..13b62ded 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -8626,7 +8626,7 @@ class basic_json using reference = typename Base::reference; /// create reverse iterator from iterator - json_reverse_iterator(const typename base_iterator::iterator_type& it) noexcept + json_reverse_iterator(const typename base_iterator::iterator_type& it) noexcept : base_iterator(it) {} From 5b245dae06f0c61418a11d596ed28916c9647eca Mon Sep 17 00:00:00 2001 From: HenryLee Date: Mon, 29 May 2017 22:19:10 +1000 Subject: [PATCH 211/530] Comment out the exception message check in operator[] of reverse iterator --- test/src/unit-iterators2.cpp | 16 ++++++++-------- 1 file changed, 8 insertions(+), 8 deletions(-) diff --git a/test/src/unit-iterators2.cpp b/test/src/unit-iterators2.cpp index 2fcca263..a27a924b 100644 --- a/test/src/unit-iterators2.cpp +++ b/test/src/unit-iterators2.cpp @@ -821,15 +821,15 @@ TEST_CASE("iterators 2") auto it = j_object.rbegin(); CHECK_THROWS_AS(it[0], json::invalid_iterator); CHECK_THROWS_AS(it[1], json::invalid_iterator); - CHECK_THROWS_WITH(it[0], "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); - CHECK_THROWS_WITH(it[1], "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); + //CHECK_THROWS_WITH(it[0], "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); + //CHECK_THROWS_WITH(it[1], "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } { auto it = j_object.crbegin(); CHECK_THROWS_AS(it[0], json::invalid_iterator); CHECK_THROWS_AS(it[1], json::invalid_iterator); - CHECK_THROWS_WITH(it[0], "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); - CHECK_THROWS_WITH(it[1], "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); + //CHECK_THROWS_WITH(it[0], "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); + //CHECK_THROWS_WITH(it[1], "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } } @@ -861,15 +861,15 @@ TEST_CASE("iterators 2") auto it = j_null.rbegin(); CHECK_THROWS_AS(it[0], json::invalid_iterator); CHECK_THROWS_AS(it[1], json::invalid_iterator); - CHECK_THROWS_WITH(it[0], "[json.exception.invalid_iterator.214] cannot get value"); - CHECK_THROWS_WITH(it[1], "[json.exception.invalid_iterator.214] cannot get value"); + //CHECK_THROWS_WITH(it[0], "[json.exception.invalid_iterator.214] cannot get value"); + //CHECK_THROWS_WITH(it[1], "[json.exception.invalid_iterator.214] cannot get value"); } { auto it = j_null.crbegin(); CHECK_THROWS_AS(it[0], json::invalid_iterator); CHECK_THROWS_AS(it[1], json::invalid_iterator); - CHECK_THROWS_WITH(it[0], "[json.exception.invalid_iterator.214] cannot get value"); - CHECK_THROWS_WITH(it[1], "[json.exception.invalid_iterator.214] cannot get value"); + //CHECK_THROWS_WITH(it[0], "[json.exception.invalid_iterator.214] cannot get value"); + //CHECK_THROWS_WITH(it[1], "[json.exception.invalid_iterator.214] cannot get value"); } } From 0c2ed00ebe1b1c8f289c4131cdc915979e8c5095 Mon Sep 17 00:00:00 2001 From: HenryLee Date: Mon, 29 May 2017 22:39:27 +1000 Subject: [PATCH 212/530] Add missing test cases for n + it arithmetic in iterators --- test/src/unit-iterators2.cpp | 12 ++++++++++++ 1 file changed, 12 insertions(+) diff --git a/test/src/unit-iterators2.cpp b/test/src/unit-iterators2.cpp index a27a924b..2be349b3 100644 --- a/test/src/unit-iterators2.cpp +++ b/test/src/unit-iterators2.cpp @@ -317,6 +317,7 @@ TEST_CASE("iterators 2") auto it = j_array.begin(); it += 3; CHECK((j_array.begin() + 3) == it); + CHECK(json::iterator(3 + j_array.begin()) == it); CHECK((it - 3) == j_array.begin()); CHECK((it - j_array.begin()) == 3); CHECK(*it == json(4)); @@ -327,6 +328,7 @@ TEST_CASE("iterators 2") auto it = j_array.cbegin(); it += 3; CHECK((j_array.cbegin() + 3) == it); + CHECK(json::const_iterator(3 + j_array.cbegin()) == it); CHECK((it - 3) == j_array.cbegin()); CHECK((it - j_array.cbegin()) == 3); CHECK(*it == json(4)); @@ -341,6 +343,7 @@ TEST_CASE("iterators 2") auto it = j_null.begin(); it += 3; CHECK((j_null.begin() + 3) == it); + CHECK(json::iterator(3 + j_null.begin()) == it); CHECK((it - 3) == j_null.begin()); CHECK((it - j_null.begin()) == 3); CHECK(it != j_null.end()); @@ -351,6 +354,7 @@ TEST_CASE("iterators 2") auto it = j_null.cbegin(); it += 3; CHECK((j_null.cbegin() + 3) == it); + CHECK(json::const_iterator(3 + j_null.cbegin()) == it); CHECK((it - 3) == j_null.cbegin()); CHECK((it - j_null.cbegin()) == 3); CHECK(it != j_null.cend()); @@ -365,6 +369,7 @@ TEST_CASE("iterators 2") auto it = j_value.begin(); it += 3; CHECK((j_value.begin() + 3) == it); + CHECK(json::iterator(3 + j_value.begin()) == it); CHECK((it - 3) == j_value.begin()); CHECK((it - j_value.begin()) == 3); CHECK(it != j_value.end()); @@ -375,6 +380,7 @@ TEST_CASE("iterators 2") auto it = j_value.cbegin(); it += 3; CHECK((j_value.cbegin() + 3) == it); + CHECK(json::const_iterator(3 + j_value.cbegin()) == it); CHECK((it - 3) == j_value.cbegin()); CHECK((it - j_value.cbegin()) == 3); CHECK(it != j_value.cend()); @@ -746,6 +752,7 @@ TEST_CASE("iterators 2") auto it = j_array.rbegin(); it += 3; CHECK((j_array.rbegin() + 3) == it); + CHECK(json::reverse_iterator(3 + j_array.rbegin()) == it); CHECK((it - 3) == j_array.rbegin()); CHECK((it - j_array.rbegin()) == 3); CHECK(*it == json(3)); @@ -756,6 +763,7 @@ TEST_CASE("iterators 2") auto it = j_array.crbegin(); it += 3; CHECK((j_array.crbegin() + 3) == it); + CHECK(json::const_reverse_iterator(3 + j_array.crbegin()) == it); CHECK((it - 3) == j_array.crbegin()); CHECK((it - j_array.crbegin()) == 3); CHECK(*it == json(3)); @@ -770,6 +778,7 @@ TEST_CASE("iterators 2") auto it = j_null.rbegin(); it += 3; CHECK((j_null.rbegin() + 3) == it); + CHECK(json::reverse_iterator(3 + j_null.rbegin()) == it); CHECK((it - 3) == j_null.rbegin()); CHECK((it - j_null.rbegin()) == 3); CHECK(it != j_null.rend()); @@ -780,6 +789,7 @@ TEST_CASE("iterators 2") auto it = j_null.crbegin(); it += 3; CHECK((j_null.crbegin() + 3) == it); + CHECK(json::const_reverse_iterator(3 + j_null.crbegin()) == it); CHECK((it - 3) == j_null.crbegin()); CHECK((it - j_null.crbegin()) == 3); CHECK(it != j_null.crend()); @@ -794,6 +804,7 @@ TEST_CASE("iterators 2") auto it = j_value.rbegin(); it += 3; CHECK((j_value.rbegin() + 3) == it); + CHECK(json::reverse_iterator(3 + j_value.rbegin()) == it); CHECK((it - 3) == j_value.rbegin()); CHECK((it - j_value.rbegin()) == 3); CHECK(it != j_value.rend()); @@ -804,6 +815,7 @@ TEST_CASE("iterators 2") auto it = j_value.crbegin(); it += 3; CHECK((j_value.crbegin() + 3) == it); + CHECK(json::const_reverse_iterator(3 + j_value.crbegin()) == it); CHECK((it - 3) == j_value.crbegin()); CHECK((it - j_value.crbegin()) == 3); CHECK(it != j_value.crend()); From daea213b44135f63409136bbd20cfae2f63b1aa6 Mon Sep 17 00:00:00 2001 From: HenryLee Date: Mon, 29 May 2017 23:56:21 +1000 Subject: [PATCH 213/530] Use static cast in the implementation of the operators of reverse iterator --- src/json.hpp | 17 +++++++---------- 1 file changed, 7 insertions(+), 10 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 13b62ded..5c7074f4 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -8638,46 +8638,43 @@ class basic_json /// post-increment (it++) json_reverse_iterator operator++(int) { - return base_iterator::operator++(1); + return static_cast(base_iterator::operator++(1)); } /// pre-increment (++it) json_reverse_iterator& operator++() { - base_iterator::operator++(); - return *this; + return static_cast(base_iterator::operator++()); } /// post-decrement (it--) json_reverse_iterator operator--(int) { - return base_iterator::operator--(1); + return static_cast(base_iterator::operator--(1)); } /// pre-decrement (--it) json_reverse_iterator& operator--() { - base_iterator::operator--(); - return *this; + return static_cast(base_iterator::operator--()); } /// add to iterator json_reverse_iterator& operator+=(difference_type i) { - base_iterator::operator+=(i); - return *this; + return static_cast(base_iterator::operator+=(i)); } /// add to iterator json_reverse_iterator operator+(difference_type i) const { - return json_reverse_iterator(base_iterator::operator+(i)); + return static_cast(base_iterator::operator+(i)); } /// subtract from iterator json_reverse_iterator operator-(difference_type i) const { - return json_reverse_iterator(base_iterator::operator-(i)); + return static_cast(base_iterator::operator-(i)); } /// return difference From c4ab8f827e45cd6e6f1e80eb77ab850fa228e433 Mon Sep 17 00:00:00 2001 From: HenryLee Date: Tue, 30 May 2017 19:44:55 +1000 Subject: [PATCH 214/530] Change iterator category to random access --- src/json.hpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/json.hpp b/src/json.hpp index 5c7074f4..51221207 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -8015,7 +8015,7 @@ class basic_json typename basic_json::const_reference, typename basic_json::reference>::type; /// the category of the iterator - using iterator_category = std::bidirectional_iterator_tag; + using iterator_category = std::random_access_iterator_tag; /// default constructor iter_impl() = default; From ce39330ff8529265a995d1aa155259508259ff18 Mon Sep 17 00:00:00 2001 From: HenryLee Date: Tue, 30 May 2017 20:23:25 +1000 Subject: [PATCH 215/530] Add converting constructors for iterator --- src/json.hpp | 64 ++++++++++++++++++++++++++++++++-------------------- 1 file changed, 40 insertions(+), 24 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 486b9c6c..c66b45f0 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -8053,42 +8053,58 @@ class basic_json } } - /* - Use operator `const_iterator` instead of `const_iterator(const iterator& - other) noexcept` to avoid two class definitions for @ref iterator and - @ref const_iterator. - - This function is only called if this class is an @ref iterator. If this - class is a @ref const_iterator this function is not called. + /*! + @note The conventional copy constructor is not defined. It is replaced + by either of the following two converting constructors. + They support copy from iterator to iterator, + copy from const iterator to const iterator, + and conversion from iterator to const iterator. + However conversion from const iterator to iterator is not defined. */ - operator const_iterator() const - { - const_iterator ret; - - if (m_object) - { - ret.m_object = m_object; - ret.m_it = m_it; - } - - return ret; - } /*! - @brief copy constructor - @param[in] other iterator to copy from + @brief converting constructor + @param[in] other non-const iterator to copy from @note It is not checked whether @a other is initialized. */ - iter_impl(const iter_impl& other) noexcept + iter_impl(const iter_impl& other) noexcept + : m_object(other.m_object), m_it(other.m_it) + {} + + /*! + @brief converting constructor + @param[in] other const iterator to copy from + @note It is not checked whether @a other is initialized. + */ + iter_impl(const iter_impl& other) noexcept : m_object(other.m_object), m_it(other.m_it) {} /*! @brief copy assignment - @param[in,out] other iterator to copy from + @param[in,out] other non-const iterator to copy from + @return const/non-const iterator @note It is not checked whether @a other is initialized. */ - iter_impl& operator=(iter_impl other) noexcept( + iter_impl& operator=(iter_impl other) noexcept( + std::is_nothrow_move_constructible::value and + std::is_nothrow_move_assignable::value and + std::is_nothrow_move_constructible::value and + std::is_nothrow_move_assignable::value + ) + { + std::swap(m_object, other.m_object); + std::swap(m_it, other.m_it); + return *this; + } + + /*! + @brief copy assignment + @param[in,out] other const iterator to copy from + @return const iterator + @note It is not checked whether @a other is initialized. + */ + iter_impl& operator=(iter_impl other) noexcept( std::is_nothrow_move_constructible::value and std::is_nothrow_move_assignable::value and std::is_nothrow_move_constructible::value and From 790e7ca9e9b075f9156f86045e26313c5d7825b1 Mon Sep 17 00:00:00 2001 From: HenryLee Date: Tue, 30 May 2017 20:27:05 +1000 Subject: [PATCH 216/530] Add struct keyword in front of internal_iterator --- src/json.hpp | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index c66b45f0..88123d58 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -8089,8 +8089,8 @@ class basic_json iter_impl& operator=(iter_impl other) noexcept( std::is_nothrow_move_constructible::value and std::is_nothrow_move_assignable::value and - std::is_nothrow_move_constructible::value and - std::is_nothrow_move_assignable::value + std::is_nothrow_move_constructible::value and + std::is_nothrow_move_assignable::value ) { std::swap(m_object, other.m_object); @@ -8107,8 +8107,8 @@ class basic_json iter_impl& operator=(iter_impl other) noexcept( std::is_nothrow_move_constructible::value and std::is_nothrow_move_assignable::value and - std::is_nothrow_move_constructible::value and - std::is_nothrow_move_assignable::value + std::is_nothrow_move_constructible::value and + std::is_nothrow_move_assignable::value ) { std::swap(m_object, other.m_object); @@ -8601,7 +8601,7 @@ class basic_json /// associated JSON instance pointer m_object = nullptr; /// the actual iterator of the associated instance - internal_iterator m_it = internal_iterator(); + struct internal_iterator m_it = internal_iterator(); }; /*! From 881cd3f42024394906a337372816ce6de0d2f241 Mon Sep 17 00:00:00 2001 From: HenryLee Date: Tue, 30 May 2017 20:57:22 +1000 Subject: [PATCH 217/530] Remove the iter_impl copy constructor and copy assignment --- src/json.hpp | 40 +++++++--------------------------------- 1 file changed, 7 insertions(+), 33 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 88123d58..588cf81b 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -8054,11 +8054,12 @@ class basic_json } /*! - @note The conventional copy constructor is not defined. It is replaced - by either of the following two converting constructors. - They support copy from iterator to iterator, - copy from const iterator to const iterator, - and conversion from iterator to const iterator. + @note The conventional copy constructor and copy assignment are + implicitly defined. + Combined with the following converting constructor and assigment, + they support: copy from iterator to iterator, + copy from const iterator to const iterator, + and conversion from iterator to const iterator. However conversion from const iterator to iterator is not defined. */ @@ -8072,16 +8073,7 @@ class basic_json {} /*! - @brief converting constructor - @param[in] other const iterator to copy from - @note It is not checked whether @a other is initialized. - */ - iter_impl(const iter_impl& other) noexcept - : m_object(other.m_object), m_it(other.m_it) - {} - - /*! - @brief copy assignment + @brief converting assignment @param[in,out] other non-const iterator to copy from @return const/non-const iterator @note It is not checked whether @a other is initialized. @@ -8098,24 +8090,6 @@ class basic_json return *this; } - /*! - @brief copy assignment - @param[in,out] other const iterator to copy from - @return const iterator - @note It is not checked whether @a other is initialized. - */ - iter_impl& operator=(iter_impl other) noexcept( - std::is_nothrow_move_constructible::value and - std::is_nothrow_move_assignable::value and - std::is_nothrow_move_constructible::value and - std::is_nothrow_move_assignable::value - ) - { - std::swap(m_object, other.m_object); - std::swap(m_it, other.m_it); - return *this; - } - private: /*! @brief set the iterator to the first value From 0a51fb22eda45ae2aee7bec4b70cbaa450b216ad Mon Sep 17 00:00:00 2001 From: HenryLee Date: Tue, 30 May 2017 23:37:24 +1000 Subject: [PATCH 218/530] Fix indentation --- src/json.hpp | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 588cf81b..e92580c2 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -8056,9 +8056,9 @@ class basic_json /*! @note The conventional copy constructor and copy assignment are implicitly defined. - Combined with the following converting constructor and assigment, - they support: copy from iterator to iterator, - copy from const iterator to const iterator, + Combined with the following converting constructor and assigment, + they support: copy from iterator to iterator, + copy from const iterator to const iterator, and conversion from iterator to const iterator. However conversion from const iterator to iterator is not defined. */ @@ -8075,7 +8075,7 @@ class basic_json /*! @brief converting assignment @param[in,out] other non-const iterator to copy from - @return const/non-const iterator + @return const/non-const iterator @note It is not checked whether @a other is initialized. */ iter_impl& operator=(iter_impl other) noexcept( From c09a4cbbd7e175062db101898d2b4bd2b26128f0 Mon Sep 17 00:00:00 2001 From: HenryLee Date: Wed, 31 May 2017 00:31:16 +1000 Subject: [PATCH 219/530] Add test cases for iterator to const iterator conversion --- test/src/unit-iterators1.cpp | 52 ++++++++++++++++++++++++++++++++++++ 1 file changed, 52 insertions(+) diff --git a/test/src/unit-iterators1.cpp b/test/src/unit-iterators1.cpp index 6605076a..348c7249 100644 --- a/test/src/unit-iterators1.cpp +++ b/test/src/unit-iterators1.cpp @@ -1511,4 +1511,56 @@ TEST_CASE("iterators 1") } } } + + SECTION("conversion from iterator to const iterator") + { + SECTION("boolean") + { + json j = true; + json::const_iterator it = j.begin(); + CHECK(it == j.cbegin()); + } + SECTION("string") + { + json j = "hello world"; + json::const_iterator it = j.begin(); + CHECK(it == j.cbegin()); + } + SECTION("array") + { + json j = {1, 2, 3}; + json::const_iterator it = j.begin(); + CHECK(it == j.cbegin()); + } + SECTION("object") + { + json j = {{"A", 1}, {"B", 2}, {"C", 3}}; + json::const_iterator it = j.begin(); + CHECK(it == j.cbegin()); + } + SECTION("number (integer)") + { + json j = 23; + json::const_iterator it = j.begin(); + CHECK(it == j.cbegin()); + } + SECTION("number (unsigned)") + { + json j = 23u; + json::const_iterator it = j.begin(); + CHECK(it == j.cbegin()); + } + SECTION("number (float)") + { + json j = 23.42; + json::const_iterator it = j.begin(); + CHECK(it == j.cbegin()); + } + SECTION("null") + { + json j = nullptr; + json::const_iterator it = j.begin(); + CHECK(it == j.cbegin()); + } + } } From 2d5f0c05499e7a0c0f94e82b8c35b03b30a1f94d Mon Sep 17 00:00:00 2001 From: HenryLee Date: Wed, 31 May 2017 00:50:14 +1000 Subject: [PATCH 220/530] Redefine the converting assignment in iterator --- src/json.hpp | 11 +++-------- 1 file changed, 3 insertions(+), 8 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index e92580c2..1bd8addc 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -8078,15 +8078,10 @@ class basic_json @return const/non-const iterator @note It is not checked whether @a other is initialized. */ - iter_impl& operator=(iter_impl other) noexcept( - std::is_nothrow_move_constructible::value and - std::is_nothrow_move_assignable::value and - std::is_nothrow_move_constructible::value and - std::is_nothrow_move_assignable::value - ) + iter_impl& operator=(const iter_impl& other) noexcept { - std::swap(m_object, other.m_object); - std::swap(m_it, other.m_it); + m_object = other.m_object; + m_it = other.m_it; return *this; } From f2e164303953d468eec74a238d9539f3f5a6ff4e Mon Sep 17 00:00:00 2001 From: HenryLee Date: Wed, 31 May 2017 00:50:40 +1000 Subject: [PATCH 221/530] Add test cases for iterator to const iterator assignment --- test/src/unit-iterators1.cpp | 16 ++++++++++++++++ 1 file changed, 16 insertions(+) diff --git a/test/src/unit-iterators1.cpp b/test/src/unit-iterators1.cpp index 348c7249..2dc9aae8 100644 --- a/test/src/unit-iterators1.cpp +++ b/test/src/unit-iterators1.cpp @@ -1519,48 +1519,64 @@ TEST_CASE("iterators 1") json j = true; json::const_iterator it = j.begin(); CHECK(it == j.cbegin()); + it = j.begin(); + CHECK(it == j.cbegin()); } SECTION("string") { json j = "hello world"; json::const_iterator it = j.begin(); CHECK(it == j.cbegin()); + it = j.begin(); + CHECK(it == j.cbegin()); } SECTION("array") { json j = {1, 2, 3}; json::const_iterator it = j.begin(); CHECK(it == j.cbegin()); + it = j.begin(); + CHECK(it == j.cbegin()); } SECTION("object") { json j = {{"A", 1}, {"B", 2}, {"C", 3}}; json::const_iterator it = j.begin(); CHECK(it == j.cbegin()); + it = j.begin(); + CHECK(it == j.cbegin()); } SECTION("number (integer)") { json j = 23; json::const_iterator it = j.begin(); CHECK(it == j.cbegin()); + it = j.begin(); + CHECK(it == j.cbegin()); } SECTION("number (unsigned)") { json j = 23u; json::const_iterator it = j.begin(); CHECK(it == j.cbegin()); + it = j.begin(); + CHECK(it == j.cbegin()); } SECTION("number (float)") { json j = 23.42; json::const_iterator it = j.begin(); CHECK(it == j.cbegin()); + it = j.begin(); + CHECK(it == j.cbegin()); } SECTION("null") { json j = nullptr; json::const_iterator it = j.begin(); CHECK(it == j.cbegin()); + it = j.begin(); + CHECK(it == j.cbegin()); } } } From c043ba6978002bd062c70eae0842c46938d4155b Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Thu, 1 Jun 2017 07:32:39 +0200 Subject: [PATCH 222/530] :fire: removed failing test #529 --- test/src/unit-regression.cpp | 2 ++ 1 file changed, 2 insertions(+) diff --git a/test/src/unit-regression.cpp b/test/src/unit-regression.cpp index 655e6b19..19fa686e 100644 --- a/test/src/unit-regression.cpp +++ b/test/src/unit-regression.cpp @@ -711,6 +711,7 @@ TEST_CASE("regression tests") "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected end of input"); } + /* SECTION("second example from #529") { std::string str = "{\n\"one\" : 1,\n\"two\" : 2\n}\n{\n\"three\" : 3\n}"; @@ -748,6 +749,7 @@ TEST_CASE("regression tests") std::remove("test.json"); } + */ } SECTION("issue #389 - Integer-overflow (OSS-Fuzz issue 267)") From d19c5ced4bec1cd81fbb02e8a9381fb6cc90f286 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Fri, 2 Jun 2017 12:38:32 +0200 Subject: [PATCH 223/530] :bug: skipping BOM for iterators #602 I totally forgot about byte order marks in this scenario. --- src/json.hpp | 10 ++++++++-- test/src/unit-regression.cpp | 6 ++++++ test/src/unit-unicode.cpp | 17 +++++++++++++---- 3 files changed, 27 insertions(+), 6 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 486b9c6c..9c27fb07 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -8814,7 +8814,7 @@ class basic_json // store number of bytes in the buffer fill_size = static_cast(is.gcount()); - // skip byte-order mark + // skip byte order mark if (fill_size >= 3 and buffer[0] == '\xEF' and buffer[1] == '\xBB' and buffer[2] == '\xBF') { buffer_pos += 3; @@ -8911,7 +8911,13 @@ class basic_json public: input_buffer_adapter(const char* b, size_t l) : input_adapter(), cursor(b), limit(b + l), start(b) - {} + { + // skip byte order mark + if (l >= 3 and b[0] == '\xEF' and b[1] == '\xBB' and b[2] == '\xBF') + { + cursor += 3; + } + } // delete because of pointer members input_buffer_adapter(const input_buffer_adapter&) = delete; diff --git a/test/src/unit-regression.cpp b/test/src/unit-regression.cpp index 19fa686e..ec8ba6ad 100644 --- a/test/src/unit-regression.cpp +++ b/test/src/unit-regression.cpp @@ -1169,4 +1169,10 @@ TEST_CASE("regression tests") std::vector vec = {'"', '\\', '"', 'X', '"', '"'}; CHECK_THROWS_AS(json::parse(vec), json::parse_error); } + + SECTION("issue #602 - BOM not skipped when using json:parse(iterator)") + { + std::string i = "\xef\xbb\xbf{\n \"foo\": true\n}"; + CHECK_NOTHROW(json::parse(i.begin(), i.end())); + } } diff --git a/test/src/unit-unicode.cpp b/test/src/unit-unicode.cpp index 17cd9aac..e55a7573 100644 --- a/test/src/unit-unicode.cpp +++ b/test/src/unit-unicode.cpp @@ -1012,10 +1012,19 @@ TEST_CASE("Unicode", "[hide]") SECTION("ignore byte-order-mark") { - // read a file with a UTF-8 BOM - std::ifstream f("test/data/json_nlohmann_tests/bom.json"); - json j; - CHECK_NOTHROW(f >> j); + SECTION("in a stream") + { + // read a file with a UTF-8 BOM + std::ifstream f("test/data/json_nlohmann_tests/bom.json"); + json j; + CHECK_NOTHROW(f >> j); + } + + SECTION("with an iterator") + { + std::string i = "\xef\xbb\xbf{\n \"foo\": true\n}"; + CHECK_NOTHROW(json::parse(i.begin(), i.end())); + } } SECTION("error for incomplete/wrong BOM") From 1a9d76679a3f41a038a89742fba29d81ee24ce82 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 4 Jun 2017 18:40:32 +0200 Subject: [PATCH 224/530] :bug: fixed the issue with GCC7 #590 --- src/json.hpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/json.hpp b/src/json.hpp index 9c27fb07..4efe51f7 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -6381,7 +6381,7 @@ class basic_json { case value_t::array: { - return *lhs.m_value.array < *rhs.m_value.array; + return (*lhs.m_value.array) < (*rhs.m_value.array); } case value_t::object: { From c98169d0d500e2a5f209c58b9e26f6d53787b782 Mon Sep 17 00:00:00 2001 From: HenryLee Date: Mon, 5 Jun 2017 21:59:17 +1000 Subject: [PATCH 225/530] Revert "Change iterator category to random access" This reverts commit c4ab8f827e45cd6e6f1e80eb77ab850fa228e433. --- src/json.hpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/json.hpp b/src/json.hpp index 35b57a78..3fb89ea2 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -8015,7 +8015,7 @@ class basic_json typename basic_json::const_reference, typename basic_json::reference>::type; /// the category of the iterator - using iterator_category = std::random_access_iterator_tag; + using iterator_category = std::bidirectional_iterator_tag; /// default constructor iter_impl() = default; From 85de93ba93a001a7d709c8ec9e0a9ae7fbd9598a Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Th=C3=A9o=20DELRIEU?= Date: Tue, 6 Jun 2017 13:44:35 +0200 Subject: [PATCH 226/530] add std::pair support --- src/json.hpp | 36 +++++++++++++++++++++++++++++++--- test/src/unit-constructor1.cpp | 26 ++++++++++++++++++++++++ 2 files changed, 59 insertions(+), 3 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 4efe51f7..87e4e546 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -285,7 +285,7 @@ class invalid_iterator : public exception Exceptions have ids 3xx. -name / id | example massage | description +name / id | example message | description ----------------------------- | --------------- | ------------------------- json.exception.type_error.301 | cannot create object from initializer list | To create an object from an initializer list, the initializer list must consist only of a list of pairs whose first element is a string. When this constraint is violated, an array is created instead. json.exception.type_error.302 | type must be object, but is array | During implicit or explicit value conversion, the JSON type must be compatible to the target type. For instance, a JSON string can only be converted into string types, but not into numbers or boolean types. @@ -324,7 +324,7 @@ class type_error : public exception Exceptions have ids 4xx. -name / id | example massage | description +name / id | example message | description ------------------------------- | --------------- | ------------------------- json.exception.out_of_range.401 | array index 3 is out of range | The provided array index @a i is larger than @a size-1. json.exception.out_of_range.402 | array index '-' (3) is out of range | The special array index `-` in a JSON Pointer never describes a valid element of the array, but the index past the end. That is, it can only be used to add elements at this position, but not to read it. @@ -355,9 +355,10 @@ class out_of_range : public exception Exceptions have ids 5xx. -name / id | example massage | description +name / id | example message | description ------------------------------ | --------------- | ------------------------- json.exception.other_error.501 | unsuccessful: {"op":"test","path":"/baz", "value":"bar"} | A JSON Patch operation 'test' failed. The unsuccessful operation is also printed. +json.exception.other_error.502 | invalid object size for conversion | Some conversions to user-defined types impose constraints on the object size (e.g. std::pair) @since version 3.0.0 */ @@ -865,6 +866,14 @@ void to_json(BasicJsonType& j, T (&arr)[N]) external_constructor::construct(j, arr); } +template ::value, int> = 0> +void to_json(BasicJsonType& j, std::pair const& p) +{ + j[p.first] = p.second; +} + /////////////// // from_json // /////////////// @@ -1086,6 +1095,27 @@ void from_json(const BasicJsonType& j, ArithmeticType& val) } } +template ::value, int> = 0> +void from_json(const BasicJsonType& j, std::pair& p) +{ + if (not j.is_object()) + { + JSON_THROW(type_error::create(302, "type must be object, but is " + j.type_name())); + } + + auto const inner_object = j.template get_ptr(); + auto const size = inner_object->size(); + if (size != 1) + { + JSON_THROW(other_error::create(502, "conversion to std::pair requires the object to have exactly one field, but it has " + std::to_string(size))); + } + auto const& obj = *inner_object->begin(); + // cannot use *inner_object, need to convert both members + p = std::make_pair(obj.first, obj.second.template get()); +} + struct to_json_fn { private: diff --git a/test/src/unit-constructor1.cpp b/test/src/unit-constructor1.cpp index 18c032e0..f16e15c1 100644 --- a/test/src/unit-constructor1.cpp +++ b/test/src/unit-constructor1.cpp @@ -156,6 +156,20 @@ TEST_CASE("constructors") CHECK(j == j_reference); } + SECTION("std::pair") + { + std::pair p{"first", "second"}; + json j(p); + + CHECK((j.get() == p)); + + std::pair p2{"first", 1}; + // use char const* + json j2(std::make_pair("first", 1)); + + CHECK((j2.get() == p2)); + } + SECTION("std::map") { std::map o {{"a", json(1)}, {"b", json(1u)}, {"c", json(2.2)}, {"d", json(false)}, {"e", json("string")}, {"f", json()}}; @@ -164,6 +178,7 @@ TEST_CASE("constructors") CHECK(j == j_reference); } + SECTION("std::multimap") { std::multimap o {{"a", json(1)}, {"b", json(1u)}, {"c", json(2.2)}, {"d", json(false)}, {"e", json("string")}, {"f", json()}}; @@ -954,6 +969,17 @@ TEST_CASE("constructors") "[json.exception.type_error.301] cannot create object from initializer list"); } + SECTION("std::pair with error") + { + json j{{"too", "much"}, {"string", "fields"}}; + CHECK_THROWS_AS((j.get>()), json::other_error); + CHECK_THROWS_WITH((j.get>()), + "[json.exception.other_error.502] conversion " + "to std::pair requires the object to have " + "exactly one field, but it has 2"); + } + + SECTION("empty array") { json j = json::array(); From 92ef19696a18187112512b8361b822d817d01e95 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 6 Jun 2017 20:48:11 +0200 Subject: [PATCH 227/530] :pencil2: cleanup after #598 --- README.md | 1 + test/src/unit-iterators1.cpp | 100 +++++++++++++++++------------------ 2 files changed, 51 insertions(+), 50 deletions(-) diff --git a/README.md b/README.md index 5d84c3f2..af4f8cba 100644 --- a/README.md +++ b/README.md @@ -840,6 +840,7 @@ I deeply appreciate the help of the following people. - [tinloaf](https://github.com/tinloaf) made sure all pushed warnings are properly popped. - [Fytch](https://github.com/Fytch) found a bug in the documentation. - [Jay Sistar](https://github.com/Type1J) implemented a Meson build description +- [Henry Lee](https://github.com/HenryRLee) fixed a warning in ICC Thanks a lot for helping out! Please [let me know](mailto:mail@nlohmann.me) if I forgot someone. diff --git a/test/src/unit-iterators1.cpp b/test/src/unit-iterators1.cpp index 2dc9aae8..783bf850 100644 --- a/test/src/unit-iterators1.cpp +++ b/test/src/unit-iterators1.cpp @@ -1513,70 +1513,70 @@ TEST_CASE("iterators 1") } SECTION("conversion from iterator to const iterator") - { + { SECTION("boolean") - { + { json j = true; - json::const_iterator it = j.begin(); - CHECK(it == j.cbegin()); - it = j.begin(); - CHECK(it == j.cbegin()); - } + json::const_iterator it = j.begin(); + CHECK(it == j.cbegin()); + it = j.begin(); + CHECK(it == j.cbegin()); + } SECTION("string") - { + { json j = "hello world"; - json::const_iterator it = j.begin(); - CHECK(it == j.cbegin()); - it = j.begin(); - CHECK(it == j.cbegin()); - } + json::const_iterator it = j.begin(); + CHECK(it == j.cbegin()); + it = j.begin(); + CHECK(it == j.cbegin()); + } SECTION("array") - { + { json j = {1, 2, 3}; - json::const_iterator it = j.begin(); - CHECK(it == j.cbegin()); - it = j.begin(); - CHECK(it == j.cbegin()); - } + json::const_iterator it = j.begin(); + CHECK(it == j.cbegin()); + it = j.begin(); + CHECK(it == j.cbegin()); + } SECTION("object") - { + { json j = {{"A", 1}, {"B", 2}, {"C", 3}}; - json::const_iterator it = j.begin(); - CHECK(it == j.cbegin()); - it = j.begin(); - CHECK(it == j.cbegin()); - } + json::const_iterator it = j.begin(); + CHECK(it == j.cbegin()); + it = j.begin(); + CHECK(it == j.cbegin()); + } SECTION("number (integer)") - { + { json j = 23; - json::const_iterator it = j.begin(); - CHECK(it == j.cbegin()); - it = j.begin(); - CHECK(it == j.cbegin()); - } + json::const_iterator it = j.begin(); + CHECK(it == j.cbegin()); + it = j.begin(); + CHECK(it == j.cbegin()); + } SECTION("number (unsigned)") - { + { json j = 23u; - json::const_iterator it = j.begin(); - CHECK(it == j.cbegin()); - it = j.begin(); - CHECK(it == j.cbegin()); - } + json::const_iterator it = j.begin(); + CHECK(it == j.cbegin()); + it = j.begin(); + CHECK(it == j.cbegin()); + } SECTION("number (float)") - { + { json j = 23.42; - json::const_iterator it = j.begin(); - CHECK(it == j.cbegin()); - it = j.begin(); - CHECK(it == j.cbegin()); - } + json::const_iterator it = j.begin(); + CHECK(it == j.cbegin()); + it = j.begin(); + CHECK(it == j.cbegin()); + } SECTION("null") - { + { json j = nullptr; - json::const_iterator it = j.begin(); - CHECK(it == j.cbegin()); - it = j.begin(); - CHECK(it == j.cbegin()); - } - } + json::const_iterator it = j.begin(); + CHECK(it == j.cbegin()); + it = j.begin(); + CHECK(it == j.cbegin()); + } + } } From 0f065edff6d741383ea1639fb6bbb8605e4e88d8 Mon Sep 17 00:00:00 2001 From: HenryLee Date: Wed, 7 Jun 2017 18:01:01 +1000 Subject: [PATCH 228/530] Revert a change in reverse iterator to pass the test cases --- src/json.hpp | 2 +- test/src/unit-iterators2.cpp | 16 ++++++++-------- 2 files changed, 9 insertions(+), 9 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 3fb89ea2..ee525c91 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -8686,7 +8686,7 @@ class basic_json /// access to successor reference operator[](difference_type n) const { - return base_iterator::operator[](n); + return *(this->operator+(n)); } /// return the key of an object iterator diff --git a/test/src/unit-iterators2.cpp b/test/src/unit-iterators2.cpp index 2be349b3..ae8ce687 100644 --- a/test/src/unit-iterators2.cpp +++ b/test/src/unit-iterators2.cpp @@ -833,15 +833,15 @@ TEST_CASE("iterators 2") auto it = j_object.rbegin(); CHECK_THROWS_AS(it[0], json::invalid_iterator); CHECK_THROWS_AS(it[1], json::invalid_iterator); - //CHECK_THROWS_WITH(it[0], "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); - //CHECK_THROWS_WITH(it[1], "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); + CHECK_THROWS_WITH(it[0], "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); + CHECK_THROWS_WITH(it[1], "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } { auto it = j_object.crbegin(); CHECK_THROWS_AS(it[0], json::invalid_iterator); CHECK_THROWS_AS(it[1], json::invalid_iterator); - //CHECK_THROWS_WITH(it[0], "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); - //CHECK_THROWS_WITH(it[1], "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); + CHECK_THROWS_WITH(it[0], "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); + CHECK_THROWS_WITH(it[1], "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } } @@ -873,15 +873,15 @@ TEST_CASE("iterators 2") auto it = j_null.rbegin(); CHECK_THROWS_AS(it[0], json::invalid_iterator); CHECK_THROWS_AS(it[1], json::invalid_iterator); - //CHECK_THROWS_WITH(it[0], "[json.exception.invalid_iterator.214] cannot get value"); - //CHECK_THROWS_WITH(it[1], "[json.exception.invalid_iterator.214] cannot get value"); + CHECK_THROWS_WITH(it[0], "[json.exception.invalid_iterator.214] cannot get value"); + CHECK_THROWS_WITH(it[1], "[json.exception.invalid_iterator.214] cannot get value"); } { auto it = j_null.crbegin(); CHECK_THROWS_AS(it[0], json::invalid_iterator); CHECK_THROWS_AS(it[1], json::invalid_iterator); - //CHECK_THROWS_WITH(it[0], "[json.exception.invalid_iterator.214] cannot get value"); - //CHECK_THROWS_WITH(it[1], "[json.exception.invalid_iterator.214] cannot get value"); + CHECK_THROWS_WITH(it[0], "[json.exception.invalid_iterator.214] cannot get value"); + CHECK_THROWS_WITH(it[1], "[json.exception.invalid_iterator.214] cannot get value"); } } From cea39dfaa892115adb05985221cfa2090f92bab9 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Th=C3=A9o=20DELRIEU?= Date: Tue, 6 Jun 2017 14:18:32 +0200 Subject: [PATCH 229/530] fix #600 Instead of calling CompatibleObjectType iterator-range constructor, first convert json::value_type to CompatibleObjectType::value_type --- src/json.hpp | 28 +++++++++++++++----- test/src/unit-constructor1.cpp | 48 +++++++++++++++++++++++++--------- 2 files changed, 56 insertions(+), 20 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 87e4e546..d5618fb8 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -867,11 +867,11 @@ void to_json(BasicJsonType& j, T (&arr)[N]) } template ::value, int> = 0> + enable_if_t::value, int> = 0> void to_json(BasicJsonType& j, std::pair const& p) { - j[p.first] = p.second; + j[p.first] = p.second; } /////////////// @@ -1046,10 +1046,24 @@ void from_json(const BasicJsonType& j, CompatibleObjectType& obj) auto inner_object = j.template get_ptr(); using std::begin; using std::end; + using value_type = typename CompatibleObjectType::value_type; + std::vector v; + v.reserve(j.size()); + std::transform( + inner_object->begin(), inner_object->end(), std::back_inserter(v), + [](typename BasicJsonType::object_t::value_type const & p) + { + return value_type + { + p.first, + p.second + .template get()}; + }); // we could avoid the assignment, but this might require a for loop, which // might be less efficient than the container constructor for some // containers (would it?) - obj = CompatibleObjectType(begin(*inner_object), end(*inner_object)); + obj = CompatibleObjectType(std::make_move_iterator(begin(v)), + std::make_move_iterator(end(v))); } // overload for arithmetic types, not chosen for basic_json template arguments @@ -1096,8 +1110,8 @@ void from_json(const BasicJsonType& j, ArithmeticType& val) } template ::value, int> = 0> + enable_if_t::value, int> = 0> void from_json(const BasicJsonType& j, std::pair& p) { if (not j.is_object()) @@ -1112,7 +1126,7 @@ void from_json(const BasicJsonType& j, std::pair& p) JSON_THROW(other_error::create(502, "conversion to std::pair requires the object to have exactly one field, but it has " + std::to_string(size))); } auto const& obj = *inner_object->begin(); - // cannot use *inner_object, need to convert both members + // cannot use *inner_object, need to convert both members p = std::make_pair(obj.first, obj.second.template get()); } diff --git a/test/src/unit-constructor1.cpp b/test/src/unit-constructor1.cpp index f16e15c1..db100a69 100644 --- a/test/src/unit-constructor1.cpp +++ b/test/src/unit-constructor1.cpp @@ -158,16 +158,27 @@ TEST_CASE("constructors") SECTION("std::pair") { - std::pair p{"first", "second"}; - json j(p); + std::pair p{"first", "second"}; + json j(p); - CHECK((j.get() == p)); + CHECK((j.get() == p)); - std::pair p2{"first", 1}; - // use char const* - json j2(std::make_pair("first", 1)); + std::pair p2{"first", 1}; + // use char const* + json j2(std::make_pair("first", 1)); - CHECK((j2.get() == p2)); + CHECK((j2.get() == p2)); + } + + SECTION("std::map #600") + { + std::map m; + m["a"] = "b"; + m["c"] = "d"; + m["e"] = "f"; + + json j(m); + CHECK((j.get() == m)); } SECTION("std::map") @@ -971,12 +982,23 @@ TEST_CASE("constructors") SECTION("std::pair with error") { - json j{{"too", "much"}, {"string", "fields"}}; - CHECK_THROWS_AS((j.get>()), json::other_error); - CHECK_THROWS_WITH((j.get>()), - "[json.exception.other_error.502] conversion " - "to std::pair requires the object to have " - "exactly one field, but it has 2"); + SECTION("wrong field number") + { + json j{{"too", "much"}, {"string", "fields"}}; + CHECK_THROWS_AS((j.get>()), json::other_error); + CHECK_THROWS_WITH((j.get>()), + "[json.exception.other_error.502] conversion " + "to std::pair requires the object to have " + "exactly one field, but it has 2"); + } + + SECTION("wrong JSON type") + { + json j(42); + CHECK_THROWS_AS((j.get>()), json::type_error); + CHECK_THROWS_WITH((j.get>()), + "[json.exception.type_error.302] type must be object, but is number"); + } } From 5fa5c17bc6130bf3ab1ffaa2a53828fbd5056c6c Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sat, 10 Jun 2017 18:29:16 +0200 Subject: [PATCH 230/530] :pencil2: cleanup after #595 --- README.md | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/README.md b/README.md index af4f8cba..6a8b9dc8 100644 --- a/README.md +++ b/README.md @@ -839,8 +839,8 @@ I deeply appreciate the help of the following people. - [ftillier](https://github.com/ftillier) fixed a compiler warning. - [tinloaf](https://github.com/tinloaf) made sure all pushed warnings are properly popped. - [Fytch](https://github.com/Fytch) found a bug in the documentation. -- [Jay Sistar](https://github.com/Type1J) implemented a Meson build description -- [Henry Lee](https://github.com/HenryRLee) fixed a warning in ICC +- [Jay Sistar](https://github.com/Type1J) implemented a Meson build description. +- [Henry Lee](https://github.com/HenryRLee) fixed a warning in ICC and improved the iterator implementation. Thanks a lot for helping out! Please [let me know](mailto:mail@nlohmann.me) if I forgot someone. From ac793e957ff7efdd37f198f86feaea02051692e7 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Mon, 12 Jun 2017 18:58:58 +0200 Subject: [PATCH 231/530] :hammer: trying to fix #367 Code from https://stackoverflow.com/a/44503794/266378 which is hopefully working with MSVC. --- src/json.hpp | 3 ++- test/src/unit-regression.cpp | 2 -- 2 files changed, 2 insertions(+), 3 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 2dd55104..9af2a98d 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -8862,7 +8862,8 @@ class basic_json // We initially read a lot of characters into the buffer, and we // may not have processed all of them. Therefore, we need to // "rewind" the stream after the last processed char. - is.seekg(start_position + static_cast(processed_chars)); + is.seekg(start_position); + is.ignore(processed_chars); // clear stream flags is.clear(); } diff --git a/test/src/unit-regression.cpp b/test/src/unit-regression.cpp index ec8ba6ad..f4f1f61d 100644 --- a/test/src/unit-regression.cpp +++ b/test/src/unit-regression.cpp @@ -711,7 +711,6 @@ TEST_CASE("regression tests") "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected end of input"); } - /* SECTION("second example from #529") { std::string str = "{\n\"one\" : 1,\n\"two\" : 2\n}\n{\n\"three\" : 3\n}"; @@ -749,7 +748,6 @@ TEST_CASE("regression tests") std::remove("test.json"); } - */ } SECTION("issue #389 - Integer-overflow (OSS-Fuzz issue 267)") From afb959a083d41b715ecdbc46295d32a3850ec7a2 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Mon, 12 Jun 2017 19:02:08 +0200 Subject: [PATCH 232/530] :hammer: removed nonportable code --- test/src/unit-regression.cpp | 2 -- 1 file changed, 2 deletions(-) diff --git a/test/src/unit-regression.cpp b/test/src/unit-regression.cpp index f4f1f61d..cd2374ff 100644 --- a/test/src/unit-regression.cpp +++ b/test/src/unit-regression.cpp @@ -734,13 +734,11 @@ TEST_CASE("regression tests") if (i == 0) { CHECK(val == json({{"one", 1}, {"two", 2}})); - CHECK(static_cast(stream.tellg()) == 28); } if (i == 1) { CHECK(val == json({{"three", 3}})); - CHECK(static_cast(stream.tellg()) == 44); } ++i; From 88dc7c11fa6ba93b33133e1d5bfc13499cb83eac Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Mon, 12 Jun 2017 19:58:44 +0200 Subject: [PATCH 233/530] :hammer: fixed a warning --- src/json.hpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/json.hpp b/src/json.hpp index 9af2a98d..eb4df22f 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -8863,7 +8863,7 @@ class basic_json // may not have processed all of them. Therefore, we need to // "rewind" the stream after the last processed char. is.seekg(start_position); - is.ignore(processed_chars); + is.ignore(static_cast(processed_chars)); // clear stream flags is.clear(); } From c98364834f9819a2bb25957dbf10456c1d42ce58 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Th=C3=A9o=20DELRIEU?= Date: Tue, 13 Jun 2017 11:11:29 +0200 Subject: [PATCH 234/530] remove std::pair support --- src/json.hpp | 48 +++++----------------------------- test/src/unit-constructor1.cpp | 36 ------------------------- 2 files changed, 6 insertions(+), 78 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index eb4df22f..a4ea2c8f 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -866,14 +866,6 @@ void to_json(BasicJsonType& j, T (&arr)[N]) external_constructor::construct(j, arr); } -template ::value, int> = 0> -void to_json(BasicJsonType& j, std::pair const& p) -{ - j[p.first] = p.second; -} - /////////////// // from_json // /////////////// @@ -1047,23 +1039,16 @@ void from_json(const BasicJsonType& j, CompatibleObjectType& obj) using std::begin; using std::end; using value_type = typename CompatibleObjectType::value_type; - std::vector v; - v.reserve(j.size()); std::transform( - inner_object->begin(), inner_object->end(), std::back_inserter(v), + inner_object->begin(), inner_object->end(), + std::inserter(obj, obj.begin()), [](typename BasicJsonType::object_t::value_type const & p) { - return value_type - { - p.first, - p.second - .template get()}; + return value_type( + p.first, + p.second + .template get()); }); - // we could avoid the assignment, but this might require a for loop, which - // might be less efficient than the container constructor for some - // containers (would it?) - obj = CompatibleObjectType(std::make_move_iterator(begin(v)), - std::make_move_iterator(end(v))); } // overload for arithmetic types, not chosen for basic_json template arguments @@ -1109,27 +1094,6 @@ void from_json(const BasicJsonType& j, ArithmeticType& val) } } -template ::value, int> = 0> -void from_json(const BasicJsonType& j, std::pair& p) -{ - if (not j.is_object()) - { - JSON_THROW(type_error::create(302, "type must be object, but is " + j.type_name())); - } - - auto const inner_object = j.template get_ptr(); - auto const size = inner_object->size(); - if (size != 1) - { - JSON_THROW(other_error::create(502, "conversion to std::pair requires the object to have exactly one field, but it has " + std::to_string(size))); - } - auto const& obj = *inner_object->begin(); - // cannot use *inner_object, need to convert both members - p = std::make_pair(obj.first, obj.second.template get()); -} - struct to_json_fn { private: diff --git a/test/src/unit-constructor1.cpp b/test/src/unit-constructor1.cpp index db100a69..0e56db7c 100644 --- a/test/src/unit-constructor1.cpp +++ b/test/src/unit-constructor1.cpp @@ -156,20 +156,6 @@ TEST_CASE("constructors") CHECK(j == j_reference); } - SECTION("std::pair") - { - std::pair p{"first", "second"}; - json j(p); - - CHECK((j.get() == p)); - - std::pair p2{"first", 1}; - // use char const* - json j2(std::make_pair("first", 1)); - - CHECK((j2.get() == p2)); - } - SECTION("std::map #600") { std::map m; @@ -980,28 +966,6 @@ TEST_CASE("constructors") "[json.exception.type_error.301] cannot create object from initializer list"); } - SECTION("std::pair with error") - { - SECTION("wrong field number") - { - json j{{"too", "much"}, {"string", "fields"}}; - CHECK_THROWS_AS((j.get>()), json::other_error); - CHECK_THROWS_WITH((j.get>()), - "[json.exception.other_error.502] conversion " - "to std::pair requires the object to have " - "exactly one field, but it has 2"); - } - - SECTION("wrong JSON type") - { - json j(42); - CHECK_THROWS_AS((j.get>()), json::type_error); - CHECK_THROWS_WITH((j.get>()), - "[json.exception.type_error.302] type must be object, but is number"); - } - } - - SECTION("empty array") { json j = json::array(); From b27d2adcbe7a3bcf2f994ef2ce0ed4acefda204a Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Fri, 16 Jun 2017 19:23:55 +0200 Subject: [PATCH 235/530] :sparkles: accept functions to check if input is valid JSON #458 --- src/json.hpp | 47 +++++++++++++++++++++++++++++++ test/src/unit-deserialization.cpp | 44 ++++++++++++++++++++++++++--- 2 files changed, 87 insertions(+), 4 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index a4ea2c8f..6bd09cd1 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -7421,6 +7421,13 @@ class basic_json return parse(std::begin(array), std::end(array), cb); } + template + static bool accept(T (&array)[N]) + { + // delegate the call to the iterator-range accept overload + return accept(std::begin(array), std::end(array)); + } + /*! @brief deserialize from string literal @@ -7462,6 +7469,15 @@ class basic_json return parser(input_adapter::create(s), cb).parse(true); } + template::value and + std::is_integral::type>::value and + sizeof(typename std::remove_pointer::type) == 1, int>::type = 0> + static bool accept(const CharT s) + { + return parser(input_adapter::create(s)).accept(true); + } + /*! @brief deserialize from stream @@ -7497,6 +7513,11 @@ class basic_json return parser(input_adapter::create(i), cb).parse(true); } + static bool accept(std::istream& i) + { + return parser(input_adapter::create(i)).accept(true); + } + /*! @copydoc parse(std::istream&, const parser_callback_t) */ @@ -7506,6 +7527,11 @@ class basic_json return parser(input_adapter::create(i), cb).parse(true); } + static bool accept(std::istream&& i) + { + return parser(input_adapter::create(i)).accept(true); + } + /*! @brief deserialize from an iterator range with contiguous storage @@ -7561,6 +7587,15 @@ class basic_json return parser(input_adapter::create(first, last), cb).parse(true); } + template::iterator_category>::value, int>::type = 0> + static bool accept(IteratorType first, IteratorType last) + { + return parser(input_adapter::create(first, last)).accept(true); + } + /*! @brief deserialize from a container with contiguous storage @@ -7618,6 +7653,18 @@ class basic_json return parse(std::begin(c), std::end(c), cb); } + template::value and + std::is_base_of< + std::random_access_iterator_tag, + typename std::iterator_traits()))>::iterator_category>::value + , int>::type = 0> + static bool accept(const ContiguousContainer& c) + { + // delegate the call to the iterator-range accept overload + return accept(std::begin(c), std::end(c)); + } + /*! @brief deserialize from stream @deprecated This stream operator is deprecated and will be removed in a diff --git a/test/src/unit-deserialization.cpp b/test/src/unit-deserialization.cpp index 49a642b1..66fb5e1b 100644 --- a/test/src/unit-deserialization.cpp +++ b/test/src/unit-deserialization.cpp @@ -39,9 +39,11 @@ TEST_CASE("deserialization") { SECTION("stream") { - std::stringstream ss; - ss << "[\"foo\",1,2,3,false,{\"one\":1}]"; - json j = json::parse(ss); + std::stringstream ss1, ss2; + ss1 << "[\"foo\",1,2,3,false,{\"one\":1}]"; + ss2 << "[\"foo\",1,2,3,false,{\"one\":1}]"; + json j = json::parse(ss1); + CHECK(json::accept(ss2)); CHECK(j == json({"foo", 1, 2, 3, false, {{"one", 1}}})); } @@ -49,6 +51,7 @@ TEST_CASE("deserialization") { auto s = "[\"foo\",1,2,3,false,{\"one\":1}]"; json j = json::parse(s); + CHECK(json::accept(s)); CHECK(j == json({"foo", 1, 2, 3, false, {{"one", 1}}})); } @@ -56,6 +59,7 @@ TEST_CASE("deserialization") { json::string_t s = "[\"foo\",1,2,3,false,{\"one\":1}]"; json j = json::parse(s); + CHECK(json::accept(s)); CHECK(j == json({"foo", 1, 2, 3, false, {{"one", 1}}})); } @@ -87,12 +91,14 @@ TEST_CASE("deserialization") { SECTION("stream") { - std::stringstream ss1, ss2; + std::stringstream ss1, ss2, ss3; ss1 << "[\"foo\",1,2,3,false,{\"one\":1}"; ss2 << "[\"foo\",1,2,3,false,{\"one\":1}"; + ss3 << "[\"foo\",1,2,3,false,{\"one\":1}"; CHECK_THROWS_AS(json::parse(ss1), json::parse_error); CHECK_THROWS_WITH(json::parse(ss2), "[json.exception.parse_error.101] parse error at 29: syntax error - unexpected end of input; expected ']'"); + CHECK(not json::accept(ss3)); } SECTION("string") @@ -101,6 +107,7 @@ TEST_CASE("deserialization") CHECK_THROWS_AS(json::parse(s), json::parse_error); CHECK_THROWS_WITH(json::parse(s), "[json.exception.parse_error.101] parse error at 29: syntax error - unexpected end of input; expected ']'"); + CHECK(not json::accept(s)); } SECTION("operator<<") @@ -141,18 +148,21 @@ TEST_CASE("deserialization") { std::vector v = {'t', 'r', 'u', 'e'}; CHECK(json::parse(v) == json(true)); + CHECK(json::accept(v)); } SECTION("from std::array") { std::array v { {'t', 'r', 'u', 'e'} }; CHECK(json::parse(v) == json(true)); + CHECK(json::accept(v)); } SECTION("from array") { uint8_t v[] = {'t', 'r', 'u', 'e'}; CHECK(json::parse(v) == json(true)); + CHECK(json::accept(v)); } SECTION("from chars") @@ -164,6 +174,7 @@ TEST_CASE("deserialization") v[3] = 'e'; v[4] = '\0'; CHECK(json::parse(v) == json(true)); + CHECK(json::accept(v)); delete[] v; } @@ -171,18 +182,21 @@ TEST_CASE("deserialization") { std::string v = {'t', 'r', 'u', 'e'}; CHECK(json::parse(v) == json(true)); + CHECK(json::accept(v)); } SECTION("from std::initializer_list") { std::initializer_list v = {'t', 'r', 'u', 'e'}; CHECK(json::parse(v) == json(true)); + CHECK(json::accept(v)); } SECTION("empty container") { std::vector v; CHECK_THROWS_AS(json::parse(v), json::parse_error); + CHECK(not json::accept(v)); } } @@ -192,42 +206,49 @@ TEST_CASE("deserialization") { std::vector v = {'t', 'r', 'u', 'e'}; CHECK(json::parse(std::begin(v), std::end(v)) == json(true)); + CHECK(json::accept(std::begin(v), std::end(v))); } SECTION("from std::array") { std::array v { {'t', 'r', 'u', 'e'} }; CHECK(json::parse(std::begin(v), std::end(v)) == json(true)); + CHECK(json::accept(std::begin(v), std::end(v))); } SECTION("from array") { uint8_t v[] = {'t', 'r', 'u', 'e'}; CHECK(json::parse(std::begin(v), std::end(v)) == json(true)); + CHECK(json::accept(std::begin(v), std::end(v))); } SECTION("from std::string") { std::string v = {'t', 'r', 'u', 'e'}; CHECK(json::parse(std::begin(v), std::end(v)) == json(true)); + CHECK(json::accept(std::begin(v), std::end(v))); } SECTION("from std::initializer_list") { std::initializer_list v = {'t', 'r', 'u', 'e'}; CHECK(json::parse(std::begin(v), std::end(v)) == json(true)); + CHECK(json::accept(std::begin(v), std::end(v))); } SECTION("from std::valarray") { std::valarray v = {'t', 'r', 'u', 'e'}; CHECK(json::parse(std::begin(v), std::end(v)) == json(true)); + CHECK(json::accept(std::begin(v), std::end(v))); } SECTION("with empty range") { std::vector v; CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); + CHECK(not json::accept(std::begin(v), std::end(v))); } } @@ -238,90 +259,105 @@ TEST_CASE("deserialization") { uint8_t v[] = {'\"', 'a', 'a', 'a', 'a', 'a', 'a', '\\', 'u'}; CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); + CHECK(not json::accept(std::begin(v), std::end(v))); } SECTION("case 2") { uint8_t v[] = {'\"', 'a', 'a', 'a', 'a', 'a', 'a', '\\', 'u', '1'}; CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); + CHECK(not json::accept(std::begin(v), std::end(v))); } SECTION("case 3") { uint8_t v[] = {'\"', 'a', 'a', 'a', 'a', 'a', 'a', '\\', 'u', '1', '1', '1', '1', '1', '1', '1', '1'}; CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); + CHECK(not json::accept(std::begin(v), std::end(v))); } SECTION("case 4") { uint8_t v[] = {'\"', 'a', 'a', 'a', 'a', 'a', 'a', 'u', '1', '1', '1', '1', '1', '1', '1', '1', '\\'}; CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); + CHECK(not json::accept(std::begin(v), std::end(v))); } SECTION("case 5") { uint8_t v[] = {'\"', 0x7F, 0xC1}; CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); + CHECK(not json::accept(std::begin(v), std::end(v))); } SECTION("case 6") { uint8_t v[] = {'\"', 0x7F, 0xDF, 0x7F}; CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); + CHECK(not json::accept(std::begin(v), std::end(v))); } SECTION("case 7") { uint8_t v[] = {'\"', 0x7F, 0xDF, 0xC0}; CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); + CHECK(not json::accept(std::begin(v), std::end(v))); } SECTION("case 8") { uint8_t v[] = {'\"', 0x7F, 0xE0, 0x9F}; CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); + CHECK(not json::accept(std::begin(v), std::end(v))); } SECTION("case 9") { uint8_t v[] = {'\"', 0x7F, 0xEF, 0xC0}; CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); + CHECK(not json::accept(std::begin(v), std::end(v))); } SECTION("case 10") { uint8_t v[] = {'\"', 0x7F, 0xED, 0x7F}; CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); + CHECK(not json::accept(std::begin(v), std::end(v))); } SECTION("case 11") { uint8_t v[] = {'\"', 0x7F, 0xF0, 0x8F}; CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); + CHECK(not json::accept(std::begin(v), std::end(v))); } SECTION("case 12") { uint8_t v[] = {'\"', 0x7F, 0xF0, 0xC0}; CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); + CHECK(not json::accept(std::begin(v), std::end(v))); } SECTION("case 13") { uint8_t v[] = {'\"', 0x7F, 0xF3, 0x7F}; CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); + CHECK(not json::accept(std::begin(v), std::end(v))); } SECTION("case 14") { uint8_t v[] = {'\"', 0x7F, 0xF3, 0xC0}; CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); + CHECK(not json::accept(std::begin(v), std::end(v))); } SECTION("case 15") { uint8_t v[] = {'\"', 0x7F, 0xF4, 0x7F}; CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); + CHECK(not json::accept(std::begin(v), std::end(v))); } } } From e1ca484bc13068857ce65fee90bd84ae2155d208 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sat, 17 Jun 2017 11:20:09 +0200 Subject: [PATCH 236/530] :memo: added note about Conan #566 --- README.md | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/README.md b/README.md index 6a8b9dc8..7e320734 100644 --- a/README.md +++ b/README.md @@ -68,6 +68,8 @@ to the files you want to use JSON objects. That's it. Do not forget to set the n If you are using the [Meson Build System](http://mesonbuild.com), then you can wrap this repo as a subproject. +If you are using [Conan](https://www.conan.io/) to manage your dependencies, merely add `jsonformoderncpp/x.y.z@vthiery/stable` to your `conanfile.py`'s requires, where `x.y.z` is the release version you want to use. Please file issues [here](https://github.com/vthiery/conan-jsonformoderncpp/issues) if you experience problems with the packages. + :warning: [Version 3.0.0](https://github.com/nlohmann/json/wiki/Road-toward-3.0.0) is currently under development. Branch `develop` is used for the ongoing work and is probably **unstable**. Please use the `master` branch for the last stable version 2.1.1. @@ -841,6 +843,8 @@ I deeply appreciate the help of the following people. - [Fytch](https://github.com/Fytch) found a bug in the documentation. - [Jay Sistar](https://github.com/Type1J) implemented a Meson build description. - [Henry Lee](https://github.com/HenryRLee) fixed a warning in ICC and improved the iterator implementation. +- [Vincent Thiery](https://github.com/vthiery) maintains a package for the Conan package manager. + Thanks a lot for helping out! Please [let me know](mailto:mail@nlohmann.me) if I forgot someone. From 889006f0069cebc351baf100784005818c57cd38 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sat, 17 Jun 2017 13:37:04 +0200 Subject: [PATCH 237/530] :white_check_mark: regression test for #600 --- test/src/unit-regression.cpp | 35 +++++++++++++++++++++++++++++++++++ 1 file changed, 35 insertions(+) diff --git a/test/src/unit-regression.cpp b/test/src/unit-regression.cpp index cd2374ff..0489e630 100644 --- a/test/src/unit-regression.cpp +++ b/test/src/unit-regression.cpp @@ -1166,6 +1166,41 @@ TEST_CASE("regression tests") CHECK_THROWS_AS(json::parse(vec), json::parse_error); } + SECTION("issue #600 - how does one convert a map in Json back to std::map?") + { + SECTION("example 1") + { + // create a map + std::map m1 {{"key", 1}}; + + // create and print a JSON from the map + json j = m1; + std::cout << j << std::endl; + + // get the map out of JSON + std::map m2 = j; + + // make sure the roundtrip succeeds + CHECK(m1 == m2); + } + + SECTION("example 2") + { + // create a map + std::map m1 {{"key", "val"}}; + + // create and print a JSON from the map + json j = m1; + std::cout << j << std::endl; + + // get the map out of JSON + std::map m2 = j; + + // make sure the roundtrip succeeds + CHECK(m1 == m2); + } + } + SECTION("issue #602 - BOM not skipped when using json:parse(iterator)") { std::string i = "\xef\xbb\xbf{\n \"foo\": true\n}"; From 1fbb82de895f5eda973150de8452c57c0b7e09ad Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sat, 17 Jun 2017 20:47:53 +0200 Subject: [PATCH 238/530] :hammer: using std::array to buffer input --- src/json.hpp | 45 +++++++++++++++++++----------------- test/src/unit-regression.cpp | 2 -- 2 files changed, 24 insertions(+), 23 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 6bd09cd1..7a1fbc23 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -8761,15 +8761,15 @@ class basic_json // native support /// input adapter for input stream - static std::shared_ptr create(std::istream& i, const size_t buffer_size = 16384) + static std::shared_ptr create(std::istream& i) { - return std::shared_ptr(new cached_input_stream_adapter(i, buffer_size)); + return std::shared_ptr(new cached_input_stream_adapter<16384>(i)); } /// input adapter for input stream - static std::shared_ptr create(std::istream&& i, const size_t buffer_size = 16384) + static std::shared_ptr create(std::istream&& i) { - return std::shared_ptr(new cached_input_stream_adapter(i, buffer_size)); + return std::shared_ptr(new cached_input_stream_adapter<16384>(i)); } /// input adapter for buffer @@ -8841,11 +8841,12 @@ class basic_json using input_adapter_t = std::shared_ptr; /// input adapter for cached stream input + template class cached_input_stream_adapter : public input_adapter { public: - cached_input_stream_adapter(std::istream& i, const size_t buffer_size) - : is(i), start_position(is.tellg()), buffer(buffer_size, '\0') + cached_input_stream_adapter(std::istream& i) + : is(i), start_position(is.tellg()) { // immediately abort if stream is erroneous if (JSON_UNLIKELY(i.fail())) @@ -8853,10 +8854,7 @@ class basic_json JSON_THROW(parse_error::create(111, 0, "bad input stream")); } - // initial fill - is.read(buffer.data(), static_cast(buffer.size())); - // store number of bytes in the buffer - fill_size = static_cast(is.gcount()); + fill_buffer(); // skip byte order mark if (fill_size >= 3 and buffer[0] == '\xEF' and buffer[1] == '\xBB' and buffer[2] == '\xBF') @@ -8884,20 +8882,17 @@ class basic_json // check if refilling is necessary and possible if (buffer_pos == fill_size and not eof) { - // refill - is.read(buffer.data(), static_cast(buffer.size())); - // store number of bytes in the buffer - fill_size = static_cast(is.gcount()); + fill_buffer(); - // the buffer is ready - buffer_pos = 0; - - // remember that filling did not yield new input + // check and remember that filling did not yield new input if (fill_size == 0) { eof = true; return std::char_traits::eof(); } + + // the buffer is ready + buffer_pos = 0; } ++processed_chars; @@ -8910,9 +8905,9 @@ class basic_json std::string result(length, '\0'); // save stream position - auto current_pos = is.tellg(); + const auto current_pos = is.tellg(); // save stream flags - auto flags = is.rdstate(); + const auto flags = is.rdstate(); // clear stream flags is.clear(); @@ -8930,6 +8925,14 @@ class basic_json } private: + void fill_buffer() + { + // fill + is.read(buffer.data(), static_cast(buffer.size())); + // store number of bytes in the buffer + fill_size = static_cast(is.gcount()); + } + /// the associated input stream std::istream& is; @@ -8947,7 +8950,7 @@ class basic_json const std::streampos start_position; /// internal buffer - std::vector buffer; + std::array buffer; }; /// input adapter for buffer input diff --git a/test/src/unit-regression.cpp b/test/src/unit-regression.cpp index 0489e630..4d1231c2 100644 --- a/test/src/unit-regression.cpp +++ b/test/src/unit-regression.cpp @@ -1175,7 +1175,6 @@ TEST_CASE("regression tests") // create and print a JSON from the map json j = m1; - std::cout << j << std::endl; // get the map out of JSON std::map m2 = j; @@ -1191,7 +1190,6 @@ TEST_CASE("regression tests") // create and print a JSON from the map json j = m1; - std::cout << j << std::endl; // get the map out of JSON std::map m2 = j; From 04a1666ef2e16de1af840f8bfb5f8e473271159a Mon Sep 17 00:00:00 2001 From: Steffen Brem Date: Sat, 17 Jun 2017 22:01:18 +0200 Subject: [PATCH 239/530] Add parentheses around std::min so it bypasses the macro introduced by windows.h. --- src/json.hpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/json.hpp b/src/json.hpp index 7a1fbc23..2ed408c8 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -8987,7 +8987,7 @@ class basic_json { // avoid reading too many characters const size_t max_length = static_cast(limit - start); - return std::string(start + offset, std::min(length, max_length - offset)); + return std::string(start + offset, (std::min)(length, max_length - offset)); } private: From b79a1a26787c92a4eb2561a8cf1bb6a2cdf6973b Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 18 Jun 2017 00:00:31 +0200 Subject: [PATCH 240/530] :memo: update after #626 --- README.md | 1 + 1 file changed, 1 insertion(+) diff --git a/README.md b/README.md index 7e320734..513b9db4 100644 --- a/README.md +++ b/README.md @@ -844,6 +844,7 @@ I deeply appreciate the help of the following people. - [Jay Sistar](https://github.com/Type1J) implemented a Meson build description. - [Henry Lee](https://github.com/HenryRLee) fixed a warning in ICC and improved the iterator implementation. - [Vincent Thiery](https://github.com/vthiery) maintains a package for the Conan package manager. +- [Steffen](https://github.com/koemeet) fixed a potential issue with MSVC and `std::min`. Thanks a lot for helping out! Please [let me know](mailto:mail@nlohmann.me) if I forgot someone. From 112a6f4e53bc3417c1fc30d1b3097b0b68556cfb Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 18 Jun 2017 13:03:14 +0200 Subject: [PATCH 241/530] :hammer: replaced lexer's error message by const char* --- src/json.hpp | 23 ++++++++++++----------- test/src/unit-class_parser.cpp | 16 ++++++++-------- 2 files changed, 20 insertions(+), 19 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 2ed408c8..a1c15528 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -8896,7 +8896,8 @@ class basic_json } ++processed_chars; - return buffer[buffer_pos++] & 0xFF;; + assert(buffer_pos < buffer.size()); + return buffer[buffer_pos++] & 0xFF; } std::string read(size_t offset, size_t length) override @@ -8971,7 +8972,7 @@ class basic_json input_buffer_adapter(const input_buffer_adapter&) = delete; input_buffer_adapter& operator=(input_buffer_adapter&) = delete; - int get_character() override + int get_character() noexcept override { if (JSON_LIKELY(cursor < limit)) { @@ -11510,13 +11511,13 @@ class basic_json } else { - error_message = "invalid string: surrogate " + codepoint_to_string(codepoint1) + " must be followed by U+DC00..U+DFFF instead of " + codepoint_to_string(codepoint2); + error_message = "invalid string: surrogate U+DC00..U+DFFF must be followed by U+DC00..U+DFFF"; return token_type::parse_error; } } else { - error_message = "invalid string: surrogate " + codepoint_to_string(codepoint1) + " must be followed by U+DC00..U+DFFF"; + error_message = "invalid string: surrogate U+DC00..U+DFFF must be followed by U+DC00..U+DFFF"; return token_type::parse_error; } } @@ -11524,7 +11525,7 @@ class basic_json { if (JSON_UNLIKELY(0xDC00 <= codepoint1 and codepoint1 <= 0xDFFF)) { - error_message = "invalid string: surrogate " + codepoint_to_string(codepoint1) + " must follow U+D800..U+DBFF"; + error_message = "invalid string: surrogate U+DC00..U+DFFF must follow U+D800..U+DBFF"; return token_type::parse_error; } @@ -11609,7 +11610,7 @@ class basic_json case 0x1e: case 0x1f: { - error_message = "invalid string: control character " + codepoint_to_string(current) + " must be escaped"; + error_message = "invalid string: control character must be escaped"; return token_type::parse_error; } @@ -12450,7 +12451,7 @@ scan_number_done: } /// return syntax error message - const std::string& get_error_message() const noexcept + const char* get_error_message() const noexcept { return error_message; } @@ -12544,7 +12545,7 @@ scan_number_done: size_t yylen = 0; /// a description of occurred lexer errors - std::string error_message = ""; + const char* error_message = ""; // number values number_integer_t value_integer = 0; @@ -12975,7 +12976,7 @@ scan_number_done: std::string error_msg = "syntax error - "; if (last_token == lexer::token_type::parse_error) { - error_msg += m_lexer.get_error_message() + "; last read: '" + m_lexer.get_token_string() + "'"; + error_msg += std::string(m_lexer.get_error_message()) + "; last read: '" + m_lexer.get_token_string() + "'"; } else { @@ -12997,7 +12998,7 @@ scan_number_done: std::string error_msg = "syntax error - "; if (last_token == lexer::token_type::parse_error) { - error_msg += m_lexer.get_error_message() + "; last read '" + m_lexer.get_token_string() + "'"; + error_msg += std::string(m_lexer.get_error_message()) + "; last read '" + m_lexer.get_token_string() + "'"; } else { @@ -14189,7 +14190,7 @@ scan_number_done: case patch_operations::copy: { - const std::string from_path = get_value("copy", "from", true);; + const std::string from_path = get_value("copy", "from", true); const json_pointer from_ptr(from_path); // the "from" location must exist - use at() diff --git a/test/src/unit-class_parser.cpp b/test/src/unit-class_parser.cpp index 5a2ff350..920054d2 100644 --- a/test/src/unit-class_parser.cpp +++ b/test/src/unit-class_parser.cpp @@ -98,18 +98,18 @@ TEST_CASE("parser class") // error: tab in string CHECK_THROWS_AS(parse_string("\"\t\"").parse(), json::parse_error); CHECK_THROWS_WITH(parse_string("\"\t\"").parse(), - "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: control character U+0009 must be escaped; last read '\"'"); + "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: control character must be escaped; last read '\"'"); // error: newline in string CHECK_THROWS_AS(parse_string("\"\n\"").parse(), json::parse_error); CHECK_THROWS_AS(parse_string("\"\r\"").parse(), json::parse_error); CHECK_THROWS_WITH(parse_string("\"\n\"").parse(), - "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: control character U+000A must be escaped; last read '\"'"); + "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: control character must be escaped; last read '\"'"); CHECK_THROWS_WITH(parse_string("\"\r\"").parse(), - "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: control character U+000D must be escaped; last read '\"'"); + "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: control character must be escaped; last read '\"'"); // error: backspace in string CHECK_THROWS_AS(parse_string("\"\b\"").parse(), json::parse_error); CHECK_THROWS_WITH(parse_string("\"\b\"").parse(), - "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: control character U+0008 must be escaped; last read '\"'"); + "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: control character must be escaped; last read '\"'"); // improve code coverage CHECK_THROWS_AS(parse_string("\uFF01").parse(), json::parse_error); CHECK_THROWS_AS(parse_string("[-4:1,]").parse(), json::parse_error); @@ -910,17 +910,17 @@ TEST_CASE("parser class") // missing part of a surrogate pair CHECK_THROWS_AS(json::parse("\"\\uD80C\""), json::parse_error); CHECK_THROWS_WITH(json::parse("\"\\uD80C\""), - "[json.exception.parse_error.101] parse error at 8: syntax error - invalid string: surrogate U+D80C must be followed by U+DC00..U+DFFF; last read '\"\\uD80C\"'"); + "[json.exception.parse_error.101] parse error at 8: syntax error - invalid string: surrogate U+DC00..U+DFFF must be followed by U+DC00..U+DFFF; last read '\"\\uD80C\"'"); // invalid surrogate pair CHECK_THROWS_AS(json::parse("\"\\uD80C\\uD80C\""), json::parse_error); CHECK_THROWS_AS(json::parse("\"\\uD80C\\u0000\""), json::parse_error); CHECK_THROWS_AS(json::parse("\"\\uD80C\\uFFFF\""), json::parse_error); CHECK_THROWS_WITH(json::parse("\"\\uD80C\\uD80C\""), - "[json.exception.parse_error.101] parse error at 13: syntax error - invalid string: surrogate U+D80C must be followed by U+DC00..U+DFFF instead of U+D80C; last read '\"\\uD80C\\uD80C'"); + "[json.exception.parse_error.101] parse error at 13: syntax error - invalid string: surrogate U+DC00..U+DFFF must be followed by U+DC00..U+DFFF; last read '\"\\uD80C\\uD80C'"); CHECK_THROWS_WITH(json::parse("\"\\uD80C\\u0000\""), - "[json.exception.parse_error.101] parse error at 13: syntax error - invalid string: surrogate U+D80C must be followed by U+DC00..U+DFFF instead of U+0000; last read '\"\\uD80C\\u0000'"); + "[json.exception.parse_error.101] parse error at 13: syntax error - invalid string: surrogate U+DC00..U+DFFF must be followed by U+DC00..U+DFFF; last read '\"\\uD80C\\u0000'"); CHECK_THROWS_WITH(json::parse("\"\\uD80C\\uFFFF\""), - "[json.exception.parse_error.101] parse error at 13: syntax error - invalid string: surrogate U+D80C must be followed by U+DC00..U+DFFF instead of U+FFFF; last read '\"\\uD80C\\uFFFF'"); + "[json.exception.parse_error.101] parse error at 13: syntax error - invalid string: surrogate U+DC00..U+DFFF must be followed by U+DC00..U+DFFF; last read '\"\\uD80C\\uFFFF'"); } SECTION("parse errors (accept)") From 6e4910d5c5638bedbc3ff650d1b6b91249a927a3 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Th=C3=A9o=20DELRIEU?= Date: Sat, 17 Jun 2017 14:24:07 +0200 Subject: [PATCH 242/530] add pair/tuple conversions --- src/json.hpp | 69 ++++++++++++++++++++++++++++++++++ test/src/unit-constructor1.cpp | 52 +++++++++++++++++++++++++ 2 files changed, 121 insertions(+) diff --git a/src/json.hpp b/src/json.hpp index a4ea2c8f..ccf83d5e 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -465,6 +465,39 @@ using enable_if_t = typename std::enable_if::type; template using uncvref_t = typename std::remove_cv::type>::type; +// implementation of C++14 index_sequence and affiliates +// source: https://stackoverflow.com/a/32223343 +template +struct index_sequence +{ + using type = index_sequence; + using value_type = std::size_t; + static constexpr std::size_t size() noexcept + { + return sizeof...(Ints); + } +}; + +template +struct merge_and_renumber; + +template +struct merge_and_renumber, index_sequence> + : index_sequence < I1..., (sizeof...(I1) + I2)... > + { }; + +template +struct make_index_sequence + : merge_and_renumber < typename make_index_sequence < N / 2 >::type, + typename make_index_sequence < N - N / 2 >::type > +{ }; + +template<> struct make_index_sequence<0> : index_sequence<> { }; +template<> struct make_index_sequence<1> : index_sequence<0> { }; + +template +using index_sequence_for = make_index_sequence; + /* Implementation of two C++17 constructs: conjunction, negation. This is needed to avoid evaluating all the traits in a condition @@ -866,6 +899,24 @@ void to_json(BasicJsonType& j, T (&arr)[N]) external_constructor::construct(j, arr); } +template +void to_json(BasicJsonType& j, const std::pair& p) +{ + j = {p.first, p.second}; +} + +template +void to_json_tuple_impl(BasicJsonType& j, const Tuple& t, index_sequence) +{ + j = {std::get(t)...}; +} + +template +void to_json(BasicJsonType& j, const std::tuple& t) +{ + to_json_tuple_impl(j, t, index_sequence_for {}); +} + /////////////// // from_json // /////////////// @@ -1094,6 +1145,24 @@ void from_json(const BasicJsonType& j, ArithmeticType& val) } } +template +void from_json(const BasicJsonType& j, std::pair& p) +{ + p = {j.at(0), j.at(1)}; +} + +template +void from_json_tuple_impl(const BasicJsonType& j, Tuple& t, index_sequence) +{ + t = std::make_tuple(j.at(Idx)...); +} + +template +void from_json(const BasicJsonType& j, std::tuple& t) +{ + from_json_tuple_impl(j, t, index_sequence_for {}); +} + struct to_json_fn { private: diff --git a/test/src/unit-constructor1.cpp b/test/src/unit-constructor1.cpp index 0e56db7c..a74e9a63 100644 --- a/test/src/unit-constructor1.cpp +++ b/test/src/unit-constructor1.cpp @@ -239,6 +239,58 @@ TEST_CASE("constructors") CHECK(j == j_reference); } + SECTION("std::pair") + { + std::pair p{1.0, "string"}; + json j(p); + + CHECK(j.type() == json::value_t::array); + REQUIRE(j.size() == 2); + CHECK(j[0] == std::get<0>(p)); + CHECK(j[1] == std::get<1>(p)); + } + + SECTION("std::pair with discarded values") + { + json j{1, 2.0, "string"}; + + const auto p = j.get>(); + CHECK(p.first == j[0]); + CHECK(p.second == j[1]); + } + + SECTION("std::tuple") + { + const auto t = std::make_tuple(1.0, "string", 42, std::vector {0, 1}); + json j(t); + + CHECK(j.type() == json::value_t::array); + REQUIRE(j.size() == 4); + CHECK(j[0] == std::get<0>(t)); + CHECK(j[1] == std::get<1>(t)); + CHECK(j[2] == std::get<2>(t)); + CHECK(j[3][0] == 0); + CHECK(j[3][1] == 1); + } + + SECTION("std::tuple with discarded values") + { + json j{1, 2.0, "string", 42}; + + const auto t = j.get>(); + CHECK(std::get<0>(t) == j[0]); + CHECK(std::get<1>(t) == j[1]); + CHECK(std::get<2>(t) == j[2]); + } + + SECTION("std::pair/tuple failures") + { + json j{1}; + + CHECK_THROWS((j.get>())); + CHECK_THROWS((j.get>())); + } + SECTION("std::forward_list") { std::forward_list a {json(1), json(1u), json(2.2), json(false), json("string"), json()}; From 08d781058c118836c9fc8a50b7d4675558ec08d8 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Th=C3=A9o=20DELRIEU?= Date: Mon, 19 Jun 2017 12:03:38 +0200 Subject: [PATCH 243/530] add from_json support for std::array --- src/json.hpp | 11 ++++++++++- test/src/unit-constructor1.cpp | 8 ++++++-- 2 files changed, 16 insertions(+), 3 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index ccf83d5e..87c8d21f 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -1063,6 +1063,15 @@ auto from_json_array_impl(const BasicJsonType& j, CompatibleArrayType& arr, prio }); } +template +void from_json_array_impl(const BasicJsonType& j, std::array& arr, priority_tag<2>) +{ + for (std::size_t i = 0; i < N; ++i) + { + arr[i] = j.at(i).template get(); + } +} + template::value and std::is_convertible::value and @@ -1074,7 +1083,7 @@ void from_json(const BasicJsonType& j, CompatibleArrayType& arr) JSON_THROW(type_error::create(302, "type must be array, but is " + j.type_name())); } - from_json_array_impl(j, arr, priority_tag<1> {}); + from_json_array_impl(j, arr, priority_tag<2> {}); } template(t) == j[2]); } - SECTION("std::pair/tuple failures") + SECTION("std::pair/tuple/array failures") { json j{1}; CHECK_THROWS((j.get>())); CHECK_THROWS((j.get>())); + CHECK_THROWS((j.get>())); } SECTION("std::forward_list") @@ -299,12 +300,15 @@ TEST_CASE("constructors") CHECK(j == j_reference); } - SECTION("std::array") + SECTION("std::array") { std::array a {{json(1), json(1u), json(2.2), json(false), json("string"), json()}}; json j(a); CHECK(j.type() == json::value_t::array); CHECK(j == j_reference); + + const auto a2 = j.get>(); + CHECK(a2 == a); } SECTION("std::vector") From 70b7455a139b57df884e46da3c7ed778cd762acd Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Mon, 19 Jun 2017 21:50:36 +0200 Subject: [PATCH 244/530] :hammer: minor change --- src/json.hpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/json.hpp b/src/json.hpp index 146b5286..b4dc422d 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -12529,7 +12529,7 @@ scan_number_done: } /// return syntax error message - const char* get_error_message() const noexcept + constexpr const char* get_error_message() const noexcept { return error_message; } From c2db4a118d726cb9df9acc12f6b86cd1aff187b0 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Mon, 19 Jun 2017 21:51:24 +0200 Subject: [PATCH 245/530] :construction_worker: update Travis to use Ubuntu Trusty 14.04 images --- .travis.yml | 1 + 1 file changed, 1 insertion(+) diff --git a/.travis.yml b/.travis.yml index 089524f4..8f19baa2 100644 --- a/.travis.yml +++ b/.travis.yml @@ -7,6 +7,7 @@ language: cpp dist: trusty sudo: required +group: edge ################### From c2044205efee1ae651a9ef8ba9c33d1647575089 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 20 Jun 2017 16:12:14 +0200 Subject: [PATCH 246/530] :memo: updated used compilers --- README.md | 24 ++++++++++++------------ 1 file changed, 12 insertions(+), 12 deletions(-) diff --git a/README.md b/README.md index 513b9db4..7fbcf48a 100644 --- a/README.md +++ b/README.md @@ -735,24 +735,24 @@ The following compilers are currently used in continuous integration at [Travis] | Compiler | Operating System | Version String | |-----------------|------------------------------|----------------| -| GCC 4.9.3 | Ubuntu 14.04.4 LTS | g++-4.9 (Ubuntu 4.9.3-8ubuntu2~14.04) 4.9.3 | -| GCC 5.3.0 | Ubuntu 14.04.4 LTS | g++-5 (Ubuntu 5.3.0-3ubuntu1~14.04) 5.3.0 20151204 | -| GCC 6.1.1 | Ubuntu 14.04.4 LTS | g++-6 (Ubuntu 6.1.1-3ubuntu11~14.04.1) 6.1.1 20160511 | -| Clang 3.6.0 | Ubuntu 14.04.4 LTS | clang version 3.6.0 (tags/RELEASE_360/final) | -| Clang 3.6.1 | Ubuntu 14.04.4 LTS | clang version 3.6.1 (tags/RELEASE_361/final) | -| Clang 3.6.2 | Ubuntu 14.04.4 LTS | clang version 3.6.2 (tags/RELEASE_362/final) | -| Clang 3.7.0 | Ubuntu 14.04.4 LTS | clang version 3.7.0 (tags/RELEASE_370/final) | -| Clang 3.7.1 | Ubuntu 14.04.4 LTS | clang version 3.7.1 (tags/RELEASE_371/final) | -| Clang 3.8.0 | Ubuntu 14.04.4 LTS | clang version 3.8.0 (tags/RELEASE_380/final) | -| Clang 3.8.1 | Ubuntu 14.04.4 LTS | clang version 3.8.1 (tags/RELEASE_381/final) | +| GCC 4.9.4 | Ubuntu 14.04.5 LTS | g++-4.9 (Ubuntu 4.9.4-2ubuntu1~14.04.1) 4.9.4 | +| GCC 5.4.1 | Ubuntu 14.04.5 LTS | g++-5 (Ubuntu 5.4.1-2ubuntu1~14.04) 5.4.1 20160904 | +| GCC 6.3.0 | Ubuntu 14.04.5 LTS | g++-6 (Ubuntu/Linaro 6.3.0-18ubuntu2~14.04) 6.3.0 20170519 | +| Clang 3.6.0 | Ubuntu 14.04.5 LTS | clang version 3.6.0 (tags/RELEASE_360/final) | +| Clang 3.6.1 | Ubuntu 14.04.5 LTS | clang version 3.6.1 (tags/RELEASE_361/final) | +| Clang 3.6.2 | Ubuntu 14.04.5 LTS | clang version 3.6.2 (tags/RELEASE_362/final) | +| Clang 3.7.0 | Ubuntu 14.04.5 LTS | clang version 3.7.0 (tags/RELEASE_370/final) | +| Clang 3.7.1 | Ubuntu 14.04.5 LTS | clang version 3.7.1 (tags/RELEASE_371/final) | +| Clang 3.8.0 | Ubuntu 14.04.5 LTS | clang version 3.8.0 (tags/RELEASE_380/final) | +| Clang 3.8.1 | Ubuntu 14.04.5 LTS | clang version 3.8.1 (tags/RELEASE_381/final) | | Clang Xcode 6.4 | Darwin Kernel Version 14.3.0 (OSX 10.10.3) | Apple LLVM version 6.1.0 (clang-602.0.53) (based on LLVM 3.6.0svn) | | Clang Xcode 7.3 | Darwin Kernel Version 15.0.0 (OSX 10.10.5) | Apple LLVM version 7.3.0 (clang-703.0.29) | | Clang Xcode 8.0 | Darwin Kernel Version 15.6.0 | Apple LLVM version 8.0.0 (clang-800.0.38) | | Clang Xcode 8.1 | Darwin Kernel Version 16.1.0 (macOS 10.12.1) | Apple LLVM version 8.0.0 (clang-800.0.42.1) | | Clang Xcode 8.2 | Darwin Kernel Version 16.1.0 (macOS 10.12.1) | Apple LLVM version 8.0.0 (clang-800.0.42.1) | | Clang Xcode 8.3 | Darwin Kernel Version 16.5.0 (macOS 10.12.4) | Apple LLVM version 8.1.0 (clang-802.0.38) | -| Visual Studio 14 2015 | Windows Server 2012 R2 (x64) | Microsoft (R) Build Engine version 14.0.25123.0 | -| Visual Studio 2017 | Windows Server 2016 | Microsoft (R) Build Engine version 15.1.548.43366 | +| Visual Studio 14 2015 | Windows Server 2012 R2 (x64) | Microsoft (R) Build Engine version 14.0.25420.1 | +| Visual Studio 2017 | Windows Server 2016 | Microsoft (R) Build Engine version 15.1.1012.6693 | ## License From f2cdb3d5948e102b7e7ab2dacb7df48fad9a1bbe Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 20 Jun 2017 20:13:56 +0200 Subject: [PATCH 247/530] :arrow_up: Catch v1.9.5 --- test/Makefile | 2 +- test/thirdparty/catch/catch.hpp | 963 ++++++++++++++++++++------------ 2 files changed, 599 insertions(+), 366 deletions(-) diff --git a/test/Makefile b/test/Makefile index 8f4e215f..f4103ba9 100644 --- a/test/Makefile +++ b/test/Makefile @@ -4,7 +4,7 @@ # additional flags CXXFLAGS += -std=c++11 -Wall -Wextra -pedantic -Weffc++ -Wcast-align -Wcast-qual -Wno-ctor-dtor-privacy -Wdisabled-optimization -Wformat=2 -Winit-self -Wmissing-declarations -Wmissing-include-dirs -Wold-style-cast -Woverloaded-virtual -Wredundant-decls -Wshadow -Wsign-conversion -Wsign-promo -Wstrict-overflow=5 -Wswitch -Wundef -Wno-unused -Wnon-virtual-dtor -Wreorder -Wdeprecated -Wno-float-equal -CPPFLAGS += -I ../src -I . -I thirdparty/catch +CPPFLAGS += -I ../src -I . -I thirdparty/catch -DCATCH_CONFIG_FAST_COMPILE SOURCES = src/unit.cpp \ src/unit-algorithms.cpp \ diff --git a/test/thirdparty/catch/catch.hpp b/test/thirdparty/catch/catch.hpp index 6f9334ba..2c93e370 100644 --- a/test/thirdparty/catch/catch.hpp +++ b/test/thirdparty/catch/catch.hpp @@ -1,6 +1,6 @@ /* - * Catch v1.8.1 - * Generated: 2017-03-01 16:04:19.016511 + * Catch v1.9.5 + * Generated: 2017-06-15 12:03:23.301505 * ---------------------------------------------------------- * This file has been merged from multiple headers. Please don't edit it directly * Copyright (c) 2012 Two Blue Cubes Ltd. All rights reserved. @@ -40,11 +40,7 @@ #elif defined __GNUC__ # pragma GCC diagnostic ignored "-Wvariadic-macros" # pragma GCC diagnostic ignored "-Wunused-variable" - - // For newer version we can use __Pragma to disable the warnings locally -# if __GNUC__ == 4 && __GNUC_MINOR__ >= 4 && __GNUC_MINOR__ <= 7 -# pragma GCC diagnostic ignored "-Wparentheses" -# endif +# pragma GCC diagnostic ignored "-Wparentheses" # pragma GCC diagnostic push # pragma GCC diagnostic ignored "-Wpadded" @@ -124,6 +120,12 @@ # endif # if defined(CATCH_CPP11_OR_GREATER) +# define CATCH_INTERNAL_SUPPRESS_ETD_WARNINGS \ + _Pragma( "clang diagnostic push" ) \ + _Pragma( "clang diagnostic ignored \"-Wexit-time-destructors\"" ) +# define CATCH_INTERNAL_UNSUPPRESS_ETD_WARNINGS \ + _Pragma( "clang diagnostic pop" ) + # define CATCH_INTERNAL_SUPPRESS_PARENTHESES_WARNINGS \ _Pragma( "clang diagnostic push" ) \ _Pragma( "clang diagnostic ignored \"-Wparentheses\"" ) @@ -134,13 +136,19 @@ #endif // __clang__ //////////////////////////////////////////////////////////////////////////////// -// Cygwin -#ifdef __CYGWIN__ +// We know some environments not to support full POSIX signals +#if defined(__CYGWIN__) || defined(__QNX__) # if !defined(CATCH_CONFIG_POSIX_SIGNALS) # define CATCH_INTERNAL_CONFIG_NO_POSIX_SIGNALS # endif +#endif + +//////////////////////////////////////////////////////////////////////////////// +// Cygwin +#ifdef __CYGWIN__ + // Required for some versions of Cygwin to declare gettimeofday // see: http://stackoverflow.com/questions/36901803/gettimeofday-not-declared-in-this-scope-cygwin # define _BSD_SOURCE @@ -169,22 +177,10 @@ // GCC #ifdef __GNUC__ -# if __GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 8) -# define CATCH_GCC_HAS_NEW_PRAGMA -# endif - # if __GNUC__ == 4 && __GNUC_MINOR__ >= 6 && defined(__GXX_EXPERIMENTAL_CXX0X__) # define CATCH_INTERNAL_CONFIG_CPP11_NULLPTR # endif -# if !defined(CATCH_INTERNAL_SUPPRESS_PARENTHESES_WARNINGS) && defined(CATCH_GCC_HAS_NEW_PRAGMA) -# define CATCH_INTERNAL_SUPPRESS_PARENTHESES_WARNINGS \ - _Pragma( "GCC diagnostic push" ) \ - _Pragma( "GCC diagnostic ignored \"-Wparentheses\"" ) -# define CATCH_INTERNAL_UNSUPPRESS_PARENTHESES_WARNINGS \ - _Pragma( "GCC diagnostic pop" ) -# endif - // - otherwise more recent versions define __cplusplus >= 201103L // and will get picked up below @@ -224,7 +220,7 @@ // Use __COUNTER__ if the compiler supports it #if ( defined _MSC_VER && _MSC_VER >= 1300 ) || \ - ( defined __GNUC__ && __GNUC__ >= 4 && __GNUC_MINOR__ >= 3 ) || \ + ( defined __GNUC__ && ( __GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 3 )) ) || \ ( defined __clang__ && __clang_major__ >= 3 ) #define CATCH_INTERNAL_CONFIG_COUNTER @@ -332,6 +328,10 @@ # define CATCH_INTERNAL_SUPPRESS_PARENTHESES_WARNINGS # define CATCH_INTERNAL_UNSUPPRESS_PARENTHESES_WARNINGS #endif +#if !defined(CATCH_INTERNAL_SUPPRESS_ETD_WARNINGS) +# define CATCH_INTERNAL_SUPPRESS_ETD_WARNINGS +# define CATCH_INTERNAL_UNSUPPRESS_ETD_WARNINGS +#endif // noexcept support: #if defined(CATCH_CONFIG_CPP11_NOEXCEPT) && !defined(CATCH_NOEXCEPT) @@ -770,59 +770,76 @@ void registerTestCaseFunction /////////////////////////////////////////////////////////////////////////////// #define INTERNAL_CATCH_TESTCASE2( TestName, ... ) \ static void TestName(); \ - namespace{ Catch::AutoReg INTERNAL_CATCH_UNIQUE_NAME( autoRegistrar )( &TestName, CATCH_INTERNAL_LINEINFO, Catch::NameAndDesc( __VA_ARGS__ ) ); }\ + CATCH_INTERNAL_SUPPRESS_ETD_WARNINGS \ + namespace{ Catch::AutoReg INTERNAL_CATCH_UNIQUE_NAME( autoRegistrar )( &TestName, CATCH_INTERNAL_LINEINFO, Catch::NameAndDesc( __VA_ARGS__ ) ); } \ + CATCH_INTERNAL_UNSUPPRESS_ETD_WARNINGS \ static void TestName() #define INTERNAL_CATCH_TESTCASE( ... ) \ INTERNAL_CATCH_TESTCASE2( INTERNAL_CATCH_UNIQUE_NAME( ____C_A_T_C_H____T_E_S_T____ ), __VA_ARGS__ ) /////////////////////////////////////////////////////////////////////////////// #define INTERNAL_CATCH_METHOD_AS_TEST_CASE( QualifiedMethod, ... ) \ - namespace{ Catch::AutoReg INTERNAL_CATCH_UNIQUE_NAME( autoRegistrar )( &QualifiedMethod, "&" #QualifiedMethod, Catch::NameAndDesc( __VA_ARGS__ ), CATCH_INTERNAL_LINEINFO ); } + CATCH_INTERNAL_SUPPRESS_ETD_WARNINGS \ + namespace{ Catch::AutoReg INTERNAL_CATCH_UNIQUE_NAME( autoRegistrar )( &QualifiedMethod, "&" #QualifiedMethod, Catch::NameAndDesc( __VA_ARGS__ ), CATCH_INTERNAL_LINEINFO ); } \ + CATCH_INTERNAL_UNSUPPRESS_ETD_WARNINGS /////////////////////////////////////////////////////////////////////////////// #define INTERNAL_CATCH_TEST_CASE_METHOD2( TestName, ClassName, ... )\ + CATCH_INTERNAL_SUPPRESS_ETD_WARNINGS \ namespace{ \ struct TestName : ClassName{ \ void test(); \ }; \ Catch::AutoReg INTERNAL_CATCH_UNIQUE_NAME( autoRegistrar ) ( &TestName::test, #ClassName, Catch::NameAndDesc( __VA_ARGS__ ), CATCH_INTERNAL_LINEINFO ); \ } \ + CATCH_INTERNAL_UNSUPPRESS_ETD_WARNINGS \ void TestName::test() #define INTERNAL_CATCH_TEST_CASE_METHOD( ClassName, ... ) \ INTERNAL_CATCH_TEST_CASE_METHOD2( INTERNAL_CATCH_UNIQUE_NAME( ____C_A_T_C_H____T_E_S_T____ ), ClassName, __VA_ARGS__ ) /////////////////////////////////////////////////////////////////////////////// #define INTERNAL_CATCH_REGISTER_TESTCASE( Function, ... ) \ - Catch::AutoReg( Function, CATCH_INTERNAL_LINEINFO, Catch::NameAndDesc( __VA_ARGS__ ) ); + CATCH_INTERNAL_SUPPRESS_ETD_WARNINGS \ + Catch::AutoReg( Function, CATCH_INTERNAL_LINEINFO, Catch::NameAndDesc( __VA_ARGS__ ) ); \ + CATCH_INTERNAL_UNSUPPRESS_ETD_WARNINGS #else /////////////////////////////////////////////////////////////////////////////// #define INTERNAL_CATCH_TESTCASE2( TestName, Name, Desc ) \ static void TestName(); \ + CATCH_INTERNAL_SUPPRESS_ETD_WARNINGS \ namespace{ Catch::AutoReg INTERNAL_CATCH_UNIQUE_NAME( autoRegistrar )( &TestName, CATCH_INTERNAL_LINEINFO, Catch::NameAndDesc( Name, Desc ) ); }\ + CATCH_INTERNAL_UNSUPPRESS_ETD_WARNINGS \ static void TestName() #define INTERNAL_CATCH_TESTCASE( Name, Desc ) \ INTERNAL_CATCH_TESTCASE2( INTERNAL_CATCH_UNIQUE_NAME( ____C_A_T_C_H____T_E_S_T____ ), Name, Desc ) /////////////////////////////////////////////////////////////////////////////// #define INTERNAL_CATCH_METHOD_AS_TEST_CASE( QualifiedMethod, Name, Desc ) \ - namespace{ Catch::AutoReg INTERNAL_CATCH_UNIQUE_NAME( autoRegistrar )( &QualifiedMethod, "&" #QualifiedMethod, Catch::NameAndDesc( Name, Desc ), CATCH_INTERNAL_LINEINFO ); } + CATCH_INTERNAL_SUPPRESS_ETD_WARNINGS \ + namespace{ Catch::AutoReg INTERNAL_CATCH_UNIQUE_NAME( autoRegistrar )( &QualifiedMethod, "&" #QualifiedMethod, Catch::NameAndDesc( Name, Desc ), CATCH_INTERNAL_LINEINFO ); } \ + CATCH_INTERNAL_UNSUPPRESS_ETD_WARNINGS /////////////////////////////////////////////////////////////////////////////// #define INTERNAL_CATCH_TEST_CASE_METHOD2( TestCaseName, ClassName, TestName, Desc )\ + CATCH_INTERNAL_SUPPRESS_ETD_WARNINGS \ namespace{ \ struct TestCaseName : ClassName{ \ void test(); \ }; \ Catch::AutoReg INTERNAL_CATCH_UNIQUE_NAME( autoRegistrar ) ( &TestCaseName::test, #ClassName, Catch::NameAndDesc( TestName, Desc ), CATCH_INTERNAL_LINEINFO ); \ } \ + CATCH_INTERNAL_UNSUPPRESS_ETD_WARNINGS \ void TestCaseName::test() #define INTERNAL_CATCH_TEST_CASE_METHOD( ClassName, TestName, Desc )\ INTERNAL_CATCH_TEST_CASE_METHOD2( INTERNAL_CATCH_UNIQUE_NAME( ____C_A_T_C_H____T_E_S_T____ ), ClassName, TestName, Desc ) /////////////////////////////////////////////////////////////////////////////// #define INTERNAL_CATCH_REGISTER_TESTCASE( Function, Name, Desc ) \ - Catch::AutoReg( Function, CATCH_INTERNAL_LINEINFO, Catch::NameAndDesc( Name, Desc ) ); + CATCH_INTERNAL_SUPPRESS_ETD_WARNINGS \ + Catch::AutoReg( Function, CATCH_INTERNAL_LINEINFO, Catch::NameAndDesc( Name, Desc ) ); \ + CATCH_INTERNAL_UNSUPPRESS_ETD_WARNINGS + #endif // #included from: internal/catch_capture.hpp @@ -910,8 +927,8 @@ namespace Catch { template STATIC_ASSERT_Expression_Too_Complex_Please_Rewrite_As_Binary_Comparison& operator && ( T const& ); template STATIC_ASSERT_Expression_Too_Complex_Please_Rewrite_As_Binary_Comparison& operator || ( T const& ); - private: - DecomposedExpression& operator = (DecomposedExpression const&); + private: + DecomposedExpression& operator = (DecomposedExpression const&); }; struct AssertionInfo @@ -1021,16 +1038,24 @@ namespace Matchers { } protected: + virtual ~MatcherUntypedBase(); virtual std::string describe() const = 0; mutable std::string m_cachedToString; private: MatcherUntypedBase& operator = ( MatcherUntypedBase const& ); }; - template - struct MatcherBase : MatcherUntypedBase { - + template + struct MatcherMethod { virtual bool match( ObjectT const& arg ) const = 0; + }; + template + struct MatcherMethod { + virtual bool match( PtrT* arg ) const = 0; + }; + + template + struct MatcherBase : MatcherUntypedBase, MatcherMethod { MatchAllOf operator && ( MatcherBase const& other ) const; MatchAnyOf operator || ( MatcherBase const& other ) const; @@ -1184,6 +1209,7 @@ namespace Catch { char const* capturedExpression, ResultDisposition::Flags resultDisposition, char const* secondArg = "" ); + ~ResultBuilder(); template ExpressionLhs operator <= ( T const& operand ); @@ -1218,6 +1244,9 @@ namespace Catch { template void captureMatch( ArgT const& arg, MatcherT const& matcher, char const* matcherString ); + void setExceptionGuard(); + void unsetExceptionGuard(); + private: AssertionInfo m_assertionInfo; AssertionResultData m_data; @@ -1225,6 +1254,7 @@ namespace Catch { bool m_shouldDebugBreak; bool m_shouldThrow; + bool m_guardException; }; } // namespace Catch @@ -1541,7 +1571,7 @@ std::string toString( std::nullptr_t ); #ifdef __OBJC__ std::string toString( NSString const * const& nsstring ); - std::string toString( NSString * CATCH_ARC_STRONG const& nsstring ); + std::string toString( NSString * CATCH_ARC_STRONG & nsstring ); std::string toString( NSObject* const& nsObject ); #endif @@ -1549,6 +1579,7 @@ namespace Detail { extern const std::string unprintableString; + #if !defined(CATCH_CONFIG_CPP11_STREAM_INSERTABLE_CHECK) struct BorgType { template BorgType( T const& ); }; @@ -1567,6 +1598,20 @@ namespace Detail { static T const&t; enum { value = sizeof( testStreamable(s << t) ) == sizeof( TrueType ) }; }; +#else + template + class IsStreamInsertable { + template + static auto test(int) + -> decltype( std::declval() << std::declval(), std::true_type() ); + + template + static auto test(...) -> std::false_type; + + public: + static const bool value = decltype(test(0))::value; + }; +#endif #if defined(CATCH_CONFIG_CPP11_IS_ENUM) template -#endif - -namespace Catch { - -#if defined(CATCH_CONFIG_CPP11_TYPE_TRAITS) - - template - using add_lvalue_reference = std::add_lvalue_reference; - - template - using add_const = std::add_const; - -#else - - template - struct add_const { - typedef const T type; - }; - - template - struct add_lvalue_reference { - typedef T& type; - }; - template - struct add_lvalue_reference { - typedef T& type; - }; - // No && overload, because that is C++11, in which case we have - // proper type_traits implementation from the standard library - -#endif - -} - #if defined(CATCH_CONFIG_FAST_COMPILE) /////////////////////////////////////////////////////////////////////////////// // We can speedup compilation significantly by breaking into debugger lower in @@ -2139,6 +2147,33 @@ namespace Catch { // macro in each assertion #define INTERNAL_CATCH_REACT( resultBuilder ) \ resultBuilder.react(); + +/////////////////////////////////////////////////////////////////////////////// +// Another way to speed-up compilation is to omit local try-catch for REQUIRE* +// macros. +// This can potentially cause false negative, if the test code catches +// the exception before it propagates back up to the runner. +#define INTERNAL_CATCH_TEST_NO_TRY( macroName, resultDisposition, expr ) \ + do { \ + Catch::ResultBuilder __catchResult( macroName, CATCH_INTERNAL_LINEINFO, #expr, resultDisposition ); \ + __catchResult.setExceptionGuard(); \ + CATCH_INTERNAL_SUPPRESS_PARENTHESES_WARNINGS \ + ( __catchResult <= expr ).endExpression(); \ + CATCH_INTERNAL_UNSUPPRESS_PARENTHESES_WARNINGS \ + __catchResult.unsetExceptionGuard(); \ + INTERNAL_CATCH_REACT( __catchResult ) \ + } while( Catch::isTrue( false && static_cast( !!(expr) ) ) ) // expr here is never evaluated at runtime but it forces the compiler to give it a look +// The double negation silences MSVC's C4800 warning, the static_cast forces short-circuit evaluation if the type has overloaded &&. + +#define INTERNAL_CHECK_THAT_NO_TRY( macroName, matcher, resultDisposition, arg ) \ + do { \ + Catch::ResultBuilder __catchResult( macroName, CATCH_INTERNAL_LINEINFO, #arg ", " #matcher, resultDisposition ); \ + __catchResult.setExceptionGuard(); \ + __catchResult.captureMatch( arg, matcher, #matcher ); \ + __catchResult.unsetExceptionGuard(); \ + INTERNAL_CATCH_REACT( __catchResult ) \ + } while( Catch::alwaysFalse() ) + #else /////////////////////////////////////////////////////////////////////////////// // In the event of a failure works out if the debugger needs to be invoked @@ -2151,7 +2186,7 @@ namespace Catch { #endif /////////////////////////////////////////////////////////////////////////////// -#define INTERNAL_CATCH_TEST( expr, resultDisposition, macroName ) \ +#define INTERNAL_CATCH_TEST( macroName, resultDisposition, expr ) \ do { \ Catch::ResultBuilder __catchResult( macroName, CATCH_INTERNAL_LINEINFO, #expr, resultDisposition ); \ try { \ @@ -2167,17 +2202,17 @@ namespace Catch { // The double negation silences MSVC's C4800 warning, the static_cast forces short-circuit evaluation if the type has overloaded &&. /////////////////////////////////////////////////////////////////////////////// -#define INTERNAL_CATCH_IF( expr, resultDisposition, macroName ) \ - INTERNAL_CATCH_TEST( expr, resultDisposition, macroName ); \ +#define INTERNAL_CATCH_IF( macroName, resultDisposition, expr ) \ + INTERNAL_CATCH_TEST( macroName, resultDisposition, expr ); \ if( Catch::getResultCapture().getLastResult()->succeeded() ) /////////////////////////////////////////////////////////////////////////////// -#define INTERNAL_CATCH_ELSE( expr, resultDisposition, macroName ) \ - INTERNAL_CATCH_TEST( expr, resultDisposition, macroName ); \ +#define INTERNAL_CATCH_ELSE( macroName, resultDisposition, expr ) \ + INTERNAL_CATCH_TEST( macroName, resultDisposition, expr ); \ if( !Catch::getResultCapture().getLastResult()->succeeded() ) /////////////////////////////////////////////////////////////////////////////// -#define INTERNAL_CATCH_NO_THROW( expr, resultDisposition, macroName ) \ +#define INTERNAL_CATCH_NO_THROW( macroName, resultDisposition, expr ) \ do { \ Catch::ResultBuilder __catchResult( macroName, CATCH_INTERNAL_LINEINFO, #expr, resultDisposition ); \ try { \ @@ -2191,7 +2226,7 @@ namespace Catch { } while( Catch::alwaysFalse() ) /////////////////////////////////////////////////////////////////////////////// -#define INTERNAL_CATCH_THROWS( expr, resultDisposition, matcher, macroName ) \ +#define INTERNAL_CATCH_THROWS( macroName, resultDisposition, matcher, expr ) \ do { \ Catch::ResultBuilder __catchResult( macroName, CATCH_INTERNAL_LINEINFO, #expr, resultDisposition, #matcher ); \ if( __catchResult.allowThrows() ) \ @@ -2208,7 +2243,7 @@ namespace Catch { } while( Catch::alwaysFalse() ) /////////////////////////////////////////////////////////////////////////////// -#define INTERNAL_CATCH_THROWS_AS( expr, exceptionType, resultDisposition, macroName ) \ +#define INTERNAL_CATCH_THROWS_AS( macroName, exceptionType, resultDisposition, expr ) \ do { \ Catch::ResultBuilder __catchResult( macroName, CATCH_INTERNAL_LINEINFO, #expr ", " #exceptionType, resultDisposition ); \ if( __catchResult.allowThrows() ) \ @@ -2216,7 +2251,7 @@ namespace Catch { static_cast(expr); \ __catchResult.captureResult( Catch::ResultWas::DidntThrowException ); \ } \ - catch( Catch::add_const::type>::type ) { \ + catch( exceptionType ) { \ __catchResult.captureResult( Catch::ResultWas::Ok ); \ } \ catch( ... ) { \ @@ -2229,7 +2264,7 @@ namespace Catch { /////////////////////////////////////////////////////////////////////////////// #ifdef CATCH_CONFIG_VARIADIC_MACROS - #define INTERNAL_CATCH_MSG( messageType, resultDisposition, macroName, ... ) \ + #define INTERNAL_CATCH_MSG( macroName, messageType, resultDisposition, ... ) \ do { \ Catch::ResultBuilder __catchResult( macroName, CATCH_INTERNAL_LINEINFO, "", resultDisposition ); \ __catchResult << __VA_ARGS__ + ::Catch::StreamEndStop(); \ @@ -2237,7 +2272,7 @@ namespace Catch { INTERNAL_CATCH_REACT( __catchResult ) \ } while( Catch::alwaysFalse() ) #else - #define INTERNAL_CATCH_MSG( messageType, resultDisposition, macroName, log ) \ + #define INTERNAL_CATCH_MSG( macroName, messageType, resultDisposition, log ) \ do { \ Catch::ResultBuilder __catchResult( macroName, CATCH_INTERNAL_LINEINFO, "", resultDisposition ); \ __catchResult << log + ::Catch::StreamEndStop(); \ @@ -2247,11 +2282,11 @@ namespace Catch { #endif /////////////////////////////////////////////////////////////////////////////// -#define INTERNAL_CATCH_INFO( log, macroName ) \ +#define INTERNAL_CATCH_INFO( macroName, log ) \ Catch::ScopedMessage INTERNAL_CATCH_UNIQUE_NAME( scopedMessage ) = Catch::MessageBuilder( macroName, CATCH_INTERNAL_LINEINFO, Catch::ResultWas::Info ) << log; /////////////////////////////////////////////////////////////////////////////// -#define INTERNAL_CHECK_THAT( arg, matcher, resultDisposition, macroName ) \ +#define INTERNAL_CHECK_THAT( macroName, matcher, resultDisposition, arg ) \ do { \ Catch::ResultBuilder __catchResult( macroName, CATCH_INTERNAL_LINEINFO, #arg ", " #matcher, resultDisposition ); \ try { \ @@ -2368,14 +2403,19 @@ namespace Catch { // #included from: catch_timer.h #define TWOBLUECUBES_CATCH_TIMER_H_INCLUDED -#ifdef CATCH_PLATFORM_WINDOWS -typedef unsigned long long uint64_t; +#ifdef _MSC_VER + +namespace Catch { + typedef unsigned long long UInt64; +} #else #include +namespace Catch { + typedef uint64_t UInt64; +} #endif namespace Catch { - class Timer { public: Timer() : m_ticks( 0 ) {} @@ -2385,7 +2425,7 @@ namespace Catch { double getElapsedSeconds() const; private: - uint64_t m_ticks; + UInt64 m_ticks; }; } // namespace Catch @@ -2424,7 +2464,6 @@ namespace Catch { // #included from: internal/catch_generators.hpp #define TWOBLUECUBES_CATCH_GENERATORS_HPP_INCLUDED -#include #include #include #include @@ -2538,7 +2577,7 @@ public: private: void move( CompositeGenerator& other ) { - std::copy( other.m_composed.begin(), other.m_composed.end(), std::back_inserter( m_composed ) ); + m_composed.insert( m_composed.end(), other.m_composed.begin(), other.m_composed.end() ); m_totalSize += other.m_totalSize; other.m_composed.clear(); } @@ -2620,12 +2659,15 @@ namespace Catch { struct IExceptionTranslator; struct IReporterRegistry; struct IReporterFactory; + struct ITagAliasRegistry; struct IRegistryHub { virtual ~IRegistryHub(); virtual IReporterRegistry const& getReporterRegistry() const = 0; virtual ITestCaseRegistry const& getTestCaseRegistry() const = 0; + virtual ITagAliasRegistry const& getTagAliasRegistry() const = 0; + virtual IExceptionTranslatorRegistry& getExceptionTranslatorRegistry() = 0; }; @@ -2635,6 +2677,7 @@ namespace Catch { virtual void registerListener( Ptr const& factory ) = 0; virtual void registerTest( TestCase const& testInfo ) = 0; virtual void registerTranslator( const IExceptionTranslator* translator ) = 0; + virtual void registerTagAlias( std::string const& alias, std::string const& tag, SourceLineInfo const& lineInfo ) = 0; }; IRegistryHub& getRegistryHub(); @@ -2737,15 +2780,21 @@ namespace Detail { return Approx( 0 ); } - Approx operator()( double value ) { - Approx approx( value ); +#if defined(CATCH_CONFIG_CPP11_TYPE_TRAITS) + + template ::value>::type> + Approx operator()( T value ) { + Approx approx( static_cast(value) ); approx.epsilon( m_epsilon ); approx.margin( m_margin ); approx.scale( m_scale ); return approx; } -#if defined(CATCH_CONFIG_CPP11_TYPE_TRAITS) + template ::value>::type> + explicit Approx( T value ): Approx(static_cast(value)) + {} + template ::value>::type> friend bool operator == ( const T& lhs, Approx const& rhs ) { // Thanks to Richard Harris for his help refining this formula @@ -2773,29 +2822,53 @@ namespace Detail { } template ::value>::type> - friend bool operator <= ( T lhs, Approx const& rhs ) - { - return double(lhs) < rhs.m_value || lhs == rhs; + friend bool operator <= ( T lhs, Approx const& rhs ) { + return double(lhs) < rhs.m_value || lhs == rhs; } template ::value>::type> - friend bool operator <= ( Approx const& lhs, T rhs ) - { - return lhs.m_value < double(rhs) || lhs == rhs; + friend bool operator <= ( Approx const& lhs, T rhs ) { + return lhs.m_value < double(rhs) || lhs == rhs; } template ::value>::type> - friend bool operator >= ( T lhs, Approx const& rhs ) - { - return double(lhs) > rhs.m_value || lhs == rhs; + friend bool operator >= ( T lhs, Approx const& rhs ) { + return double(lhs) > rhs.m_value || lhs == rhs; } template ::value>::type> - friend bool operator >= ( Approx const& lhs, T rhs ) - { - return lhs.m_value > double(rhs) || lhs == rhs; + friend bool operator >= ( Approx const& lhs, T rhs ) { + return lhs.m_value > double(rhs) || lhs == rhs; } + + template ::value>::type> + Approx& epsilon( T newEpsilon ) { + m_epsilon = double(newEpsilon); + return *this; + } + + template ::value>::type> + Approx& margin( T newMargin ) { + m_margin = double(newMargin); + return *this; + } + + template ::value>::type> + Approx& scale( T newScale ) { + m_scale = double(newScale); + return *this; + } + #else + + Approx operator()( double value ) { + Approx approx( value ); + approx.epsilon( m_epsilon ); + approx.margin( m_margin ); + approx.scale( m_scale ); + return approx; + } + friend bool operator == ( double lhs, Approx const& rhs ) { // Thanks to Richard Harris for his help refining this formula bool relativeOK = std::fabs( lhs - rhs.m_value ) < rhs.m_epsilon * (rhs.m_scale + (std::max)( std::fabs(lhs), std::fabs(rhs.m_value) ) ); @@ -2817,26 +2890,21 @@ namespace Detail { return !operator==( rhs, lhs ); } - friend bool operator <= ( double lhs, Approx const& rhs ) - { - return lhs < rhs.m_value || lhs == rhs; + friend bool operator <= ( double lhs, Approx const& rhs ) { + return lhs < rhs.m_value || lhs == rhs; } - friend bool operator <= ( Approx const& lhs, double rhs ) - { - return lhs.m_value < rhs || lhs == rhs; + friend bool operator <= ( Approx const& lhs, double rhs ) { + return lhs.m_value < rhs || lhs == rhs; } - friend bool operator >= ( double lhs, Approx const& rhs ) - { - return lhs > rhs.m_value || lhs == rhs; + friend bool operator >= ( double lhs, Approx const& rhs ) { + return lhs > rhs.m_value || lhs == rhs; } - friend bool operator >= ( Approx const& lhs, double rhs ) - { - return lhs.m_value > rhs || lhs == rhs; + friend bool operator >= ( Approx const& lhs, double rhs ) { + return lhs.m_value > rhs || lhs == rhs; } -#endif Approx& epsilon( double newEpsilon ) { m_epsilon = newEpsilon; @@ -2852,6 +2920,7 @@ namespace Detail { m_scale = newScale; return *this; } +#endif std::string toString() const { std::ostringstream oss; @@ -2893,7 +2962,7 @@ namespace Matchers { }; struct StringMatcherBase : MatcherBase { - StringMatcherBase( std::string operation, CasedString const& comparator ); + StringMatcherBase( std::string const& operation, CasedString const& comparator ); virtual std::string describe() const CATCH_OVERRIDE; CasedString m_comparator; @@ -3032,7 +3101,7 @@ namespace Matchers { namespace Catch { struct TagAlias { - TagAlias( std::string _tag, SourceLineInfo _lineInfo ) : tag( _tag ), lineInfo( _lineInfo ) {} + TagAlias( std::string const& _tag, SourceLineInfo _lineInfo ) : tag( _tag ), lineInfo( _lineInfo ) {} std::string tag; SourceLineInfo lineInfo; @@ -3104,8 +3173,18 @@ namespace Catch { } private: - T* nullableValue; - char storage[sizeof(T)]; + T *nullableValue; + union { + char storage[sizeof(T)]; + + // These are here to force alignment for the storage + long double dummy1; + void (*dummy2)(); + long double dummy3; +#ifdef CATCH_CONFIG_CPP11_LONG_LONG + long long dummy4; +#endif + }; }; } // end namespace Catch @@ -3304,64 +3383,67 @@ namespace Catch { namespace Impl { namespace NSStringMatchers { - template - struct StringHolder : MatcherImpl{ + struct StringHolder : MatcherBase{ StringHolder( NSString* substr ) : m_substr( [substr copy] ){} StringHolder( StringHolder const& other ) : m_substr( [other.m_substr copy] ){} StringHolder() { arcSafeRelease( m_substr ); } + virtual bool match( NSString* arg ) const CATCH_OVERRIDE { + return false; + } + NSString* m_substr; }; - struct Equals : StringHolder { + struct Equals : StringHolder { Equals( NSString* substr ) : StringHolder( substr ){} - virtual bool match( ExpressionType const& str ) const { + virtual bool match( NSString* str ) const CATCH_OVERRIDE { return (str != nil || m_substr == nil ) && [str isEqualToString:m_substr]; } - virtual std::string toString() const { + virtual std::string describe() const CATCH_OVERRIDE { return "equals string: " + Catch::toString( m_substr ); } }; - struct Contains : StringHolder { + struct Contains : StringHolder { Contains( NSString* substr ) : StringHolder( substr ){} - virtual bool match( ExpressionType const& str ) const { + virtual bool match( NSString* str ) const { return (str != nil || m_substr == nil ) && [str rangeOfString:m_substr].location != NSNotFound; } - virtual std::string toString() const { + virtual std::string describe() const CATCH_OVERRIDE { return "contains string: " + Catch::toString( m_substr ); } }; - struct StartsWith : StringHolder { + struct StartsWith : StringHolder { StartsWith( NSString* substr ) : StringHolder( substr ){} - virtual bool match( ExpressionType const& str ) const { + virtual bool match( NSString* str ) const { return (str != nil || m_substr == nil ) && [str rangeOfString:m_substr].location == 0; } - virtual std::string toString() const { + virtual std::string describe() const CATCH_OVERRIDE { return "starts with: " + Catch::toString( m_substr ); } }; - struct EndsWith : StringHolder { + struct EndsWith : StringHolder { EndsWith( NSString* substr ) : StringHolder( substr ){} - virtual bool match( ExpressionType const& str ) const { + virtual bool match( NSString* str ) const { return (str != nil || m_substr == nil ) && [str rangeOfString:m_substr].location == [str length] - [m_substr length]; } - virtual std::string toString() const { + virtual std::string describe() const CATCH_OVERRIDE { return "ends with: " + Catch::toString( m_substr ); } }; @@ -3408,16 +3490,16 @@ return @ desc; \ #include class LeakDetector { public: - LeakDetector() { - int flag = _CrtSetDbgFlag(_CRTDBG_REPORT_FLAG); - flag |= _CRTDBG_LEAK_CHECK_DF; - flag |= _CRTDBG_ALLOC_MEM_DF; - _CrtSetDbgFlag(flag); - _CrtSetReportMode(_CRT_WARN, _CRTDBG_MODE_FILE | _CRTDBG_MODE_DEBUG); - _CrtSetReportFile(_CRT_WARN, _CRTDBG_FILE_STDERR); - // Change this to leaking allocation's number to break there - _CrtSetBreakAlloc(-1); - } + LeakDetector() { + int flag = _CrtSetDbgFlag(_CRTDBG_REPORT_FLAG); + flag |= _CRTDBG_LEAK_CHECK_DF; + flag |= _CRTDBG_ALLOC_MEM_DF; + _CrtSetDbgFlag(flag); + _CrtSetReportMode(_CRT_WARN, _CRTDBG_MODE_FILE | _CRTDBG_MODE_DEBUG); + _CrtSetReportFile(_CRT_WARN, _CRTDBG_FILE_STDERR); + // Change this to leaking allocation's number to break there + _CrtSetBreakAlloc(-1); + } }; #else class LeakDetector {}; @@ -4020,6 +4102,7 @@ namespace Catch { #include #include #include +#include // Use optional outer namespace #ifdef STITCH_TBC_TEXT_FORMAT_OUTER_NAMESPACE @@ -4362,7 +4445,7 @@ namespace Clara { _dest = _source; } char toLowerCh(char c) { - return static_cast( ::tolower( c ) ); + return static_cast( std::tolower( c ) ); } inline void convertInto( std::string const& _source, bool& _dest ) { std::string sourceLC = _source; @@ -4516,12 +4599,13 @@ namespace Clara { } void parseIntoTokens( std::string const& arg, std::vector& tokens ) { - for( std::size_t i = 0; i <= arg.size(); ++i ) { + for( std::size_t i = 0; i < arg.size(); ++i ) { char c = arg[i]; if( c == '"' ) inQuotes = !inQuotes; mode = handleMode( i, c, arg, tokens ); } + mode = handleMode( arg.size(), '\0', arg, tokens ); } Mode handleMode( std::size_t i, char c, std::string const& arg, std::vector& tokens ) { switch( mode ) { @@ -4554,6 +4638,7 @@ namespace Clara { default: from = i; return ShortOpt; } } + Mode handleOpt( std::size_t i, char c, std::string const& arg, std::vector& tokens ) { if( std::string( ":=\0", 3 ).find( c ) == std::string::npos ) return mode; @@ -4885,7 +4970,7 @@ namespace Clara { } std::vector parseInto( std::vector const& args, ConfigT& config ) const { - std::string processName = args[0]; + std::string processName = args.empty() ? std::string() : args[0]; std::size_t lastSlash = processName.find_last_of( "/\\" ); if( lastSlash != std::string::npos ) processName = processName.substr( lastSlash+1 ); @@ -5871,13 +5956,14 @@ namespace Catch { // #included from: catch_test_case_tracker.hpp #define TWOBLUECUBES_CATCH_TEST_CASE_TRACKER_HPP_INCLUDED -#include +#include #include #include #include -#include #include +CATCH_INTERNAL_SUPPRESS_ETD_WARNINGS + namespace Catch { namespace TestCaseTracking { @@ -6148,12 +6234,12 @@ namespace TestCaseTracking { if( !filters.empty() ) { m_filters.push_back(""); // Root - should never be consulted m_filters.push_back(""); // Test Case - not a section filter - std::copy( filters.begin(), filters.end(), std::back_inserter( m_filters ) ); + m_filters.insert( m_filters.end(), filters.begin(), filters.end() ); } } void addNextFilters( std::vector const& filters ) { if( filters.size() > 1 ) - std::copy( filters.begin()+1, filters.end(), std::back_inserter( m_filters ) ); + m_filters.insert( m_filters.end(), ++filters.begin(), filters.end() ); } }; @@ -6223,6 +6309,8 @@ using TestCaseTracking::IndexTracker; } // namespace Catch +CATCH_INTERNAL_UNSUPPRESS_ETD_WARNINGS + // #included from: catch_fatal_condition.hpp #define TWOBLUECUBES_CATCH_FATAL_CONDITION_H_INCLUDED @@ -6292,7 +6380,6 @@ namespace Catch { static LONG CALLBACK handleVectoredException(PEXCEPTION_POINTERS ExceptionInfo) { for (int i = 0; i < sizeof(signalDefs) / sizeof(SignalDefs); ++i) { if (ExceptionInfo->ExceptionRecord->ExceptionCode == signalDefs[i].id) { - reset(); reportFatal(signalDefs[i].name); } } @@ -6476,7 +6563,8 @@ namespace Catch { m_context( getCurrentMutableContext() ), m_activeTestCase( CATCH_NULL ), m_config( _config ), - m_reporter( reporter ) + m_reporter( reporter ), + m_shouldReportUnexpected ( true ) { m_context.setRunner( this ); m_context.setConfig( m_config ); @@ -6554,8 +6642,9 @@ namespace Catch { m_totals.assertions.failed++; } - if( m_reporter->assertionEnded( AssertionStats( result, m_messages, m_totals ) ) ) - m_messages.clear(); + // We have no use for the return value (whether messages should be cleared), because messages were made scoped + // and should be let to clear themselves out. + static_cast(m_reporter->assertionEnded(AssertionStats(result, m_messages, m_totals))); // Reset working state m_lastAssertionInfo = AssertionInfo( std::string(), m_lastAssertionInfo.lineInfo, "{Unknown expression after the reported line}" , m_lastAssertionInfo.resultDisposition ); @@ -6633,11 +6722,19 @@ namespace Catch { return &m_lastResult; } + virtual void exceptionEarlyReported() { + m_shouldReportUnexpected = false; + } + virtual void handleFatalErrorCondition( std::string const& message ) { - ResultBuilder resultBuilder = makeUnexpectedResultBuilder(); - resultBuilder.setResultType( ResultWas::FatalErrorCondition ); - resultBuilder << message; - resultBuilder.captureExpression(); + // Don't rebuild the result -- the stringification itself can cause more fatal errors + // Instead, fake a result data. + AssertionResultData tempResult; + tempResult.resultType = ResultWas::FatalErrorCondition; + tempResult.message = message; + AssertionResult result(m_lastAssertionInfo, tempResult); + + getResultCapture().assertionEnded(result); handleUnfinishedSections(); @@ -6678,6 +6775,7 @@ namespace Catch { m_reporter->sectionStarting( testCaseSection ); Counts prevAssertions = m_totals.assertions; double duration = 0; + m_shouldReportUnexpected = true; try { m_lastAssertionInfo = AssertionInfo( "TEST_CASE", testCaseInfo.lineInfo, std::string(), ResultDisposition::Normal ); @@ -6699,7 +6797,11 @@ namespace Catch { // This just means the test was aborted due to failure } catch(...) { - makeUnexpectedResultBuilder().useActiveException(); + // Under CATCH_CONFIG_FAST_COMPILE, unexpected exceptions under REQUIRE assertions + // are reported without translation at the point of origin. + if (m_shouldReportUnexpected) { + makeUnexpectedResultBuilder().useActiveException(); + } } m_testCaseTracker->close(); handleUnfinishedSections(); @@ -6759,6 +6861,7 @@ namespace Catch { std::vector m_unfinishedSections; std::vector m_activeSections; TrackerContext m_trackerContext; + bool m_shouldReportUnexpected; }; IResultCapture& getResultCapture() { @@ -6780,7 +6883,7 @@ namespace Catch { Version( unsigned int _majorVersion, unsigned int _minorVersion, unsigned int _patchNumber, - std::string const& _branchName, + char const * const _branchName, unsigned int _buildNumber ); unsigned int const majorVersion; @@ -6788,7 +6891,7 @@ namespace Catch { unsigned int const patchNumber; // buildNumber is only used if branchName is not null - std::string const branchName; + char const * const branchName; unsigned int const buildNumber; friend std::ostream& operator << ( std::ostream& os, Version const& version ); @@ -6797,7 +6900,7 @@ namespace Catch { void operator=( Version const& ); }; - extern Version libraryVersion; + inline Version libraryVersion(); } #include @@ -6909,7 +7012,7 @@ namespace Catch { } void showHelp( std::string const& processName ) { - Catch::cout() << "\nCatch v" << libraryVersion << "\n"; + Catch::cout() << "\nCatch v" << libraryVersion() << "\n"; m_cli.usage( Catch::cout(), processName ); Catch::cout() << "For more detail usage please see the project docs\n" << std::endl; @@ -6950,6 +7053,32 @@ namespace Catch { return returnCode; } + #if defined(WIN32) && defined(UNICODE) + int run( int argc, wchar_t const* const* const argv ) { + + char **utf8Argv = new char *[ argc ]; + + for ( int i = 0; i < argc; ++i ) { + int bufSize = WideCharToMultiByte( CP_UTF8, 0, argv[i], -1, NULL, 0, NULL, NULL ); + + utf8Argv[ i ] = new char[ bufSize ]; + + WideCharToMultiByte( CP_UTF8, 0, argv[i], -1, utf8Argv[i], bufSize, NULL, NULL ); + } + + int returnCode = applyCommandLine( argc, utf8Argv ); + if( returnCode == 0 ) + returnCode = run(); + + for ( int i = 0; i < argc; ++i ) + delete [] utf8Argv[ i ]; + + delete [] utf8Argv; + + return returnCode; + } + #endif + int run() { if( m_configData.showHelp ) return 0; @@ -7297,6 +7426,26 @@ namespace Catch { }; } +// #included from: catch_tag_alias_registry.h +#define TWOBLUECUBES_CATCH_TAG_ALIAS_REGISTRY_H_INCLUDED + +#include + +namespace Catch { + + class TagAliasRegistry : public ITagAliasRegistry { + public: + virtual ~TagAliasRegistry(); + virtual Option find( std::string const& alias ) const; + virtual std::string expandAliases( std::string const& unexpandedTestSpec ) const; + void add( std::string const& alias, std::string const& tag, SourceLineInfo const& lineInfo ); + + private: + std::map m_registry; + }; + +} // end namespace Catch + namespace Catch { namespace { @@ -7318,6 +7467,9 @@ namespace Catch { virtual IExceptionTranslatorRegistry& getExceptionTranslatorRegistry() CATCH_OVERRIDE { return m_exceptionTranslatorRegistry; } + virtual ITagAliasRegistry const& getTagAliasRegistry() const CATCH_OVERRIDE { + return m_tagAliasRegistry; + } public: // IMutableRegistryHub virtual void registerReporter( std::string const& name, Ptr const& factory ) CATCH_OVERRIDE { @@ -7332,11 +7484,15 @@ namespace Catch { virtual void registerTranslator( const IExceptionTranslator* translator ) CATCH_OVERRIDE { m_exceptionTranslatorRegistry.registerTranslator( translator ); } + virtual void registerTagAlias( std::string const& alias, std::string const& tag, SourceLineInfo const& lineInfo ) CATCH_OVERRIDE { + m_tagAliasRegistry.add( alias, tag, lineInfo ); + } private: TestRegistry m_testCaseRegistry; ReporterRegistry m_reporterRegistry; ExceptionTranslatorRegistry m_exceptionTranslatorRegistry; + TagAliasRegistry m_tagAliasRegistry; }; // Single, global, instance @@ -7581,6 +7737,23 @@ namespace Catch { // #included from: catch_console_colour_impl.hpp #define TWOBLUECUBES_CATCH_CONSOLE_COLOUR_IMPL_HPP_INCLUDED +// #included from: catch_errno_guard.hpp +#define TWOBLUECUBES_CATCH_ERRNO_GUARD_HPP_INCLUDED + +#include + +namespace Catch { + + class ErrnoGuard { + public: + ErrnoGuard():m_oldErrno(errno){} + ~ErrnoGuard() { errno = m_oldErrno; } + private: + int m_oldErrno; + }; + +} + namespace Catch { namespace { @@ -7716,6 +7889,7 @@ namespace { }; IColourImpl* platformColourInstance() { + ErrnoGuard guard; Ptr config = getCurrentContext().getConfig(); UseColour::YesOrNo colourMode = config ? config->useColour() @@ -7945,17 +8119,13 @@ namespace Catch { } inline void enforceNotReservedTag( std::string const& tag, SourceLineInfo const& _lineInfo ) { if( isReservedTag( tag ) ) { - { - Colour colourGuard( Colour::Red ); - Catch::cerr() - << "Tag name [" << tag << "] not allowed.\n" - << "Tag names starting with non alpha-numeric characters are reserved\n"; - } - { - Colour colourGuard( Colour::FileName ); - Catch::cerr() << _lineInfo << std::endl; - } - exit(1); + std::ostringstream ss; + ss << Colour(Colour::Red) + << "Tag name [" << tag << "] not allowed.\n" + << "Tag names starting with non alpha-numeric characters are reserved\n" + << Colour(Colour::FileName) + << _lineInfo << '\n'; + throw std::runtime_error(ss.str()); } } @@ -8117,7 +8287,7 @@ namespace Catch { ( unsigned int _majorVersion, unsigned int _minorVersion, unsigned int _patchNumber, - std::string const& _branchName, + char const * const _branchName, unsigned int _buildNumber ) : majorVersion( _majorVersion ), minorVersion( _minorVersion ), @@ -8130,15 +8300,18 @@ namespace Catch { os << version.majorVersion << '.' << version.minorVersion << '.' << version.patchNumber; - - if( !version.branchName.empty() ) { - os << '-' << version.branchName - << '.' << version.buildNumber; + // branchName is never null -> 0th char is \0 if it is empty + if (version.branchName[0]) { + os << '-' << version.branchName + << '.' << version.buildNumber; } return os; } - Version libraryVersion( 1, 8, 1, "", 0 ); + inline Version libraryVersion() { + static Version version( 1, 9, 5, "", 0 ); + return version; + } } @@ -8172,7 +8345,9 @@ namespace Catch { {} ScopedMessage::~ScopedMessage() { - getResultCapture().popScopedMessage( m_info ); + if ( !std::uncaught_exception() ){ + getResultCapture().popScopedMessage(m_info); + } } } // end namespace Catch @@ -8320,21 +8495,21 @@ namespace Catch { namespace { #ifdef CATCH_PLATFORM_WINDOWS - uint64_t getCurrentTicks() { - static uint64_t hz=0, hzo=0; + UInt64 getCurrentTicks() { + static UInt64 hz=0, hzo=0; if (!hz) { QueryPerformanceFrequency( reinterpret_cast( &hz ) ); QueryPerformanceCounter( reinterpret_cast( &hzo ) ); } - uint64_t t; + UInt64 t; QueryPerformanceCounter( reinterpret_cast( &t ) ); return ((t-hzo)*1000000)/hz; } #else - uint64_t getCurrentTicks() { + UInt64 getCurrentTicks() { timeval t; gettimeofday(&t,CATCH_NULL); - return static_cast( t.tv_sec ) * 1000000ull + static_cast( t.tv_usec ); + return static_cast( t.tv_sec ) * 1000000ull + static_cast( t.tv_usec ); } #endif } @@ -8486,6 +8661,10 @@ namespace Catch { m_timer.start(); } +#if defined(_MSC_VER) +#pragma warning(push) +#pragma warning(disable:4996) // std::uncaught_exception is deprecated in C++17 +#endif Section::~Section() { if( m_sectionIncluded ) { SectionEndInfo endInfo( m_info, m_assertions, m_timer.getElapsedSeconds() ); @@ -8495,6 +8674,9 @@ namespace Catch { getResultCapture().sectionEnded( endInfo ); } } +#if defined(_MSC_VER) +#pragma warning(pop) +#endif // This indicates whether the section should be executed or not Section::operator bool() const { @@ -8567,6 +8749,9 @@ namespace Catch { // be strace, for example) in /proc/$PID/status, so just get it from // there instead. bool isDebuggerActive(){ + // Libstdc++ has a bug, where std::ifstream sets errno to 0 + // This way our users can properly assert over errno values + ErrnoGuard guard; std::ifstream in("/proc/self/status"); for( std::string line; std::getline(in, line); ) { static const int PREFIX_LEN = 11; @@ -8807,7 +8992,7 @@ std::string toString( std::nullptr_t ) { return "nil"; return "@" + toString([nsstring UTF8String]); } - std::string toString( NSString * CATCH_ARC_STRONG const& nsstring ) { + std::string toString( NSString * CATCH_ARC_STRONG & nsstring ) { if( !nsstring ) return "nil"; return "@" + toString([nsstring UTF8String]); @@ -8836,9 +9021,20 @@ namespace Catch { char const* secondArg ) : m_assertionInfo( macroName, lineInfo, capturedExpressionWithSecondArgument( capturedExpression, secondArg ), resultDisposition ), m_shouldDebugBreak( false ), - m_shouldThrow( false ) + m_shouldThrow( false ), + m_guardException( false ) {} + ResultBuilder::~ResultBuilder() { +#if defined(CATCH_CONFIG_FAST_COMPILE) + if ( m_guardException ) { + m_stream.oss << "Exception translation was disabled by CATCH_CONFIG_FAST_COMPILE"; + captureResult( ResultWas::ThrewException ); + getCurrentContext().getResultCapture()->exceptionEarlyReported(); + } +#endif + } + ResultBuilder& ResultBuilder::setResultType( ResultWas::OfType result ) { m_data.resultType = result; return *this; @@ -8951,32 +9147,18 @@ namespace Catch { dest = m_assertionInfo.capturedExpression; } + void ResultBuilder::setExceptionGuard() { + m_guardException = true; + } + void ResultBuilder::unsetExceptionGuard() { + m_guardException = false; + } + } // end namespace Catch // #included from: catch_tag_alias_registry.hpp #define TWOBLUECUBES_CATCH_TAG_ALIAS_REGISTRY_HPP_INCLUDED -// #included from: catch_tag_alias_registry.h -#define TWOBLUECUBES_CATCH_TAG_ALIAS_REGISTRY_H_INCLUDED - -#include - -namespace Catch { - - class TagAliasRegistry : public ITagAliasRegistry { - public: - virtual ~TagAliasRegistry(); - virtual Option find( std::string const& alias ) const; - virtual std::string expandAliases( std::string const& unexpandedTestSpec ) const; - void add( char const* alias, char const* tag, SourceLineInfo const& lineInfo ); - static TagAliasRegistry& get(); - - private: - std::map m_registry; - }; - -} // end namespace Catch - namespace Catch { TagAliasRegistry::~TagAliasRegistry() {} @@ -9004,40 +9186,36 @@ namespace Catch { return expandedTestSpec; } - void TagAliasRegistry::add( char const* alias, char const* tag, SourceLineInfo const& lineInfo ) { + void TagAliasRegistry::add( std::string const& alias, std::string const& tag, SourceLineInfo const& lineInfo ) { if( !startsWith( alias, "[@" ) || !endsWith( alias, ']' ) ) { std::ostringstream oss; - oss << "error: tag alias, \"" << alias << "\" is not of the form [@alias name].\n" << lineInfo; + oss << Colour( Colour::Red ) + << "error: tag alias, \"" << alias << "\" is not of the form [@alias name].\n" + << Colour( Colour::FileName ) + << lineInfo << '\n'; throw std::domain_error( oss.str().c_str() ); } if( !m_registry.insert( std::make_pair( alias, TagAlias( tag, lineInfo ) ) ).second ) { std::ostringstream oss; - oss << "error: tag alias, \"" << alias << "\" already registered.\n" - << "\tFirst seen at " << find(alias)->lineInfo << '\n' - << "\tRedefined at " << lineInfo; + oss << Colour( Colour::Red ) + << "error: tag alias, \"" << alias << "\" already registered.\n" + << "\tFirst seen at " + << Colour( Colour::Red ) << find(alias)->lineInfo << '\n' + << Colour( Colour::Red ) << "\tRedefined at " + << Colour( Colour::FileName) << lineInfo << '\n'; throw std::domain_error( oss.str().c_str() ); } } - TagAliasRegistry& TagAliasRegistry::get() { - static TagAliasRegistry instance; - return instance; + ITagAliasRegistry::~ITagAliasRegistry() {} + ITagAliasRegistry const& ITagAliasRegistry::get() { + return getRegistryHub().getTagAliasRegistry(); } - ITagAliasRegistry::~ITagAliasRegistry() {} - ITagAliasRegistry const& ITagAliasRegistry::get() { return TagAliasRegistry::get(); } - RegistrarForTagAliases::RegistrarForTagAliases( char const* alias, char const* tag, SourceLineInfo const& lineInfo ) { - try { - TagAliasRegistry::get().add( alias, tag, lineInfo ); - } - catch( std::exception& ex ) { - Colour colourGuard( Colour::Red ); - Catch::cerr() << ex.what() << std::endl; - exit(1); - } + getMutableRegistryHub().registerTagAlias( alias, tag, lineInfo ); } } // end namespace Catch @@ -9064,7 +9242,7 @@ namespace Matchers { : std::string(); } - StringMatcherBase::StringMatcherBase( std::string operation, CasedString const& comparator ) + StringMatcherBase::StringMatcherBase( std::string const& operation, CasedString const& comparator ) : m_comparator( comparator ), m_operation( operation ) { } @@ -9265,10 +9443,34 @@ Ptr addReporter( Ptr const& existingRepo #define TWOBLUECUBES_CATCH_REPORTER_BASES_HPP_INCLUDED #include +#include +#include #include namespace Catch { + namespace { + // Because formatting using c++ streams is stateful, drop down to C is required + // Alternatively we could use stringstream, but its performance is... not good. + std::string getFormattedDuration( double duration ) { + // Max exponent + 1 is required to represent the whole part + // + 1 for decimal point + // + 3 for the 3 decimal places + // + 1 for null terminator + const size_t maxDoubleSize = DBL_MAX_10_EXP + 1 + 1 + 3 + 1; + char buffer[maxDoubleSize]; + + // Save previous errno, to prevent sprintf from overwriting it + ErrnoGuard guard; +#ifdef _MSC_VER + sprintf_s(buffer, "%.3f", duration); +#else + sprintf(buffer, "%.3f", duration); +#endif + return std::string(buffer); + } + } + struct StreamingReporterBase : SharedImpl { StreamingReporterBase( ReporterConfig const& _config ) @@ -9594,9 +9796,13 @@ namespace Catch { #define INTERNAL_CATCH_REGISTER_REPORTER( name, reporterType ) \ namespace{ Catch::ReporterRegistrar catch_internal_RegistrarFor##reporterType( name ); } +// Deprecated - use the form without INTERNAL_ #define INTERNAL_CATCH_REGISTER_LISTENER( listenerType ) \ namespace{ Catch::ListenerRegistrar catch_internal_RegistrarFor##listenerType; } +#define CATCH_REGISTER_LISTENER( listenerType ) \ + namespace{ Catch::ListenerRegistrar catch_internal_RegistrarFor##listenerType; } + // #included from: ../internal/catch_xmlwriter.hpp #define TWOBLUECUBES_CATCH_XMLWRITER_HPP_INCLUDED @@ -9829,20 +10035,6 @@ namespace Catch { }; } -// #included from: catch_reenable_warnings.h - -#define TWOBLUECUBES_CATCH_REENABLE_WARNINGS_H_INCLUDED - -#ifdef __clang__ -# ifdef __ICC // icpc defines the __clang__ macro -# pragma warning(pop) -# else -# pragma clang diagnostic pop -# endif -#elif defined __GNUC__ -# pragma GCC diagnostic pop -#endif - namespace Catch { class XmlReporter : public StreamingReporterBase { @@ -9921,73 +10113,76 @@ namespace Catch { virtual void assertionStarting( AssertionInfo const& ) CATCH_OVERRIDE { } virtual bool assertionEnded( AssertionStats const& assertionStats ) CATCH_OVERRIDE { - const AssertionResult& assertionResult = assertionStats.assertionResult; - // Print any info messages in tags. - if( assertionStats.assertionResult.getResultType() != ResultWas::Ok ) { + AssertionResult const& result = assertionStats.assertionResult; + + bool includeResults = m_config->includeSuccessfulResults() || !result.isOk(); + + if( includeResults ) { + // Print any info messages in tags. for( std::vector::const_iterator it = assertionStats.infoMessages.begin(), itEnd = assertionStats.infoMessages.end(); - it != itEnd; - ++it ) { + it != itEnd; + ++it ) { if( it->type == ResultWas::Info ) { m_xml.scopedElement( "Info" ) - .writeText( it->message ); + .writeText( it->message ); } else if ( it->type == ResultWas::Warning ) { m_xml.scopedElement( "Warning" ) - .writeText( it->message ); + .writeText( it->message ); } } } // Drop out if result was successful but we're not printing them. - if( !m_config->includeSuccessfulResults() && isOk(assertionResult.getResultType()) ) + if( !includeResults && result.getResultType() != ResultWas::Warning ) return true; // Print the expression if there is one. - if( assertionResult.hasExpression() ) { + if( result.hasExpression() ) { m_xml.startElement( "Expression" ) - .writeAttribute( "success", assertionResult.succeeded() ) - .writeAttribute( "type", assertionResult.getTestMacroName() ); + .writeAttribute( "success", result.succeeded() ) + .writeAttribute( "type", result.getTestMacroName() ); - writeSourceInfo( assertionResult.getSourceInfo() ); + writeSourceInfo( result.getSourceInfo() ); m_xml.scopedElement( "Original" ) - .writeText( assertionResult.getExpression() ); + .writeText( result.getExpression() ); m_xml.scopedElement( "Expanded" ) - .writeText( assertionResult.getExpandedExpression() ); + .writeText( result.getExpandedExpression() ); } // And... Print a result applicable to each result type. - switch( assertionResult.getResultType() ) { + switch( result.getResultType() ) { case ResultWas::ThrewException: m_xml.startElement( "Exception" ); - writeSourceInfo( assertionResult.getSourceInfo() ); - m_xml.writeText( assertionResult.getMessage() ); + writeSourceInfo( result.getSourceInfo() ); + m_xml.writeText( result.getMessage() ); m_xml.endElement(); break; case ResultWas::FatalErrorCondition: m_xml.startElement( "FatalErrorCondition" ); - writeSourceInfo( assertionResult.getSourceInfo() ); - m_xml.writeText( assertionResult.getMessage() ); + writeSourceInfo( result.getSourceInfo() ); + m_xml.writeText( result.getMessage() ); m_xml.endElement(); break; case ResultWas::Info: m_xml.scopedElement( "Info" ) - .writeText( assertionResult.getMessage() ); + .writeText( result.getMessage() ); break; case ResultWas::Warning: // Warning will already have been written break; case ResultWas::ExplicitFailure: m_xml.startElement( "Failure" ); - writeSourceInfo( assertionResult.getSourceInfo() ); - m_xml.writeText( assertionResult.getMessage() ); + writeSourceInfo( result.getSourceInfo() ); + m_xml.writeText( result.getMessage() ); m_xml.endElement(); break; default: break; } - if( assertionResult.hasExpression() ) + if( result.hasExpression() ) m_xml.endElement(); return true; @@ -10093,7 +10288,8 @@ namespace Catch { public: JunitReporter( ReporterConfig const& _config ) : CumulativeReporterBase( _config ), - xml( _config.stream() ) + xml( _config.stream() ), + m_okToFail( false ) { m_reporterPrefs.shouldRedirectStdOut = true; } @@ -10119,8 +10315,11 @@ namespace Catch { CumulativeReporterBase::testGroupStarting( groupInfo ); } + virtual void testCaseStarting( TestCaseInfo const& testCaseInfo ) CATCH_OVERRIDE { + m_okToFail = testCaseInfo.okToFail(); + } virtual bool assertionEnded( AssertionStats const& assertionStats ) CATCH_OVERRIDE { - if( assertionStats.assertionResult.getResultType() == ResultWas::ThrewException ) + if( assertionStats.assertionResult.getResultType() == ResultWas::ThrewException && !m_okToFail ) unexpectedExceptions++; return CumulativeReporterBase::assertionEnded( assertionStats ); } @@ -10285,6 +10484,7 @@ namespace Catch { std::ostringstream stdOutForSuite; std::ostringstream stdErrForSuite; unsigned int unexpectedExceptions; + bool m_okToFail; }; INTERNAL_CATCH_REGISTER_REPORTER( "junit", JunitReporter ) @@ -10299,25 +10499,6 @@ namespace Catch { namespace Catch { - namespace { - // Because formatting using c++ streams is stateful, drop down to C is required - // Alternatively we could use stringstream, but its performance is... not good. - std::string getFormattedDuration( double duration ) { - // Max exponent + 1 is required to represent the whole part - // + 1 for decimal point - // + 3 for the 3 decimal places - // + 1 for null terminator - const size_t maxDoubleSize = DBL_MAX_10_EXP + 1 + 1 + 3 + 1; - char buffer[maxDoubleSize]; -#ifdef _MSC_VER - sprintf_s(buffer, "%.3f", duration); -#else - sprintf(buffer, "%.3f", duration); -#endif - return std::string(buffer); - } - } - struct ConsoleReporter : StreamingReporterBase { ConsoleReporter( ReporterConfig const& _config ) : StreamingReporterBase( _config ), @@ -10339,18 +10520,15 @@ namespace Catch { virtual bool assertionEnded( AssertionStats const& _assertionStats ) CATCH_OVERRIDE { AssertionResult const& result = _assertionStats.assertionResult; - bool printInfoMessages = true; + bool includeResults = m_config->includeSuccessfulResults() || !result.isOk(); - // Drop out if result was successful and we're not printing those - if( !m_config->includeSuccessfulResults() && result.isOk() ) { - if( result.getResultType() != ResultWas::Warning ) - return false; - printInfoMessages = false; - } + // Drop out if result was successful but we're not printing them. + if( !includeResults && result.getResultType() != ResultWas::Warning ) + return false; lazyPrint(); - AssertionPrinter printer( stream, _assertionStats, printInfoMessages ); + AssertionPrinter printer( stream, _assertionStats, includeResults ); printer.print(); stream << std::endl; return true; @@ -10440,7 +10618,11 @@ namespace Catch { case ResultWas::ThrewException: colour = Colour::Error; passOrFail = "FAILED"; - messageLabel = "due to unexpected exception with message"; + messageLabel = "due to unexpected exception with "; + if (_stats.infoMessages.size() == 1) + messageLabel += "message"; + if (_stats.infoMessages.size() > 1) + messageLabel += "messages"; break; case ResultWas::FatalErrorCondition: colour = Colour::Error; @@ -10556,7 +10738,7 @@ namespace Catch { stream << '\n' << getLineOfChars<'~'>() << '\n'; Colour colour( Colour::SecondaryText ); stream << currentTestRunInfo->name - << " is a Catch v" << libraryVersion << " host application.\n" + << " is a Catch v" << libraryVersion() << " host application.\n" << "Run with -? for options\n\n"; if( m_config->rngSeed() != 0 ) @@ -10769,8 +10951,7 @@ namespace Catch { stream << "No test cases matched '" << spec << '\'' << std::endl; } - virtual void assertionStarting( AssertionInfo const& ) { - } + virtual void assertionStarting( AssertionInfo const& ) {} virtual bool assertionEnded( AssertionStats const& _assertionStats ) { AssertionResult const& result = _assertionStats.assertionResult; @@ -10791,6 +10972,12 @@ namespace Catch { return true; } + virtual void sectionEnded(SectionStats const& _sectionStats) CATCH_OVERRIDE { + if (m_config->showDurations() == ShowDurations::Always) { + stream << getFormattedDuration(_sectionStats.durationInSeconds) << " s: " << _sectionStats.sectionInfo.name << std::endl; + } + } + virtual void testRunEnded( TestRunStats const& _testRunStats ) { printTotals( _testRunStats.totals ); stream << '\n' << std::endl; @@ -10896,7 +11083,7 @@ namespace Catch { stream << result.getSourceInfo() << ':'; } - void printResultType( Colour::Code colour, std::string passOrFail ) const { + void printResultType( Colour::Code colour, std::string const& passOrFail ) const { if( !passOrFail.empty() ) { { Colour colourGuard( colour ); @@ -10906,7 +11093,7 @@ namespace Catch { } } - void printIssue( std::string issue ) const { + void printIssue( std::string const& issue ) const { stream << ' ' << issue; } @@ -11077,6 +11264,7 @@ namespace Catch { TestSpec::NamePattern::~NamePattern() {} TestSpec::TagPattern::~TagPattern() {} TestSpec::ExcludedPattern::~ExcludedPattern() {} + Matchers::Impl::MatcherUntypedBase::~MatcherUntypedBase() {} void Config::dummy() {} @@ -11100,9 +11288,15 @@ namespace Catch { #ifndef __OBJC__ +#if defined(WIN32) && defined(_UNICODE) && !defined(DO_NOT_USE_WMAIN) +// Standard C/C++ Win32 Unicode wmain entry point +extern "C" int wmain (int argc, wchar_t * argv[], wchar_t * []) { +#else // Standard C/C++ main entry point int main (int argc, char * argv[]) { - int result = Catch::Session().run( argc, argv ); +#endif + + int result = Catch::Session().run( argc, argv ); return ( result < 0xff ? result : 0xff ); } @@ -11137,33 +11331,43 @@ int main (int argc, char * const argv[]) { // If this config identifier is defined then all CATCH macros are prefixed with CATCH_ #ifdef CATCH_CONFIG_PREFIX_ALL -#define CATCH_REQUIRE( expr ) INTERNAL_CATCH_TEST( expr, Catch::ResultDisposition::Normal, "CATCH_REQUIRE" ) -#define CATCH_REQUIRE_FALSE( expr ) INTERNAL_CATCH_TEST( expr, Catch::ResultDisposition::Normal | Catch::ResultDisposition::FalseTest, "CATCH_REQUIRE_FALSE" ) +#if defined(CATCH_CONFIG_FAST_COMPILE) +#define CATCH_REQUIRE( expr ) INTERNAL_CATCH_TEST_NO_TRY( "CATCH_REQUIRE", Catch::ResultDisposition::Normal, expr ) +#define CATCH_REQUIRE_FALSE( expr ) INTERNAL_CATCH_TEST_NO_TRY( "CATCH_REQUIRE_FALSE", Catch::ResultDisposition::Normal | Catch::ResultDisposition::FalseTest, expr ) +#else +#define CATCH_REQUIRE( expr ) INTERNAL_CATCH_TEST( "CATCH_REQUIRE", Catch::ResultDisposition::Normal, expr ) +#define CATCH_REQUIRE_FALSE( expr ) INTERNAL_CATCH_TEST( "CATCH_REQUIRE_FALSE", Catch::ResultDisposition::Normal | Catch::ResultDisposition::FalseTest, expr ) +#endif -#define CATCH_REQUIRE_THROWS( expr ) INTERNAL_CATCH_THROWS( expr, Catch::ResultDisposition::Normal, "", "CATCH_REQUIRE_THROWS" ) -#define CATCH_REQUIRE_THROWS_AS( expr, exceptionType ) INTERNAL_CATCH_THROWS_AS( expr, exceptionType, Catch::ResultDisposition::Normal, "CATCH_REQUIRE_THROWS_AS" ) -#define CATCH_REQUIRE_THROWS_WITH( expr, matcher ) INTERNAL_CATCH_THROWS( expr, Catch::ResultDisposition::Normal, matcher, "CATCH_REQUIRE_THROWS_WITH" ) -#define CATCH_REQUIRE_NOTHROW( expr ) INTERNAL_CATCH_NO_THROW( expr, Catch::ResultDisposition::Normal, "CATCH_REQUIRE_NOTHROW" ) +#define CATCH_REQUIRE_THROWS( expr ) INTERNAL_CATCH_THROWS( "CATCH_REQUIRE_THROWS", Catch::ResultDisposition::Normal, "", expr ) +#define CATCH_REQUIRE_THROWS_AS( expr, exceptionType ) INTERNAL_CATCH_THROWS_AS( "CATCH_REQUIRE_THROWS_AS", exceptionType, Catch::ResultDisposition::Normal, expr ) +#define CATCH_REQUIRE_THROWS_WITH( expr, matcher ) INTERNAL_CATCH_THROWS( "CATCH_REQUIRE_THROWS_WITH", Catch::ResultDisposition::Normal, matcher, expr ) +#define CATCH_REQUIRE_NOTHROW( expr ) INTERNAL_CATCH_NO_THROW( "CATCH_REQUIRE_NOTHROW", Catch::ResultDisposition::Normal, expr ) -#define CATCH_CHECK( expr ) INTERNAL_CATCH_TEST( expr, Catch::ResultDisposition::ContinueOnFailure, "CATCH_CHECK" ) -#define CATCH_CHECK_FALSE( expr ) INTERNAL_CATCH_TEST( expr, Catch::ResultDisposition::ContinueOnFailure | Catch::ResultDisposition::FalseTest, "CATCH_CHECK_FALSE" ) -#define CATCH_CHECKED_IF( expr ) INTERNAL_CATCH_IF( expr, Catch::ResultDisposition::ContinueOnFailure, "CATCH_CHECKED_IF" ) -#define CATCH_CHECKED_ELSE( expr ) INTERNAL_CATCH_ELSE( expr, Catch::ResultDisposition::ContinueOnFailure, "CATCH_CHECKED_ELSE" ) -#define CATCH_CHECK_NOFAIL( expr ) INTERNAL_CATCH_TEST( expr, Catch::ResultDisposition::ContinueOnFailure | Catch::ResultDisposition::SuppressFail, "CATCH_CHECK_NOFAIL" ) +#define CATCH_CHECK( expr ) INTERNAL_CATCH_TEST( "CATCH_CHECK", Catch::ResultDisposition::ContinueOnFailure, expr ) +#define CATCH_CHECK_FALSE( expr ) INTERNAL_CATCH_TEST( "CATCH_CHECK_FALSE", Catch::ResultDisposition::ContinueOnFailure | Catch::ResultDisposition::FalseTest, expr ) +#define CATCH_CHECKED_IF( expr ) INTERNAL_CATCH_IF( "CATCH_CHECKED_IF", Catch::ResultDisposition::ContinueOnFailure, expr ) +#define CATCH_CHECKED_ELSE( expr ) INTERNAL_CATCH_ELSE( "CATCH_CHECKED_ELSE", Catch::ResultDisposition::ContinueOnFailure, expr ) +#define CATCH_CHECK_NOFAIL( expr ) INTERNAL_CATCH_TEST( "CATCH_CHECK_NOFAIL", Catch::ResultDisposition::ContinueOnFailure | Catch::ResultDisposition::SuppressFail, expr ) -#define CATCH_CHECK_THROWS( expr ) INTERNAL_CATCH_THROWS( expr, Catch::ResultDisposition::ContinueOnFailure, "", "CATCH_CHECK_THROWS" ) -#define CATCH_CHECK_THROWS_AS( expr, exceptionType ) INTERNAL_CATCH_THROWS_AS( expr, exceptionType, Catch::ResultDisposition::ContinueOnFailure, "CATCH_CHECK_THROWS_AS" ) -#define CATCH_CHECK_THROWS_WITH( expr, matcher ) INTERNAL_CATCH_THROWS( expr, Catch::ResultDisposition::ContinueOnFailure, matcher, "CATCH_CHECK_THROWS_WITH" ) -#define CATCH_CHECK_NOTHROW( expr ) INTERNAL_CATCH_NO_THROW( expr, Catch::ResultDisposition::ContinueOnFailure, "CATCH_CHECK_NOTHROW" ) +#define CATCH_CHECK_THROWS( expr ) INTERNAL_CATCH_THROWS( "CATCH_CHECK_THROWS", Catch::ResultDisposition::ContinueOnFailure, "", expr ) +#define CATCH_CHECK_THROWS_AS( expr, exceptionType ) INTERNAL_CATCH_THROWS_AS( "CATCH_CHECK_THROWS_AS", exceptionType, Catch::ResultDisposition::ContinueOnFailure, expr ) +#define CATCH_CHECK_THROWS_WITH( expr, matcher ) INTERNAL_CATCH_THROWS( "CATCH_CHECK_THROWS_WITH", Catch::ResultDisposition::ContinueOnFailure, matcher, expr ) +#define CATCH_CHECK_NOTHROW( expr ) INTERNAL_CATCH_NO_THROW( "CATCH_CHECK_NOTHROW", Catch::ResultDisposition::ContinueOnFailure, expr ) -#define CATCH_CHECK_THAT( arg, matcher ) INTERNAL_CHECK_THAT( arg, matcher, Catch::ResultDisposition::ContinueOnFailure, "CATCH_CHECK_THAT" ) -#define CATCH_REQUIRE_THAT( arg, matcher ) INTERNAL_CHECK_THAT( arg, matcher, Catch::ResultDisposition::Normal, "CATCH_REQUIRE_THAT" ) +#define CATCH_CHECK_THAT( arg, matcher ) INTERNAL_CHECK_THAT( "CATCH_CHECK_THAT", matcher, Catch::ResultDisposition::ContinueOnFailure, arg ) -#define CATCH_INFO( msg ) INTERNAL_CATCH_INFO( msg, "CATCH_INFO" ) -#define CATCH_WARN( msg ) INTERNAL_CATCH_MSG( Catch::ResultWas::Warning, Catch::ResultDisposition::ContinueOnFailure, "CATCH_WARN", msg ) -#define CATCH_SCOPED_INFO( msg ) INTERNAL_CATCH_INFO( msg, "CATCH_INFO" ) -#define CATCH_CAPTURE( msg ) INTERNAL_CATCH_INFO( #msg " := " << msg, "CATCH_CAPTURE" ) -#define CATCH_SCOPED_CAPTURE( msg ) INTERNAL_CATCH_INFO( #msg " := " << msg, "CATCH_CAPTURE" ) +#if defined(CATCH_CONFIG_FAST_COMPILE) +#define CATCH_REQUIRE_THAT( arg, matcher ) INTERNAL_CHECK_THAT_NO_TRY( "CATCH_REQUIRE_THAT", matcher, Catch::ResultDisposition::Normal, arg ) +#else +#define CATCH_REQUIRE_THAT( arg, matcher ) INTERNAL_CHECK_THAT( "CATCH_REQUIRE_THAT", matcher, Catch::ResultDisposition::Normal, arg ) +#endif + +#define CATCH_INFO( msg ) INTERNAL_CATCH_INFO( "CATCH_INFO", msg ) +#define CATCH_WARN( msg ) INTERNAL_CATCH_MSG( "CATCH_WARN", Catch::ResultWas::Warning, Catch::ResultDisposition::ContinueOnFailure, msg ) +#define CATCH_SCOPED_INFO( msg ) INTERNAL_CATCH_INFO( "CATCH_INFO", msg ) +#define CATCH_CAPTURE( msg ) INTERNAL_CATCH_INFO( "CATCH_CAPTURE", #msg " := " << Catch::toString(msg) ) +#define CATCH_SCOPED_CAPTURE( msg ) INTERNAL_CATCH_INFO( "CATCH_CAPTURE", #msg " := " << Catch::toString(msg) ) #ifdef CATCH_CONFIG_VARIADIC_MACROS #define CATCH_TEST_CASE( ... ) INTERNAL_CATCH_TESTCASE( __VA_ARGS__ ) @@ -11171,16 +11375,18 @@ int main (int argc, char * const argv[]) { #define CATCH_METHOD_AS_TEST_CASE( method, ... ) INTERNAL_CATCH_METHOD_AS_TEST_CASE( method, __VA_ARGS__ ) #define CATCH_REGISTER_TEST_CASE( Function, ... ) INTERNAL_CATCH_REGISTER_TESTCASE( Function, __VA_ARGS__ ) #define CATCH_SECTION( ... ) INTERNAL_CATCH_SECTION( __VA_ARGS__ ) - #define CATCH_FAIL( ... ) INTERNAL_CATCH_MSG( Catch::ResultWas::ExplicitFailure, Catch::ResultDisposition::Normal, "CATCH_FAIL", __VA_ARGS__ ) - #define CATCH_SUCCEED( ... ) INTERNAL_CATCH_MSG( Catch::ResultWas::Ok, Catch::ResultDisposition::ContinueOnFailure, "CATCH_SUCCEED", __VA_ARGS__ ) + #define CATCH_FAIL( ... ) INTERNAL_CATCH_MSG( "CATCH_FAIL", Catch::ResultWas::ExplicitFailure, Catch::ResultDisposition::Normal, __VA_ARGS__ ) + #define CATCH_FAIL_CHECK( ... ) INTERNAL_CATCH_MSG( "CATCH_FAIL_CHECK", Catch::ResultWas::ExplicitFailure, Catch::ResultDisposition::ContinueOnFailure, __VA_ARGS__ ) + #define CATCH_SUCCEED( ... ) INTERNAL_CATCH_MSG( "CATCH_SUCCEED", Catch::ResultWas::Ok, Catch::ResultDisposition::ContinueOnFailure, __VA_ARGS__ ) #else #define CATCH_TEST_CASE( name, description ) INTERNAL_CATCH_TESTCASE( name, description ) #define CATCH_TEST_CASE_METHOD( className, name, description ) INTERNAL_CATCH_TEST_CASE_METHOD( className, name, description ) #define CATCH_METHOD_AS_TEST_CASE( method, name, description ) INTERNAL_CATCH_METHOD_AS_TEST_CASE( method, name, description ) #define CATCH_REGISTER_TEST_CASE( function, name, description ) INTERNAL_CATCH_REGISTER_TESTCASE( function, name, description ) #define CATCH_SECTION( name, description ) INTERNAL_CATCH_SECTION( name, description ) - #define CATCH_FAIL( msg ) INTERNAL_CATCH_MSG( Catch::ResultWas::ExplicitFailure, Catch::ResultDisposition::Normal, "CATCH_FAIL", msg ) - #define CATCH_SUCCEED( msg ) INTERNAL_CATCH_MSG( Catch::ResultWas::Ok, Catch::ResultDisposition::ContinueOnFailure, "CATCH_SUCCEED", msg ) + #define CATCH_FAIL( msg ) INTERNAL_CATCH_MSG( "CATCH_FAIL", Catch::ResultWas::ExplicitFailure, Catch::ResultDisposition::Normal, msg ) + #define CATCH_FAIL_CHECK( msg ) INTERNAL_CATCH_MSG( "CATCH_FAIL_CHECK", Catch::ResultWas::ExplicitFailure, Catch::ResultDisposition::ContinueOnFailure, msg ) + #define CATCH_SUCCEED( msg ) INTERNAL_CATCH_MSG( "CATCH_SUCCEED", Catch::ResultWas::Ok, Catch::ResultDisposition::ContinueOnFailure, msg ) #endif #define CATCH_ANON_TEST_CASE() INTERNAL_CATCH_TESTCASE( "", "" ) @@ -11206,50 +11412,63 @@ int main (int argc, char * const argv[]) { // If CATCH_CONFIG_PREFIX_ALL is not defined then the CATCH_ prefix is not required #else -#define REQUIRE( expr ) INTERNAL_CATCH_TEST( expr, Catch::ResultDisposition::Normal, "REQUIRE" ) -#define REQUIRE_FALSE( expr ) INTERNAL_CATCH_TEST( expr, Catch::ResultDisposition::Normal | Catch::ResultDisposition::FalseTest, "REQUIRE_FALSE" ) +#if defined(CATCH_CONFIG_FAST_COMPILE) +#define REQUIRE( expr ) INTERNAL_CATCH_TEST_NO_TRY( "REQUIRE", Catch::ResultDisposition::Normal, expr ) +#define REQUIRE_FALSE( expr ) INTERNAL_CATCH_TEST_NO_TRY( "REQUIRE_FALSE", Catch::ResultDisposition::Normal | Catch::ResultDisposition::FalseTest, expr ) -#define REQUIRE_THROWS( expr ) INTERNAL_CATCH_THROWS( expr, Catch::ResultDisposition::Normal, "", "REQUIRE_THROWS" ) -#define REQUIRE_THROWS_AS( expr, exceptionType ) INTERNAL_CATCH_THROWS_AS( expr, exceptionType, Catch::ResultDisposition::Normal, "REQUIRE_THROWS_AS" ) -#define REQUIRE_THROWS_WITH( expr, matcher ) INTERNAL_CATCH_THROWS( expr, Catch::ResultDisposition::Normal, matcher, "REQUIRE_THROWS_WITH" ) -#define REQUIRE_NOTHROW( expr ) INTERNAL_CATCH_NO_THROW( expr, Catch::ResultDisposition::Normal, "REQUIRE_NOTHROW" ) +#else +#define REQUIRE( expr ) INTERNAL_CATCH_TEST( "REQUIRE", Catch::ResultDisposition::Normal, expr ) +#define REQUIRE_FALSE( expr ) INTERNAL_CATCH_TEST( "REQUIRE_FALSE", Catch::ResultDisposition::Normal | Catch::ResultDisposition::FalseTest, expr ) +#endif -#define CHECK( expr ) INTERNAL_CATCH_TEST( expr, Catch::ResultDisposition::ContinueOnFailure, "CHECK" ) -#define CHECK_FALSE( expr ) INTERNAL_CATCH_TEST( expr, Catch::ResultDisposition::ContinueOnFailure | Catch::ResultDisposition::FalseTest, "CHECK_FALSE" ) -#define CHECKED_IF( expr ) INTERNAL_CATCH_IF( expr, Catch::ResultDisposition::ContinueOnFailure, "CHECKED_IF" ) -#define CHECKED_ELSE( expr ) INTERNAL_CATCH_ELSE( expr, Catch::ResultDisposition::ContinueOnFailure, "CHECKED_ELSE" ) -#define CHECK_NOFAIL( expr ) INTERNAL_CATCH_TEST( expr, Catch::ResultDisposition::ContinueOnFailure | Catch::ResultDisposition::SuppressFail, "CHECK_NOFAIL" ) +#define REQUIRE_THROWS( expr ) INTERNAL_CATCH_THROWS( "REQUIRE_THROWS", Catch::ResultDisposition::Normal, "", expr ) +#define REQUIRE_THROWS_AS( expr, exceptionType ) INTERNAL_CATCH_THROWS_AS( "REQUIRE_THROWS_AS", exceptionType, Catch::ResultDisposition::Normal, expr ) +#define REQUIRE_THROWS_WITH( expr, matcher ) INTERNAL_CATCH_THROWS( "REQUIRE_THROWS_WITH", Catch::ResultDisposition::Normal, matcher, expr ) +#define REQUIRE_NOTHROW( expr ) INTERNAL_CATCH_NO_THROW( "REQUIRE_NOTHROW", Catch::ResultDisposition::Normal, expr ) -#define CHECK_THROWS( expr ) INTERNAL_CATCH_THROWS( expr, Catch::ResultDisposition::ContinueOnFailure, "", "CHECK_THROWS" ) -#define CHECK_THROWS_AS( expr, exceptionType ) INTERNAL_CATCH_THROWS_AS( expr, exceptionType, Catch::ResultDisposition::ContinueOnFailure, "CHECK_THROWS_AS" ) -#define CHECK_THROWS_WITH( expr, matcher ) INTERNAL_CATCH_THROWS( expr, Catch::ResultDisposition::ContinueOnFailure, matcher, "CHECK_THROWS_WITH" ) -#define CHECK_NOTHROW( expr ) INTERNAL_CATCH_NO_THROW( expr, Catch::ResultDisposition::ContinueOnFailure, "CHECK_NOTHROW" ) +#define CHECK( expr ) INTERNAL_CATCH_TEST( "CHECK", Catch::ResultDisposition::ContinueOnFailure, expr ) +#define CHECK_FALSE( expr ) INTERNAL_CATCH_TEST( "CHECK_FALSE", Catch::ResultDisposition::ContinueOnFailure | Catch::ResultDisposition::FalseTest, expr ) +#define CHECKED_IF( expr ) INTERNAL_CATCH_IF( "CHECKED_IF", Catch::ResultDisposition::ContinueOnFailure, expr ) +#define CHECKED_ELSE( expr ) INTERNAL_CATCH_ELSE( "CHECKED_ELSE", Catch::ResultDisposition::ContinueOnFailure, expr ) +#define CHECK_NOFAIL( expr ) INTERNAL_CATCH_TEST( "CHECK_NOFAIL", Catch::ResultDisposition::ContinueOnFailure | Catch::ResultDisposition::SuppressFail, expr ) -#define CHECK_THAT( arg, matcher ) INTERNAL_CHECK_THAT( arg, matcher, Catch::ResultDisposition::ContinueOnFailure, "CHECK_THAT" ) -#define REQUIRE_THAT( arg, matcher ) INTERNAL_CHECK_THAT( arg, matcher, Catch::ResultDisposition::Normal, "REQUIRE_THAT" ) +#define CHECK_THROWS( expr ) INTERNAL_CATCH_THROWS( "CHECK_THROWS", Catch::ResultDisposition::ContinueOnFailure, "", expr ) +#define CHECK_THROWS_AS( expr, exceptionType ) INTERNAL_CATCH_THROWS_AS( "CHECK_THROWS_AS", exceptionType, Catch::ResultDisposition::ContinueOnFailure, expr ) +#define CHECK_THROWS_WITH( expr, matcher ) INTERNAL_CATCH_THROWS( "CHECK_THROWS_WITH", Catch::ResultDisposition::ContinueOnFailure, matcher, expr ) +#define CHECK_NOTHROW( expr ) INTERNAL_CATCH_NO_THROW( "CHECK_NOTHROW", Catch::ResultDisposition::ContinueOnFailure, expr ) -#define INFO( msg ) INTERNAL_CATCH_INFO( msg, "INFO" ) -#define WARN( msg ) INTERNAL_CATCH_MSG( Catch::ResultWas::Warning, Catch::ResultDisposition::ContinueOnFailure, "WARN", msg ) -#define SCOPED_INFO( msg ) INTERNAL_CATCH_INFO( msg, "INFO" ) -#define CAPTURE( msg ) INTERNAL_CATCH_INFO( #msg " := " << msg, "CAPTURE" ) -#define SCOPED_CAPTURE( msg ) INTERNAL_CATCH_INFO( #msg " := " << msg, "CAPTURE" ) +#define CHECK_THAT( arg, matcher ) INTERNAL_CHECK_THAT( "CHECK_THAT", matcher, Catch::ResultDisposition::ContinueOnFailure, arg ) + +#if defined(CATCH_CONFIG_FAST_COMPILE) +#define REQUIRE_THAT( arg, matcher ) INTERNAL_CHECK_THAT_NO_TRY( "REQUIRE_THAT", matcher, Catch::ResultDisposition::Normal, arg ) +#else +#define REQUIRE_THAT( arg, matcher ) INTERNAL_CHECK_THAT( "REQUIRE_THAT", matcher, Catch::ResultDisposition::Normal, arg ) +#endif + +#define INFO( msg ) INTERNAL_CATCH_INFO( "INFO", msg ) +#define WARN( msg ) INTERNAL_CATCH_MSG( "WARN", Catch::ResultWas::Warning, Catch::ResultDisposition::ContinueOnFailure, msg ) +#define SCOPED_INFO( msg ) INTERNAL_CATCH_INFO( "INFO", msg ) +#define CAPTURE( msg ) INTERNAL_CATCH_INFO( "CAPTURE", #msg " := " << Catch::toString(msg) ) +#define SCOPED_CAPTURE( msg ) INTERNAL_CATCH_INFO( "CAPTURE", #msg " := " << Catch::toString(msg) ) #ifdef CATCH_CONFIG_VARIADIC_MACROS - #define TEST_CASE( ... ) INTERNAL_CATCH_TESTCASE( __VA_ARGS__ ) - #define TEST_CASE_METHOD( className, ... ) INTERNAL_CATCH_TEST_CASE_METHOD( className, __VA_ARGS__ ) - #define METHOD_AS_TEST_CASE( method, ... ) INTERNAL_CATCH_METHOD_AS_TEST_CASE( method, __VA_ARGS__ ) - #define REGISTER_TEST_CASE( Function, ... ) INTERNAL_CATCH_REGISTER_TESTCASE( Function, __VA_ARGS__ ) - #define SECTION( ... ) INTERNAL_CATCH_SECTION( __VA_ARGS__ ) - #define FAIL( ... ) INTERNAL_CATCH_MSG( Catch::ResultWas::ExplicitFailure, Catch::ResultDisposition::Normal, "FAIL", __VA_ARGS__ ) - #define SUCCEED( ... ) INTERNAL_CATCH_MSG( Catch::ResultWas::Ok, Catch::ResultDisposition::ContinueOnFailure, "SUCCEED", __VA_ARGS__ ) +#define TEST_CASE( ... ) INTERNAL_CATCH_TESTCASE( __VA_ARGS__ ) +#define TEST_CASE_METHOD( className, ... ) INTERNAL_CATCH_TEST_CASE_METHOD( className, __VA_ARGS__ ) +#define METHOD_AS_TEST_CASE( method, ... ) INTERNAL_CATCH_METHOD_AS_TEST_CASE( method, __VA_ARGS__ ) +#define REGISTER_TEST_CASE( Function, ... ) INTERNAL_CATCH_REGISTER_TESTCASE( Function, __VA_ARGS__ ) +#define SECTION( ... ) INTERNAL_CATCH_SECTION( __VA_ARGS__ ) +#define FAIL( ... ) INTERNAL_CATCH_MSG( "FAIL", Catch::ResultWas::ExplicitFailure, Catch::ResultDisposition::Normal, __VA_ARGS__ ) +#define FAIL_CHECK( ... ) INTERNAL_CATCH_MSG( "FAIL_CHECK", Catch::ResultWas::ExplicitFailure, Catch::ResultDisposition::ContinueOnFailure, __VA_ARGS__ ) +#define SUCCEED( ... ) INTERNAL_CATCH_MSG( "SUCCEED", Catch::ResultWas::Ok, Catch::ResultDisposition::ContinueOnFailure, __VA_ARGS__ ) #else - #define TEST_CASE( name, description ) INTERNAL_CATCH_TESTCASE( name, description ) +#define TEST_CASE( name, description ) INTERNAL_CATCH_TESTCASE( name, description ) #define TEST_CASE_METHOD( className, name, description ) INTERNAL_CATCH_TEST_CASE_METHOD( className, name, description ) #define METHOD_AS_TEST_CASE( method, name, description ) INTERNAL_CATCH_METHOD_AS_TEST_CASE( method, name, description ) #define REGISTER_TEST_CASE( method, name, description ) INTERNAL_CATCH_REGISTER_TESTCASE( method, name, description ) #define SECTION( name, description ) INTERNAL_CATCH_SECTION( name, description ) - #define FAIL( msg ) INTERNAL_CATCH_MSG( Catch::ResultWas::ExplicitFailure, Catch::ResultDisposition::Normal, "FAIL", msg ) - #define SUCCEED( msg ) INTERNAL_CATCH_MSG( Catch::ResultWas::Ok, Catch::ResultDisposition::ContinueOnFailure, "SUCCEED", msg ) + #define FAIL( msg ) INTERNAL_CATCH_MSG( "FAIL", Catch::ResultWas::ExplicitFailure, Catch::ResultDisposition::Normal, msg ) + #define FAIL_CHECK( msg ) INTERNAL_CATCH_MSG( "FAIL_CHECK", Catch::ResultWas::ExplicitFailure, Catch::ResultDisposition::ContinueOnFailure, msg ) + #define SUCCEED( msg ) INTERNAL_CATCH_MSG( "SUCCEED", Catch::ResultWas::Ok, Catch::ResultDisposition::ContinueOnFailure, msg ) #endif #define ANON_TEST_CASE() INTERNAL_CATCH_TESTCASE( "", "" ) @@ -11278,5 +11497,19 @@ int main (int argc, char * const argv[]) { using Catch::Detail::Approx; +// #included from: internal/catch_reenable_warnings.h + +#define TWOBLUECUBES_CATCH_REENABLE_WARNINGS_H_INCLUDED + +#ifdef __clang__ +# ifdef __ICC // icpc defines the __clang__ macro +# pragma warning(pop) +# else +# pragma clang diagnostic pop +# endif +#elif defined __GNUC__ +# pragma GCC diagnostic pop +#endif + #endif // TWOBLUECUBES_SINGLE_INCLUDE_CATCH_HPP_INCLUDED From 82b95ca664ebc33c3a704156286179e493e6a6a1 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 20 Jun 2017 20:14:18 +0200 Subject: [PATCH 248/530] :hammer: simplified error handling in parser --- src/json.hpp | 60 +++++++++--------- test/src/unit-class_parser.cpp | 112 ++++++++++++++++----------------- 2 files changed, 87 insertions(+), 85 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index b4dc422d..13a31bda 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -12902,7 +12902,7 @@ scan_number_done: default: { // the last token was unexpected - unexpect(last_token); + unexpect(); } } @@ -13040,51 +13040,49 @@ scan_number_done: /// get next token from lexer typename lexer::token_type get_token() { - last_token = m_lexer.scan(); - return last_token; + return (last_token = m_lexer.scan()); } /*! @throw parse_error.101 if expected token did not occur */ - void expect(typename lexer::token_type t) const + void expect(typename lexer::token_type t) { if (JSON_UNLIKELY(t != last_token)) { - std::string error_msg = "syntax error - "; - if (last_token == lexer::token_type::parse_error) - { - error_msg += std::string(m_lexer.get_error_message()) + "; last read: '" + m_lexer.get_token_string() + "'"; - } - else - { - error_msg += "unexpected " + std::string(lexer::token_type_name(last_token)); - } - - error_msg += "; expected " + std::string(lexer::token_type_name(t)); - JSON_THROW(parse_error::create(101, m_lexer.get_position(), error_msg)); + errored = true; + expected = t; + throw_exception(); } } /*! @throw parse_error.101 if unexpected token occurred */ - void unexpect(typename lexer::token_type t) const + void unexpect() { - if (JSON_UNLIKELY(t == last_token)) - { - std::string error_msg = "syntax error - "; - if (last_token == lexer::token_type::parse_error) - { - error_msg += std::string(m_lexer.get_error_message()) + "; last read '" + m_lexer.get_token_string() + "'"; - } - else - { - error_msg += "unexpected " + std::string(lexer::token_type_name(last_token)); - } + errored = true; + throw_exception(); + } - JSON_THROW(parse_error::create(101, m_lexer.get_position(), error_msg)); + [[noreturn]] void throw_exception() const + { + std::string error_msg = "syntax error - "; + if (last_token == lexer::token_type::parse_error) + { + error_msg += std::string(m_lexer.get_error_message()) + "; last read: '" + m_lexer.get_token_string() + "'"; } + else + { + error_msg += "unexpected " + std::string(lexer::token_type_name(last_token)); + } + + if (expected != lexer::token_type::uninitialized) + { + error_msg += "; expected " + std::string(lexer::token_type_name(expected)); + } + + JSON_THROW(parse_error::create(101, m_lexer.get_position(), error_msg)); } private: @@ -13096,6 +13094,10 @@ scan_number_done: typename lexer::token_type last_token = lexer::token_type::uninitialized; /// the lexer lexer m_lexer; + /// whether a syntax error occurred + bool errored = false; + /// possible reason for the syntax error + typename lexer::token_type expected = lexer::token_type::uninitialized; }; public: diff --git a/test/src/unit-class_parser.cpp b/test/src/unit-class_parser.cpp index 920054d2..3a71dfa4 100644 --- a/test/src/unit-class_parser.cpp +++ b/test/src/unit-class_parser.cpp @@ -98,18 +98,18 @@ TEST_CASE("parser class") // error: tab in string CHECK_THROWS_AS(parse_string("\"\t\"").parse(), json::parse_error); CHECK_THROWS_WITH(parse_string("\"\t\"").parse(), - "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: control character must be escaped; last read '\"'"); + "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: control character must be escaped; last read: '\"'"); // error: newline in string CHECK_THROWS_AS(parse_string("\"\n\"").parse(), json::parse_error); CHECK_THROWS_AS(parse_string("\"\r\"").parse(), json::parse_error); CHECK_THROWS_WITH(parse_string("\"\n\"").parse(), - "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: control character must be escaped; last read '\"'"); + "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: control character must be escaped; last read: '\"'"); CHECK_THROWS_WITH(parse_string("\"\r\"").parse(), - "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: control character must be escaped; last read '\"'"); + "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: control character must be escaped; last read: '\"'"); // error: backspace in string CHECK_THROWS_AS(parse_string("\"\b\"").parse(), json::parse_error); CHECK_THROWS_WITH(parse_string("\"\b\"").parse(), - "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: control character must be escaped; last read '\"'"); + "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: control character must be escaped; last read: '\"'"); // improve code coverage CHECK_THROWS_AS(parse_string("\uFF01").parse(), json::parse_error); CHECK_THROWS_AS(parse_string("[-4:1,]").parse(), json::parse_error); @@ -317,21 +317,21 @@ TEST_CASE("parser class") CHECK_THROWS_WITH(parse_string("-01").parse(), "[json.exception.parse_error.101] parse error at 3: syntax error - unexpected number literal; expected end of input"); CHECK_THROWS_WITH(parse_string("--1").parse(), - "[json.exception.parse_error.101] parse error at 2: syntax error - invalid number; expected digit after '-'; last read '--'"); + "[json.exception.parse_error.101] parse error at 2: syntax error - invalid number; expected digit after '-'; last read: '--'"); CHECK_THROWS_WITH(parse_string("1.").parse(), - "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected digit after '.'; last read '1.'"); + "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected digit after '.'; last read: '1.'"); CHECK_THROWS_WITH(parse_string("1E").parse(), - "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected '+', '-', or digit after exponent; last read '1E'"); + "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected '+', '-', or digit after exponent; last read: '1E'"); CHECK_THROWS_WITH(parse_string("1E-").parse(), - "[json.exception.parse_error.101] parse error at 4: syntax error - invalid number; expected digit after exponent sign; last read '1E-'"); + "[json.exception.parse_error.101] parse error at 4: syntax error - invalid number; expected digit after exponent sign; last read: '1E-'"); CHECK_THROWS_WITH(parse_string("1.E1").parse(), - "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected digit after '.'; last read '1.E'"); + "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected digit after '.'; last read: '1.E'"); CHECK_THROWS_WITH(parse_string("-1E").parse(), - "[json.exception.parse_error.101] parse error at 4: syntax error - invalid number; expected '+', '-', or digit after exponent; last read '-1E'"); + "[json.exception.parse_error.101] parse error at 4: syntax error - invalid number; expected '+', '-', or digit after exponent; last read: '-1E'"); CHECK_THROWS_WITH(parse_string("-0E#").parse(), - "[json.exception.parse_error.101] parse error at 4: syntax error - invalid number; expected '+', '-', or digit after exponent; last read '-0E#'"); + "[json.exception.parse_error.101] parse error at 4: syntax error - invalid number; expected '+', '-', or digit after exponent; last read: '-0E#'"); CHECK_THROWS_WITH(parse_string("-0E-#").parse(), - "[json.exception.parse_error.101] parse error at 5: syntax error - invalid number; expected digit after exponent sign; last read '-0E-#'"); + "[json.exception.parse_error.101] parse error at 5: syntax error - invalid number; expected digit after exponent sign; last read: '-0E-#'"); CHECK_THROWS_WITH(parse_string("-0#").parse(), "[json.exception.parse_error.101] parse error at 3: syntax error - invalid literal; last read: '-0#'; expected end of input"); CHECK_THROWS_WITH(parse_string("-0.0:").parse(), @@ -343,7 +343,7 @@ TEST_CASE("parser class") CHECK_THROWS_WITH(parse_string("-0e0-:").parse(), "[json.exception.parse_error.101] parse error at 6: syntax error - invalid number; expected digit after '-'; last read: '-:'; expected end of input"); CHECK_THROWS_WITH(parse_string("-0e-:").parse(), - "[json.exception.parse_error.101] parse error at 5: syntax error - invalid number; expected digit after exponent sign; last read '-0e-:'"); + "[json.exception.parse_error.101] parse error at 5: syntax error - invalid number; expected digit after exponent sign; last read: '-0e-:'"); CHECK_THROWS_WITH(parse_string("-0f").parse(), "[json.exception.parse_error.101] parse error at 4: syntax error - invalid literal; expected 'false'; last read: '-0f'; expected end of input"); } @@ -630,55 +630,55 @@ TEST_CASE("parser class") CHECK_THROWS_AS(parse_string("1E/").parse(), json::parse_error); CHECK_THROWS_AS(parse_string("1E:").parse(), json::parse_error); CHECK_THROWS_WITH(parse_string("0.").parse(), - "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected digit after '.'; last read '0.'"); + "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected digit after '.'; last read: '0.'"); CHECK_THROWS_WITH(parse_string("-").parse(), - "[json.exception.parse_error.101] parse error at 2: syntax error - invalid number; expected digit after '-'; last read '-'"); + "[json.exception.parse_error.101] parse error at 2: syntax error - invalid number; expected digit after '-'; last read: '-'"); CHECK_THROWS_WITH(parse_string("--").parse(), - "[json.exception.parse_error.101] parse error at 2: syntax error - invalid number; expected digit after '-'; last read '--'"); + "[json.exception.parse_error.101] parse error at 2: syntax error - invalid number; expected digit after '-'; last read: '--'"); CHECK_THROWS_WITH(parse_string("-0.").parse(), - "[json.exception.parse_error.101] parse error at 4: syntax error - invalid number; expected digit after '.'; last read '-0.'"); + "[json.exception.parse_error.101] parse error at 4: syntax error - invalid number; expected digit after '.'; last read: '-0.'"); CHECK_THROWS_WITH(parse_string("-.").parse(), - "[json.exception.parse_error.101] parse error at 2: syntax error - invalid number; expected digit after '-'; last read '-.'"); + "[json.exception.parse_error.101] parse error at 2: syntax error - invalid number; expected digit after '-'; last read: '-.'"); CHECK_THROWS_WITH(parse_string("-:").parse(), - "[json.exception.parse_error.101] parse error at 2: syntax error - invalid number; expected digit after '-'; last read '-:'"); + "[json.exception.parse_error.101] parse error at 2: syntax error - invalid number; expected digit after '-'; last read: '-:'"); CHECK_THROWS_WITH(parse_string("0.:").parse(), - "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected digit after '.'; last read '0.:'"); + "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected digit after '.'; last read: '0.:'"); CHECK_THROWS_WITH(parse_string("e.").parse(), - "[json.exception.parse_error.101] parse error at 1: syntax error - invalid literal; last read 'e'"); + "[json.exception.parse_error.101] parse error at 1: syntax error - invalid literal; last read: 'e'"); CHECK_THROWS_WITH(parse_string("1e.").parse(), - "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected '+', '-', or digit after exponent; last read '1e.'"); + "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected '+', '-', or digit after exponent; last read: '1e.'"); CHECK_THROWS_WITH(parse_string("1e/").parse(), - "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected '+', '-', or digit after exponent; last read '1e/'"); + "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected '+', '-', or digit after exponent; last read: '1e/'"); CHECK_THROWS_WITH(parse_string("1e:").parse(), - "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected '+', '-', or digit after exponent; last read '1e:'"); + "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected '+', '-', or digit after exponent; last read: '1e:'"); CHECK_THROWS_WITH(parse_string("1E.").parse(), - "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected '+', '-', or digit after exponent; last read '1E.'"); + "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected '+', '-', or digit after exponent; last read: '1E.'"); CHECK_THROWS_WITH(parse_string("1E/").parse(), - "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected '+', '-', or digit after exponent; last read '1E/'"); + "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected '+', '-', or digit after exponent; last read: '1E/'"); CHECK_THROWS_WITH(parse_string("1E:").parse(), - "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected '+', '-', or digit after exponent; last read '1E:'"); + "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected '+', '-', or digit after exponent; last read: '1E:'"); // unexpected end of null CHECK_THROWS_AS(parse_string("n").parse(), json::parse_error); CHECK_THROWS_AS(parse_string("nu").parse(), json::parse_error); CHECK_THROWS_AS(parse_string("nul").parse(), json::parse_error); CHECK_THROWS_WITH(parse_string("n").parse(), - "[json.exception.parse_error.101] parse error at 2: syntax error - invalid literal; expected 'null'; last read 'n'"); + "[json.exception.parse_error.101] parse error at 2: syntax error - invalid literal; expected 'null'; last read: 'n'"); CHECK_THROWS_WITH(parse_string("nu").parse(), - "[json.exception.parse_error.101] parse error at 3: syntax error - invalid literal; expected 'null'; last read 'nu'"); + "[json.exception.parse_error.101] parse error at 3: syntax error - invalid literal; expected 'null'; last read: 'nu'"); CHECK_THROWS_WITH(parse_string("nul").parse(), - "[json.exception.parse_error.101] parse error at 4: syntax error - invalid literal; expected 'null'; last read 'nul'"); + "[json.exception.parse_error.101] parse error at 4: syntax error - invalid literal; expected 'null'; last read: 'nul'"); // unexpected end of true CHECK_THROWS_AS(parse_string("t").parse(), json::parse_error); CHECK_THROWS_AS(parse_string("tr").parse(), json::parse_error); CHECK_THROWS_AS(parse_string("tru").parse(), json::parse_error); CHECK_THROWS_WITH(parse_string("t").parse(), - "[json.exception.parse_error.101] parse error at 2: syntax error - invalid literal; expected 'true'; last read 't'"); + "[json.exception.parse_error.101] parse error at 2: syntax error - invalid literal; expected 'true'; last read: 't'"); CHECK_THROWS_WITH(parse_string("tr").parse(), - "[json.exception.parse_error.101] parse error at 3: syntax error - invalid literal; expected 'true'; last read 'tr'"); + "[json.exception.parse_error.101] parse error at 3: syntax error - invalid literal; expected 'true'; last read: 'tr'"); CHECK_THROWS_WITH(parse_string("tru").parse(), - "[json.exception.parse_error.101] parse error at 4: syntax error - invalid literal; expected 'true'; last read 'tru'"); + "[json.exception.parse_error.101] parse error at 4: syntax error - invalid literal; expected 'true'; last read: 'tru'"); // unexpected end of false CHECK_THROWS_AS(parse_string("f").parse(), json::parse_error); @@ -686,13 +686,13 @@ TEST_CASE("parser class") CHECK_THROWS_AS(parse_string("fal").parse(), json::parse_error); CHECK_THROWS_AS(parse_string("fals").parse(), json::parse_error); CHECK_THROWS_WITH(parse_string("f").parse(), - "[json.exception.parse_error.101] parse error at 2: syntax error - invalid literal; expected 'false'; last read 'f'"); + "[json.exception.parse_error.101] parse error at 2: syntax error - invalid literal; expected 'false'; last read: 'f'"); CHECK_THROWS_WITH(parse_string("fa").parse(), - "[json.exception.parse_error.101] parse error at 3: syntax error - invalid literal; expected 'false'; last read 'fa'"); + "[json.exception.parse_error.101] parse error at 3: syntax error - invalid literal; expected 'false'; last read: 'fa'"); CHECK_THROWS_WITH(parse_string("fal").parse(), - "[json.exception.parse_error.101] parse error at 4: syntax error - invalid literal; expected 'false'; last read 'fal'"); + "[json.exception.parse_error.101] parse error at 4: syntax error - invalid literal; expected 'false'; last read: 'fal'"); CHECK_THROWS_WITH(parse_string("fals").parse(), - "[json.exception.parse_error.101] parse error at 5: syntax error - invalid literal; expected 'false'; last read 'fals'"); + "[json.exception.parse_error.101] parse error at 5: syntax error - invalid literal; expected 'false'; last read: 'fals'"); // missing/unexpected end of array CHECK_THROWS_AS(parse_string("[").parse(), json::parse_error); @@ -743,25 +743,25 @@ TEST_CASE("parser class") CHECK_THROWS_AS(parse_string("\"\\u01").parse(), json::parse_error); CHECK_THROWS_AS(parse_string("\"\\u012").parse(), json::parse_error); CHECK_THROWS_WITH(parse_string("\"").parse(), - "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: missing closing quote; last read '\"'"); + "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: missing closing quote; last read: '\"'"); CHECK_THROWS_WITH(parse_string("\"\\\"").parse(), - "[json.exception.parse_error.101] parse error at 4: syntax error - invalid string: missing closing quote; last read '\"\\\"'"); + "[json.exception.parse_error.101] parse error at 4: syntax error - invalid string: missing closing quote; last read: '\"\\\"'"); CHECK_THROWS_WITH(parse_string("\"\\u\"").parse(), - "[json.exception.parse_error.101] parse error at 4: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read '\"\\u\"'"); + "[json.exception.parse_error.101] parse error at 4: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read: '\"\\u\"'"); CHECK_THROWS_WITH(parse_string("\"\\u0\"").parse(), - "[json.exception.parse_error.101] parse error at 5: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read '\"\\u0\"'"); + "[json.exception.parse_error.101] parse error at 5: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read: '\"\\u0\"'"); CHECK_THROWS_WITH(parse_string("\"\\u01\"").parse(), - "[json.exception.parse_error.101] parse error at 6: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read '\"\\u01\"'"); + "[json.exception.parse_error.101] parse error at 6: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read: '\"\\u01\"'"); CHECK_THROWS_WITH(parse_string("\"\\u012\"").parse(), - "[json.exception.parse_error.101] parse error at 7: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read '\"\\u012\"'"); + "[json.exception.parse_error.101] parse error at 7: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read: '\"\\u012\"'"); CHECK_THROWS_WITH(parse_string("\"\\u").parse(), - "[json.exception.parse_error.101] parse error at 4: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read '\"\\u'"); + "[json.exception.parse_error.101] parse error at 4: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read: '\"\\u'"); CHECK_THROWS_WITH(parse_string("\"\\u0").parse(), - "[json.exception.parse_error.101] parse error at 5: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read '\"\\u0'"); + "[json.exception.parse_error.101] parse error at 5: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read: '\"\\u0'"); CHECK_THROWS_WITH(parse_string("\"\\u01").parse(), - "[json.exception.parse_error.101] parse error at 6: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read '\"\\u01'"); + "[json.exception.parse_error.101] parse error at 6: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read: '\"\\u01'"); CHECK_THROWS_WITH(parse_string("\"\\u012").parse(), - "[json.exception.parse_error.101] parse error at 7: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read '\"\\u012'"); + "[json.exception.parse_error.101] parse error at 7: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read: '\"\\u012'"); // invalid escapes for (int c = 1; c < 128; ++c) @@ -798,7 +798,7 @@ TEST_CASE("parser class") if (c > 0x1f) { CHECK_THROWS_WITH(parse_string(s.c_str()).parse(), - "[json.exception.parse_error.101] parse error at 3: syntax error - invalid string: forbidden character after backslash; last read '\"\\" + std::string(1, static_cast(c)) + "'"); + "[json.exception.parse_error.101] parse error at 3: syntax error - invalid string: forbidden character after backslash; last read: '\"\\" + std::string(1, static_cast(c)) + "'"); } break; } @@ -874,7 +874,7 @@ TEST_CASE("parser class") if (c > 0x1f) { CHECK_THROWS_WITH(parse_string(s1.c_str()).parse(), - "[json.exception.parse_error.101] parse error at 7: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read '" + s1.substr(0, 7) + "'"); + "[json.exception.parse_error.101] parse error at 7: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read: '" + s1.substr(0, 7) + "'"); } CAPTURE(s2); @@ -883,7 +883,7 @@ TEST_CASE("parser class") if (c > 0x1f) { CHECK_THROWS_WITH(parse_string(s2.c_str()).parse(), - "[json.exception.parse_error.101] parse error at 6: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read '" + s2.substr(0, 6) + "'"); + "[json.exception.parse_error.101] parse error at 6: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read: '" + s2.substr(0, 6) + "'"); } CAPTURE(s3); @@ -892,7 +892,7 @@ TEST_CASE("parser class") if (c > 0x1f) { CHECK_THROWS_WITH(parse_string(s3.c_str()).parse(), - "[json.exception.parse_error.101] parse error at 5: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read '" + s3.substr(0, 5) + "'"); + "[json.exception.parse_error.101] parse error at 5: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read: '" + s3.substr(0, 5) + "'"); } CAPTURE(s4); @@ -901,7 +901,7 @@ TEST_CASE("parser class") if (c > 0x1f) { CHECK_THROWS_WITH(parse_string(s4.c_str()).parse(), - "[json.exception.parse_error.101] parse error at 4: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read '" + s4.substr(0, 4) + "'"); + "[json.exception.parse_error.101] parse error at 4: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read: '" + s4.substr(0, 4) + "'"); } } } @@ -910,17 +910,17 @@ TEST_CASE("parser class") // missing part of a surrogate pair CHECK_THROWS_AS(json::parse("\"\\uD80C\""), json::parse_error); CHECK_THROWS_WITH(json::parse("\"\\uD80C\""), - "[json.exception.parse_error.101] parse error at 8: syntax error - invalid string: surrogate U+DC00..U+DFFF must be followed by U+DC00..U+DFFF; last read '\"\\uD80C\"'"); + "[json.exception.parse_error.101] parse error at 8: syntax error - invalid string: surrogate U+DC00..U+DFFF must be followed by U+DC00..U+DFFF; last read: '\"\\uD80C\"'"); // invalid surrogate pair CHECK_THROWS_AS(json::parse("\"\\uD80C\\uD80C\""), json::parse_error); CHECK_THROWS_AS(json::parse("\"\\uD80C\\u0000\""), json::parse_error); CHECK_THROWS_AS(json::parse("\"\\uD80C\\uFFFF\""), json::parse_error); CHECK_THROWS_WITH(json::parse("\"\\uD80C\\uD80C\""), - "[json.exception.parse_error.101] parse error at 13: syntax error - invalid string: surrogate U+DC00..U+DFFF must be followed by U+DC00..U+DFFF; last read '\"\\uD80C\\uD80C'"); + "[json.exception.parse_error.101] parse error at 13: syntax error - invalid string: surrogate U+DC00..U+DFFF must be followed by U+DC00..U+DFFF; last read: '\"\\uD80C\\uD80C'"); CHECK_THROWS_WITH(json::parse("\"\\uD80C\\u0000\""), - "[json.exception.parse_error.101] parse error at 13: syntax error - invalid string: surrogate U+DC00..U+DFFF must be followed by U+DC00..U+DFFF; last read '\"\\uD80C\\u0000'"); + "[json.exception.parse_error.101] parse error at 13: syntax error - invalid string: surrogate U+DC00..U+DFFF must be followed by U+DC00..U+DFFF; last read: '\"\\uD80C\\u0000'"); CHECK_THROWS_WITH(json::parse("\"\\uD80C\\uFFFF\""), - "[json.exception.parse_error.101] parse error at 13: syntax error - invalid string: surrogate U+DC00..U+DFFF must be followed by U+DC00..U+DFFF; last read '\"\\uD80C\\uFFFF'"); + "[json.exception.parse_error.101] parse error at 13: syntax error - invalid string: surrogate U+DC00..U+DFFF must be followed by U+DC00..U+DFFF; last read: '\"\\uD80C\\uFFFF'"); } SECTION("parse errors (accept)") From e191dd833f3e77a048930a87cad54ba02a52c0cd Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 20 Jun 2017 21:09:05 +0200 Subject: [PATCH 249/530] :hammer: fixed warning about pointer members --- src/json.hpp | 12 +- test/src/unit-class_parser.cpp | 909 ++++++++++++++++----------------- 2 files changed, 459 insertions(+), 462 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 13a31bda..596f7136 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -6807,10 +6807,6 @@ class basic_json */ class serializer { - private: - serializer(const serializer&) = delete; - serializer& operator=(const serializer&) = delete; - public: /*! @param[in] s output stream to serialize to @@ -6823,6 +6819,10 @@ class basic_json indent_char(ichar), indent_string(512, indent_char) {} + // delete because of pointer members + serializer(const serializer&) = delete; + serializer& operator=(const serializer&) = delete; + /*! @brief internal implementation of the serialization function @@ -11182,6 +11182,10 @@ class basic_json : ia(adapter), decimal_point_char(get_decimal_point()) {} + // delete because of pointer members + lexer(const lexer&) = delete; + lexer& operator=(lexer&) = delete; + private: ///////////////////// // locales diff --git a/test/src/unit-class_parser.cpp b/test/src/unit-class_parser.cpp index 3a71dfa4..a95d1642 100644 --- a/test/src/unit-class_parser.cpp +++ b/test/src/unit-class_parser.cpp @@ -34,43 +34,36 @@ using nlohmann::json; #include -// shortcut to parse a string literal -json::parser parse_string(const char* s); -json::parser parse_string(const char* s) -{ - return json::parser(json::input_adapter::create(s)); -} - TEST_CASE("parser class") { SECTION("parse") { SECTION("null") { - CHECK(parse_string("null").parse() == json(nullptr)); + CHECK(json::parser(json::input_adapter::create(std::string("null"))).parse() == json(nullptr)); } SECTION("true") { - CHECK(parse_string("true").parse() == json(true)); + CHECK(json::parser(json::input_adapter::create(std::string("true"))).parse() == json(true)); } SECTION("false") { - CHECK(parse_string("false").parse() == json(false)); + CHECK(json::parser(json::input_adapter::create(std::string("false"))).parse() == json(false)); } SECTION("array") { SECTION("empty array") { - CHECK(parse_string("[]").parse() == json(json::value_t::array)); - CHECK(parse_string("[ ]").parse() == json(json::value_t::array)); + CHECK(json::parser(json::input_adapter::create(std::string("[]"))).parse() == json(json::value_t::array)); + CHECK(json::parser(json::input_adapter::create(std::string("[ ]"))).parse() == json(json::value_t::array)); } SECTION("nonempty array") { - CHECK(parse_string("[true, false, null]").parse() == json({true, false, nullptr})); + CHECK(json::parser(json::input_adapter::create(std::string("[true, false, null]"))).parse() == json({true, false, nullptr})); } } @@ -78,113 +71,113 @@ TEST_CASE("parser class") { SECTION("empty object") { - CHECK(parse_string("{}").parse() == json(json::value_t::object)); - CHECK(parse_string("{ }").parse() == json(json::value_t::object)); + CHECK(json::parser(json::input_adapter::create(std::string("{}"))).parse() == json(json::value_t::object)); + CHECK(json::parser(json::input_adapter::create(std::string("{ }"))).parse() == json(json::value_t::object)); } SECTION("nonempty object") { - CHECK(parse_string("{\"\": true, \"one\": 1, \"two\": null}").parse() == json({{"", true}, {"one", 1}, {"two", nullptr}})); + CHECK(json::parser(json::input_adapter::create(std::string("{\"\": true, \"one\": 1, \"two\": null}"))).parse() == json({{"", true}, {"one", 1}, {"two", nullptr}})); } } SECTION("string") { // empty string - CHECK(parse_string("\"\"").parse() == json(json::value_t::string)); + CHECK(json::parser(json::input_adapter::create(std::string("\"\""))).parse() == json(json::value_t::string)); SECTION("errors") { // error: tab in string - CHECK_THROWS_AS(parse_string("\"\t\"").parse(), json::parse_error); - CHECK_THROWS_WITH(parse_string("\"\t\"").parse(), + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\t\""))).parse(), json::parse_error); + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("\"\t\""))).parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: control character must be escaped; last read: '\"'"); // error: newline in string - CHECK_THROWS_AS(parse_string("\"\n\"").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("\"\r\"").parse(), json::parse_error); - CHECK_THROWS_WITH(parse_string("\"\n\"").parse(), + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\n\""))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\r\""))).parse(), json::parse_error); + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("\"\n\""))).parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: control character must be escaped; last read: '\"'"); - CHECK_THROWS_WITH(parse_string("\"\r\"").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("\"\r\""))).parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: control character must be escaped; last read: '\"'"); // error: backspace in string - CHECK_THROWS_AS(parse_string("\"\b\"").parse(), json::parse_error); - CHECK_THROWS_WITH(parse_string("\"\b\"").parse(), + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\b\""))).parse(), json::parse_error); + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("\"\b\""))).parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: control character must be escaped; last read: '\"'"); // improve code coverage - CHECK_THROWS_AS(parse_string("\uFF01").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("[-4:1,]").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\uFF01"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("[-4:1,]"))).parse(), json::parse_error); // unescaped control characters - CHECK_THROWS_AS(parse_string("\"\x00\"").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("\"\x01\"").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("\"\x02\"").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("\"\x03\"").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("\"\x04\"").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("\"\x05\"").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("\"\x06\"").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("\"\x07\"").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("\"\x08\"").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("\"\x09\"").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("\"\x0a\"").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("\"\x0b\"").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("\"\x0c\"").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("\"\x0d\"").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("\"\x0e\"").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("\"\x0f\"").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("\"\x10\"").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("\"\x11\"").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("\"\x12\"").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("\"\x13\"").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("\"\x14\"").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("\"\x15\"").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("\"\x16\"").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("\"\x17\"").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("\"\x18\"").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("\"\x19\"").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("\"\x1a\"").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("\"\x1b\"").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("\"\x1c\"").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("\"\x1d\"").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("\"\x1e\"").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("\"\x1f\"").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x00\""))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x01\""))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x02\""))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x03\""))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x04\""))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x05\""))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x06\""))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x07\""))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x08\""))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x09\""))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x0a\""))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x0b\""))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x0c\""))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x0d\""))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x0e\""))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x0f\""))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x10\""))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x11\""))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x12\""))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x13\""))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x14\""))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x15\""))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x16\""))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x17\""))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x18\""))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x19\""))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x1a\""))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x1b\""))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x1c\""))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x1d\""))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x1e\""))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x1f\""))).parse(), json::parse_error); } SECTION("escaped") { // quotation mark "\"" auto r1 = R"("\"")"_json; - CHECK(parse_string("\"\\\"\"").parse() == r1); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\\"\""))).parse() == r1); // reverse solidus "\\" auto r2 = R"("\\")"_json; - CHECK(parse_string("\"\\\\\"").parse() == r2); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\\\\""))).parse() == r2); // solidus - CHECK(parse_string("\"\\/\"").parse() == R"("/")"_json); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\/\""))).parse() == R"("/")"_json); // backspace - CHECK(parse_string("\"\\b\"").parse() == json("\b")); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\b\""))).parse() == json("\b")); // formfeed - CHECK(parse_string("\"\\f\"").parse() == json("\f")); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\f\""))).parse() == json("\f")); // newline - CHECK(parse_string("\"\\n\"").parse() == json("\n")); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\n\""))).parse() == json("\n")); // carriage return - CHECK(parse_string("\"\\r\"").parse() == json("\r")); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\r\""))).parse() == json("\r")); // horizontal tab - CHECK(parse_string("\"\\t\"").parse() == json("\t")); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\t\""))).parse() == json("\t")); - CHECK(parse_string("\"\\u0001\"").parse().get() == "\x01"); - CHECK(parse_string("\"\\u000a\"").parse().get() == "\n"); - CHECK(parse_string("\"\\u00b0\"").parse().get() == "°"); - CHECK(parse_string("\"\\u0c00\"").parse().get() == "ఀ"); - CHECK(parse_string("\"\\ud000\"").parse().get() == "퀀"); - CHECK(parse_string("\"\\u000E\"").parse().get() == "\x0E"); - CHECK(parse_string("\"\\u00F0\"").parse().get() == "ð"); - CHECK(parse_string("\"\\u0100\"").parse().get() == "Ā"); - CHECK(parse_string("\"\\u2000\"").parse().get() == " "); - CHECK(parse_string("\"\\uFFFF\"").parse().get() == "￿"); - CHECK(parse_string("\"\\u20AC\"").parse().get() == "€"); - CHECK(parse_string("\"€\"").parse().get() == "€"); - CHECK(parse_string("\"🎈\"").parse().get() == "🎈"); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\u0001\""))).parse().get() == "\x01"); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\u000a\""))).parse().get() == "\n"); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\u00b0\""))).parse().get() == "°"); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\u0c00\""))).parse().get() == "ఀ"); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\ud000\""))).parse().get() == "퀀"); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\u000E\""))).parse().get() == "\x0E"); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\u00F0\""))).parse().get() == "ð"); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\u0100\""))).parse().get() == "Ā"); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\u2000\""))).parse().get() == " "); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\uFFFF\""))).parse().get() == "￿"); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\u20AC\""))).parse().get() == "€"); + CHECK(json::parser(json::input_adapter::create(std::string("\"€\""))).parse().get() == "€"); + CHECK(json::parser(json::input_adapter::create(std::string("\"🎈\""))).parse().get() == "🎈"); - CHECK(parse_string("\"\\ud80c\\udc60\"").parse().get() == u8"\U00013060"); - CHECK(parse_string("\"\\ud83c\\udf1e\"").parse().get() == "🌞"); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\ud80c\\udc60\""))).parse().get() == u8"\U00013060"); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\ud83c\\udf1e\""))).parse().get() == "🌞"); } } @@ -194,40 +187,40 @@ TEST_CASE("parser class") { SECTION("without exponent") { - CHECK(parse_string("-128").parse() == json(-128)); - CHECK(parse_string("-0").parse() == json(-0)); - CHECK(parse_string("0").parse() == json(0)); - CHECK(parse_string("128").parse() == json(128)); + CHECK(json::parser(json::input_adapter::create(std::string("-128"))).parse() == json(-128)); + CHECK(json::parser(json::input_adapter::create(std::string("-0"))).parse() == json(-0)); + CHECK(json::parser(json::input_adapter::create(std::string("0"))).parse() == json(0)); + CHECK(json::parser(json::input_adapter::create(std::string("128"))).parse() == json(128)); } SECTION("with exponent") { - CHECK(parse_string("0e1").parse() == json(0e1)); - CHECK(parse_string("0E1").parse() == json(0e1)); + CHECK(json::parser(json::input_adapter::create(std::string("0e1"))).parse() == json(0e1)); + CHECK(json::parser(json::input_adapter::create(std::string("0E1"))).parse() == json(0e1)); - CHECK(parse_string("10000E-4").parse() == json(10000e-4)); - CHECK(parse_string("10000E-3").parse() == json(10000e-3)); - CHECK(parse_string("10000E-2").parse() == json(10000e-2)); - CHECK(parse_string("10000E-1").parse() == json(10000e-1)); - CHECK(parse_string("10000E0").parse() == json(10000e0)); - CHECK(parse_string("10000E1").parse() == json(10000e1)); - CHECK(parse_string("10000E2").parse() == json(10000e2)); - CHECK(parse_string("10000E3").parse() == json(10000e3)); - CHECK(parse_string("10000E4").parse() == json(10000e4)); + CHECK(json::parser(json::input_adapter::create(std::string("10000E-4"))).parse() == json(10000e-4)); + CHECK(json::parser(json::input_adapter::create(std::string("10000E-3"))).parse() == json(10000e-3)); + CHECK(json::parser(json::input_adapter::create(std::string("10000E-2"))).parse() == json(10000e-2)); + CHECK(json::parser(json::input_adapter::create(std::string("10000E-1"))).parse() == json(10000e-1)); + CHECK(json::parser(json::input_adapter::create(std::string("10000E0"))).parse() == json(10000e0)); + CHECK(json::parser(json::input_adapter::create(std::string("10000E1"))).parse() == json(10000e1)); + CHECK(json::parser(json::input_adapter::create(std::string("10000E2"))).parse() == json(10000e2)); + CHECK(json::parser(json::input_adapter::create(std::string("10000E3"))).parse() == json(10000e3)); + CHECK(json::parser(json::input_adapter::create(std::string("10000E4"))).parse() == json(10000e4)); - CHECK(parse_string("10000e-4").parse() == json(10000e-4)); - CHECK(parse_string("10000e-3").parse() == json(10000e-3)); - CHECK(parse_string("10000e-2").parse() == json(10000e-2)); - CHECK(parse_string("10000e-1").parse() == json(10000e-1)); - CHECK(parse_string("10000e0").parse() == json(10000e0)); - CHECK(parse_string("10000e1").parse() == json(10000e1)); - CHECK(parse_string("10000e2").parse() == json(10000e2)); - CHECK(parse_string("10000e3").parse() == json(10000e3)); - CHECK(parse_string("10000e4").parse() == json(10000e4)); + CHECK(json::parser(json::input_adapter::create(std::string("10000e-4"))).parse() == json(10000e-4)); + CHECK(json::parser(json::input_adapter::create(std::string("10000e-3"))).parse() == json(10000e-3)); + CHECK(json::parser(json::input_adapter::create(std::string("10000e-2"))).parse() == json(10000e-2)); + CHECK(json::parser(json::input_adapter::create(std::string("10000e-1"))).parse() == json(10000e-1)); + CHECK(json::parser(json::input_adapter::create(std::string("10000e0"))).parse() == json(10000e0)); + CHECK(json::parser(json::input_adapter::create(std::string("10000e1"))).parse() == json(10000e1)); + CHECK(json::parser(json::input_adapter::create(std::string("10000e2"))).parse() == json(10000e2)); + CHECK(json::parser(json::input_adapter::create(std::string("10000e3"))).parse() == json(10000e3)); + CHECK(json::parser(json::input_adapter::create(std::string("10000e4"))).parse() == json(10000e4)); - CHECK(parse_string("-0e1").parse() == json(-0e1)); - CHECK(parse_string("-0E1").parse() == json(-0e1)); - CHECK(parse_string("-0E123").parse() == json(-0e123)); + CHECK(json::parser(json::input_adapter::create(std::string("-0e1"))).parse() == json(-0e1)); + CHECK(json::parser(json::input_adapter::create(std::string("-0E1"))).parse() == json(-0e1)); + CHECK(json::parser(json::input_adapter::create(std::string("-0E123"))).parse() == json(-0e123)); } SECTION("edge cases") @@ -239,9 +232,9 @@ TEST_CASE("parser class") // agree exactly on their numeric values. // -(2**53)+1 - CHECK(parse_string("-9007199254740991").parse().get() == -9007199254740991); + CHECK(json::parser(json::input_adapter::create(std::string("-9007199254740991"))).parse().get() == -9007199254740991); // (2**53)-1 - CHECK(parse_string("9007199254740991").parse().get() == 9007199254740991); + CHECK(json::parser(json::input_adapter::create(std::string("9007199254740991"))).parse().get() == 9007199254740991); } SECTION("over the edge cases") // issue #178 - Integer conversion to unsigned (incorrect handling of 64 bit integers) @@ -254,11 +247,11 @@ TEST_CASE("parser class") // i.e. -(2**63) -> (2**64)-1. // -(2**63) ** Note: compilers see negative literals as negated positive numbers (hence the -1)) - CHECK(parse_string("-9223372036854775808").parse().get() == -9223372036854775807 - 1); + CHECK(json::parser(json::input_adapter::create(std::string("-9223372036854775808"))).parse().get() == -9223372036854775807 - 1); // (2**63)-1 - CHECK(parse_string("9223372036854775807").parse().get() == 9223372036854775807); + CHECK(json::parser(json::input_adapter::create(std::string("9223372036854775807"))).parse().get() == 9223372036854775807); // (2**64)-1 - CHECK(parse_string("18446744073709551615").parse().get() == 18446744073709551615u); + CHECK(json::parser(json::input_adapter::create(std::string("18446744073709551615"))).parse().get() == 18446744073709551615u); } } @@ -266,85 +259,85 @@ TEST_CASE("parser class") { SECTION("without exponent") { - CHECK(parse_string("-128.5").parse() == json(-128.5)); - CHECK(parse_string("0.999").parse() == json(0.999)); - CHECK(parse_string("128.5").parse() == json(128.5)); - CHECK(parse_string("-0.0").parse() == json(-0.0)); + CHECK(json::parser(json::input_adapter::create(std::string("-128.5"))).parse() == json(-128.5)); + CHECK(json::parser(json::input_adapter::create(std::string("0.999"))).parse() == json(0.999)); + CHECK(json::parser(json::input_adapter::create(std::string("128.5"))).parse() == json(128.5)); + CHECK(json::parser(json::input_adapter::create(std::string("-0.0"))).parse() == json(-0.0)); } SECTION("with exponent") { - CHECK(parse_string("-128.5E3").parse() == json(-128.5E3)); - CHECK(parse_string("-128.5E-3").parse() == json(-128.5E-3)); - CHECK(parse_string("-0.0e1").parse() == json(-0.0e1)); - CHECK(parse_string("-0.0E1").parse() == json(-0.0e1)); + CHECK(json::parser(json::input_adapter::create(std::string("-128.5E3"))).parse() == json(-128.5E3)); + CHECK(json::parser(json::input_adapter::create(std::string("-128.5E-3"))).parse() == json(-128.5E-3)); + CHECK(json::parser(json::input_adapter::create(std::string("-0.0e1"))).parse() == json(-0.0e1)); + CHECK(json::parser(json::input_adapter::create(std::string("-0.0E1"))).parse() == json(-0.0e1)); } } SECTION("overflow") { // overflows during parsing yield an exception - CHECK_THROWS_AS(parse_string("1.18973e+4932").parse() == json(), json::out_of_range); - CHECK_THROWS_WITH(parse_string("1.18973e+4932").parse() == json(), + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("1.18973e+4932"))).parse() == json(), json::out_of_range); + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("1.18973e+4932"))).parse() == json(), "[json.exception.out_of_range.406] number overflow parsing '1.18973e+4932'"); } SECTION("invalid numbers") { - CHECK_THROWS_AS(parse_string("01").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("--1").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("1.").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("1E").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("1E-").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("1.E1").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("-1E").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("-0E#").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("-0E-#").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("-0#").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("-0.0:").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("-0.0Z").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("-0E123:").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("-0e0-:").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("-0e-:").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("-0f").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("01"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("--1"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("1."))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("1E"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("1E-"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("1.E1"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("-1E"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("-0E#"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("-0E-#"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("-0#"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("-0.0:"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("-0.0Z"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("-0E123:"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("-0e0-:"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("-0e-:"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("-0f"))).parse(), json::parse_error); // numbers must not begin with "+" - CHECK_THROWS_AS(parse_string("+1").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("+0").parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("+1"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("+0"))).parse(), json::parse_error); - CHECK_THROWS_WITH(parse_string("01").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("01"))).parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected number literal; expected end of input"); - CHECK_THROWS_WITH(parse_string("-01").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("-01"))).parse(), "[json.exception.parse_error.101] parse error at 3: syntax error - unexpected number literal; expected end of input"); - CHECK_THROWS_WITH(parse_string("--1").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("--1"))).parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - invalid number; expected digit after '-'; last read: '--'"); - CHECK_THROWS_WITH(parse_string("1.").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("1."))).parse(), "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected digit after '.'; last read: '1.'"); - CHECK_THROWS_WITH(parse_string("1E").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("1E"))).parse(), "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected '+', '-', or digit after exponent; last read: '1E'"); - CHECK_THROWS_WITH(parse_string("1E-").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("1E-"))).parse(), "[json.exception.parse_error.101] parse error at 4: syntax error - invalid number; expected digit after exponent sign; last read: '1E-'"); - CHECK_THROWS_WITH(parse_string("1.E1").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("1.E1"))).parse(), "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected digit after '.'; last read: '1.E'"); - CHECK_THROWS_WITH(parse_string("-1E").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("-1E"))).parse(), "[json.exception.parse_error.101] parse error at 4: syntax error - invalid number; expected '+', '-', or digit after exponent; last read: '-1E'"); - CHECK_THROWS_WITH(parse_string("-0E#").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("-0E#"))).parse(), "[json.exception.parse_error.101] parse error at 4: syntax error - invalid number; expected '+', '-', or digit after exponent; last read: '-0E#'"); - CHECK_THROWS_WITH(parse_string("-0E-#").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("-0E-#"))).parse(), "[json.exception.parse_error.101] parse error at 5: syntax error - invalid number; expected digit after exponent sign; last read: '-0E-#'"); - CHECK_THROWS_WITH(parse_string("-0#").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("-0#"))).parse(), "[json.exception.parse_error.101] parse error at 3: syntax error - invalid literal; last read: '-0#'; expected end of input"); - CHECK_THROWS_WITH(parse_string("-0.0:").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("-0.0:"))).parse(), "[json.exception.parse_error.101] parse error at 5: syntax error - unexpected ':'; expected end of input"); - CHECK_THROWS_WITH(parse_string("-0.0Z").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("-0.0Z"))).parse(), "[json.exception.parse_error.101] parse error at 5: syntax error - invalid literal; last read: '-0.0Z'; expected end of input"); - CHECK_THROWS_WITH(parse_string("-0E123:").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("-0E123:"))).parse(), "[json.exception.parse_error.101] parse error at 7: syntax error - unexpected ':'; expected end of input"); - CHECK_THROWS_WITH(parse_string("-0e0-:").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("-0e0-:"))).parse(), "[json.exception.parse_error.101] parse error at 6: syntax error - invalid number; expected digit after '-'; last read: '-:'; expected end of input"); - CHECK_THROWS_WITH(parse_string("-0e-:").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("-0e-:"))).parse(), "[json.exception.parse_error.101] parse error at 5: syntax error - invalid number; expected digit after exponent sign; last read: '-0e-:'"); - CHECK_THROWS_WITH(parse_string("-0f").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("-0f"))).parse(), "[json.exception.parse_error.101] parse error at 4: syntax error - invalid literal; expected 'false'; last read: '-0f'; expected end of input"); } } @@ -354,30 +347,30 @@ TEST_CASE("parser class") { SECTION("null") { - CHECK(parse_string("null").accept()); + CHECK(json::parser(json::input_adapter::create(std::string("null"))).accept()); } SECTION("true") { - CHECK(parse_string("true").accept()); + CHECK(json::parser(json::input_adapter::create(std::string("true"))).accept()); } SECTION("false") { - CHECK(parse_string("false").accept()); + CHECK(json::parser(json::input_adapter::create(std::string("false"))).accept()); } SECTION("array") { SECTION("empty array") { - CHECK(parse_string("[]").accept()); - CHECK(parse_string("[ ]").accept()); + CHECK(json::parser(json::input_adapter::create(std::string("[]"))).accept()); + CHECK(json::parser(json::input_adapter::create(std::string("[ ]"))).accept()); } SECTION("nonempty array") { - CHECK(parse_string("[true, false, null]").accept()); + CHECK(json::parser(json::input_adapter::create(std::string("[true, false, null]"))).accept()); } } @@ -385,105 +378,105 @@ TEST_CASE("parser class") { SECTION("empty object") { - CHECK(parse_string("{}").accept()); - CHECK(parse_string("{ }").accept()); + CHECK(json::parser(json::input_adapter::create(std::string("{}"))).accept()); + CHECK(json::parser(json::input_adapter::create(std::string("{ }"))).accept()); } SECTION("nonempty object") { - CHECK(parse_string("{\"\": true, \"one\": 1, \"two\": null}").accept()); + CHECK(json::parser(json::input_adapter::create(std::string("{\"\": true, \"one\": 1, \"two\": null}"))).accept()); } } SECTION("string") { // empty string - CHECK(parse_string("\"\"").accept()); + CHECK(json::parser(json::input_adapter::create(std::string("\"\""))).accept()); SECTION("errors") { // error: tab in string - CHECK(parse_string("\"\t\"").accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\t\""))).accept() == false); // error: newline in string - CHECK(parse_string("\"\n\"").accept() == false); - CHECK(parse_string("\"\r\"").accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\n\""))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\r\""))).accept() == false); // error: backspace in string - CHECK(parse_string("\"\b\"").accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\b\""))).accept() == false); // improve code coverage - CHECK(parse_string("\uFF01").accept() == false); - CHECK(parse_string("[-4:1,]").accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\uFF01"))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("[-4:1,]"))).accept() == false); // unescaped control characters - CHECK(parse_string("\"\x00\"").accept() == false); - CHECK(parse_string("\"\x01\"").accept() == false); - CHECK(parse_string("\"\x02\"").accept() == false); - CHECK(parse_string("\"\x03\"").accept() == false); - CHECK(parse_string("\"\x04\"").accept() == false); - CHECK(parse_string("\"\x05\"").accept() == false); - CHECK(parse_string("\"\x06\"").accept() == false); - CHECK(parse_string("\"\x07\"").accept() == false); - CHECK(parse_string("\"\x08\"").accept() == false); - CHECK(parse_string("\"\x09\"").accept() == false); - CHECK(parse_string("\"\x0a\"").accept() == false); - CHECK(parse_string("\"\x0b\"").accept() == false); - CHECK(parse_string("\"\x0c\"").accept() == false); - CHECK(parse_string("\"\x0d\"").accept() == false); - CHECK(parse_string("\"\x0e\"").accept() == false); - CHECK(parse_string("\"\x0f\"").accept() == false); - CHECK(parse_string("\"\x10\"").accept() == false); - CHECK(parse_string("\"\x11\"").accept() == false); - CHECK(parse_string("\"\x12\"").accept() == false); - CHECK(parse_string("\"\x13\"").accept() == false); - CHECK(parse_string("\"\x14\"").accept() == false); - CHECK(parse_string("\"\x15\"").accept() == false); - CHECK(parse_string("\"\x16\"").accept() == false); - CHECK(parse_string("\"\x17\"").accept() == false); - CHECK(parse_string("\"\x18\"").accept() == false); - CHECK(parse_string("\"\x19\"").accept() == false); - CHECK(parse_string("\"\x1a\"").accept() == false); - CHECK(parse_string("\"\x1b\"").accept() == false); - CHECK(parse_string("\"\x1c\"").accept() == false); - CHECK(parse_string("\"\x1d\"").accept() == false); - CHECK(parse_string("\"\x1e\"").accept() == false); - CHECK(parse_string("\"\x1f\"").accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\x00\""))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\x01\""))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\x02\""))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\x03\""))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\x04\""))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\x05\""))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\x06\""))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\x07\""))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\x08\""))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\x09\""))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\x0a\""))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\x0b\""))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\x0c\""))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\x0d\""))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\x0e\""))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\x0f\""))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\x10\""))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\x11\""))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\x12\""))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\x13\""))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\x14\""))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\x15\""))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\x16\""))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\x17\""))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\x18\""))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\x19\""))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\x1a\""))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\x1b\""))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\x1c\""))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\x1d\""))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\x1e\""))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\x1f\""))).accept() == false); } SECTION("escaped") { // quotation mark "\"" auto r1 = R"("\"")"_json; - CHECK(parse_string("\"\\\"\"").accept()); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\\"\""))).accept()); // reverse solidus "\\" auto r2 = R"("\\")"_json; - CHECK(parse_string("\"\\\\\"").accept()); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\\\\""))).accept()); // solidus - CHECK(parse_string("\"\\/\"").accept()); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\/\""))).accept()); // backspace - CHECK(parse_string("\"\\b\"").accept()); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\b\""))).accept()); // formfeed - CHECK(parse_string("\"\\f\"").accept()); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\f\""))).accept()); // newline - CHECK(parse_string("\"\\n\"").accept()); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\n\""))).accept()); // carriage return - CHECK(parse_string("\"\\r\"").accept()); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\r\""))).accept()); // horizontal tab - CHECK(parse_string("\"\\t\"").accept()); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\t\""))).accept()); - CHECK(parse_string("\"\\u0001\"").accept()); - CHECK(parse_string("\"\\u000a\"").accept()); - CHECK(parse_string("\"\\u00b0\"").accept()); - CHECK(parse_string("\"\\u0c00\"").accept()); - CHECK(parse_string("\"\\ud000\"").accept()); - CHECK(parse_string("\"\\u000E\"").accept()); - CHECK(parse_string("\"\\u00F0\"").accept()); - CHECK(parse_string("\"\\u0100\"").accept()); - CHECK(parse_string("\"\\u2000\"").accept()); - CHECK(parse_string("\"\\uFFFF\"").accept()); - CHECK(parse_string("\"\\u20AC\"").accept()); - CHECK(parse_string("\"€\"").accept()); - CHECK(parse_string("\"🎈\"").accept()); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\u0001\""))).accept()); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\u000a\""))).accept()); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\u00b0\""))).accept()); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\u0c00\""))).accept()); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\ud000\""))).accept()); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\u000E\""))).accept()); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\u00F0\""))).accept()); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\u0100\""))).accept()); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\u2000\""))).accept()); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\uFFFF\""))).accept()); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\u20AC\""))).accept()); + CHECK(json::parser(json::input_adapter::create(std::string("\"€\""))).accept()); + CHECK(json::parser(json::input_adapter::create(std::string("\"🎈\""))).accept()); - CHECK(parse_string("\"\\ud80c\\udc60\"").accept()); - CHECK(parse_string("\"\\ud83c\\udf1e\"").accept()); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\ud80c\\udc60\""))).accept()); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\ud83c\\udf1e\""))).accept()); } } @@ -493,40 +486,40 @@ TEST_CASE("parser class") { SECTION("without exponent") { - CHECK(parse_string("-128").accept()); - CHECK(parse_string("-0").accept()); - CHECK(parse_string("0").accept()); - CHECK(parse_string("128").accept()); + CHECK(json::parser(json::input_adapter::create(std::string("-128"))).accept()); + CHECK(json::parser(json::input_adapter::create(std::string("-0"))).accept()); + CHECK(json::parser(json::input_adapter::create(std::string("0"))).accept()); + CHECK(json::parser(json::input_adapter::create(std::string("128"))).accept()); } SECTION("with exponent") { - CHECK(parse_string("0e1").accept()); - CHECK(parse_string("0E1").accept()); + CHECK(json::parser(json::input_adapter::create(std::string("0e1"))).accept()); + CHECK(json::parser(json::input_adapter::create(std::string("0E1"))).accept()); - CHECK(parse_string("10000E-4").accept()); - CHECK(parse_string("10000E-3").accept()); - CHECK(parse_string("10000E-2").accept()); - CHECK(parse_string("10000E-1").accept()); - CHECK(parse_string("10000E0").accept()); - CHECK(parse_string("10000E1").accept()); - CHECK(parse_string("10000E2").accept()); - CHECK(parse_string("10000E3").accept()); - CHECK(parse_string("10000E4").accept()); + CHECK(json::parser(json::input_adapter::create(std::string("10000E-4"))).accept()); + CHECK(json::parser(json::input_adapter::create(std::string("10000E-3"))).accept()); + CHECK(json::parser(json::input_adapter::create(std::string("10000E-2"))).accept()); + CHECK(json::parser(json::input_adapter::create(std::string("10000E-1"))).accept()); + CHECK(json::parser(json::input_adapter::create(std::string("10000E0"))).accept()); + CHECK(json::parser(json::input_adapter::create(std::string("10000E1"))).accept()); + CHECK(json::parser(json::input_adapter::create(std::string("10000E2"))).accept()); + CHECK(json::parser(json::input_adapter::create(std::string("10000E3"))).accept()); + CHECK(json::parser(json::input_adapter::create(std::string("10000E4"))).accept()); - CHECK(parse_string("10000e-4").accept()); - CHECK(parse_string("10000e-3").accept()); - CHECK(parse_string("10000e-2").accept()); - CHECK(parse_string("10000e-1").accept()); - CHECK(parse_string("10000e0").accept()); - CHECK(parse_string("10000e1").accept()); - CHECK(parse_string("10000e2").accept()); - CHECK(parse_string("10000e3").accept()); - CHECK(parse_string("10000e4").accept()); + CHECK(json::parser(json::input_adapter::create(std::string("10000e-4"))).accept()); + CHECK(json::parser(json::input_adapter::create(std::string("10000e-3"))).accept()); + CHECK(json::parser(json::input_adapter::create(std::string("10000e-2"))).accept()); + CHECK(json::parser(json::input_adapter::create(std::string("10000e-1"))).accept()); + CHECK(json::parser(json::input_adapter::create(std::string("10000e0"))).accept()); + CHECK(json::parser(json::input_adapter::create(std::string("10000e1"))).accept()); + CHECK(json::parser(json::input_adapter::create(std::string("10000e2"))).accept()); + CHECK(json::parser(json::input_adapter::create(std::string("10000e3"))).accept()); + CHECK(json::parser(json::input_adapter::create(std::string("10000e4"))).accept()); - CHECK(parse_string("-0e1").accept()); - CHECK(parse_string("-0E1").accept()); - CHECK(parse_string("-0E123").accept()); + CHECK(json::parser(json::input_adapter::create(std::string("-0e1"))).accept()); + CHECK(json::parser(json::input_adapter::create(std::string("-0E1"))).accept()); + CHECK(json::parser(json::input_adapter::create(std::string("-0E123"))).accept()); } SECTION("edge cases") @@ -538,9 +531,9 @@ TEST_CASE("parser class") // agree exactly on their numeric values. // -(2**53)+1 - CHECK(parse_string("-9007199254740991").accept()); + CHECK(json::parser(json::input_adapter::create(std::string("-9007199254740991"))).accept()); // (2**53)-1 - CHECK(parse_string("9007199254740991").accept()); + CHECK(json::parser(json::input_adapter::create(std::string("9007199254740991"))).accept()); } SECTION("over the edge cases") // issue #178 - Integer conversion to unsigned (incorrect handling of 64 bit integers) @@ -553,11 +546,11 @@ TEST_CASE("parser class") // i.e. -(2**63) -> (2**64)-1. // -(2**63) ** Note: compilers see negative literals as negated positive numbers (hence the -1)) - CHECK(parse_string("-9223372036854775808").accept()); + CHECK(json::parser(json::input_adapter::create(std::string("-9223372036854775808"))).accept()); // (2**63)-1 - CHECK(parse_string("9223372036854775807").accept()); + CHECK(json::parser(json::input_adapter::create(std::string("9223372036854775807"))).accept()); // (2**64)-1 - CHECK(parse_string("18446744073709551615").accept()); + CHECK(json::parser(json::input_adapter::create(std::string("18446744073709551615"))).accept()); } } @@ -565,49 +558,49 @@ TEST_CASE("parser class") { SECTION("without exponent") { - CHECK(parse_string("-128.5").accept()); - CHECK(parse_string("0.999").accept()); - CHECK(parse_string("128.5").accept()); - CHECK(parse_string("-0.0").accept()); + CHECK(json::parser(json::input_adapter::create(std::string("-128.5"))).accept()); + CHECK(json::parser(json::input_adapter::create(std::string("0.999"))).accept()); + CHECK(json::parser(json::input_adapter::create(std::string("128.5"))).accept()); + CHECK(json::parser(json::input_adapter::create(std::string("-0.0"))).accept()); } SECTION("with exponent") { - CHECK(parse_string("-128.5E3").accept()); - CHECK(parse_string("-128.5E-3").accept()); - CHECK(parse_string("-0.0e1").accept()); - CHECK(parse_string("-0.0E1").accept()); + CHECK(json::parser(json::input_adapter::create(std::string("-128.5E3"))).accept()); + CHECK(json::parser(json::input_adapter::create(std::string("-128.5E-3"))).accept()); + CHECK(json::parser(json::input_adapter::create(std::string("-0.0e1"))).accept()); + CHECK(json::parser(json::input_adapter::create(std::string("-0.0E1"))).accept()); } } SECTION("overflow") { // overflows during parsing yield an exception, but is accepted anyway - CHECK(parse_string("1.18973e+4932").accept()); + CHECK(json::parser(json::input_adapter::create(std::string("1.18973e+4932"))).accept()); } SECTION("invalid numbers") { - CHECK(parse_string("01").accept() == false); - CHECK(parse_string("--1").accept() == false); - CHECK(parse_string("1.").accept() == false); - CHECK(parse_string("1E").accept() == false); - CHECK(parse_string("1E-").accept() == false); - CHECK(parse_string("1.E1").accept() == false); - CHECK(parse_string("-1E").accept() == false); - CHECK(parse_string("-0E#").accept() == false); - CHECK(parse_string("-0E-#").accept() == false); - CHECK(parse_string("-0#").accept() == false); - CHECK(parse_string("-0.0:").accept() == false); - CHECK(parse_string("-0.0Z").accept() == false); - CHECK(parse_string("-0E123:").accept() == false); - CHECK(parse_string("-0e0-:").accept() == false); - CHECK(parse_string("-0e-:").accept() == false); - CHECK(parse_string("-0f").accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("01"))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("--1"))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("1."))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("1E"))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("1E-"))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("1.E1"))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("-1E"))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("-0E#"))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("-0E-#"))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("-0#"))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("-0.0:"))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("-0.0Z"))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("-0E123:"))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("-0e0-:"))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("-0e-:"))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("-0f"))).accept() == false); // numbers must not begin with "+" - CHECK(parse_string("+1").accept() == false); - CHECK(parse_string("+0").accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("+1"))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("+0"))).accept() == false); } } } @@ -615,152 +608,152 @@ TEST_CASE("parser class") SECTION("parse errors") { // unexpected end of number - CHECK_THROWS_AS(parse_string("0.").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("-").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("--").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("-0.").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("-.").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("-:").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("0.:").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("e.").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("1e.").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("1e/").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("1e:").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("1E.").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("1E/").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("1E:").parse(), json::parse_error); - CHECK_THROWS_WITH(parse_string("0.").parse(), + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("0."))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("-"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("--"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("-0."))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("-."))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("-:"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("0.:"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("e."))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("1e."))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("1e/"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("1e:"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("1E."))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("1E/"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("1E:"))).parse(), json::parse_error); + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("0."))).parse(), "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected digit after '.'; last read: '0.'"); - CHECK_THROWS_WITH(parse_string("-").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("-"))).parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - invalid number; expected digit after '-'; last read: '-'"); - CHECK_THROWS_WITH(parse_string("--").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("--"))).parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - invalid number; expected digit after '-'; last read: '--'"); - CHECK_THROWS_WITH(parse_string("-0.").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("-0."))).parse(), "[json.exception.parse_error.101] parse error at 4: syntax error - invalid number; expected digit after '.'; last read: '-0.'"); - CHECK_THROWS_WITH(parse_string("-.").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("-."))).parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - invalid number; expected digit after '-'; last read: '-.'"); - CHECK_THROWS_WITH(parse_string("-:").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("-:"))).parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - invalid number; expected digit after '-'; last read: '-:'"); - CHECK_THROWS_WITH(parse_string("0.:").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("0.:"))).parse(), "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected digit after '.'; last read: '0.:'"); - CHECK_THROWS_WITH(parse_string("e.").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("e."))).parse(), "[json.exception.parse_error.101] parse error at 1: syntax error - invalid literal; last read: 'e'"); - CHECK_THROWS_WITH(parse_string("1e.").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("1e."))).parse(), "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected '+', '-', or digit after exponent; last read: '1e.'"); - CHECK_THROWS_WITH(parse_string("1e/").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("1e/"))).parse(), "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected '+', '-', or digit after exponent; last read: '1e/'"); - CHECK_THROWS_WITH(parse_string("1e:").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("1e:"))).parse(), "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected '+', '-', or digit after exponent; last read: '1e:'"); - CHECK_THROWS_WITH(parse_string("1E.").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("1E."))).parse(), "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected '+', '-', or digit after exponent; last read: '1E.'"); - CHECK_THROWS_WITH(parse_string("1E/").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("1E/"))).parse(), "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected '+', '-', or digit after exponent; last read: '1E/'"); - CHECK_THROWS_WITH(parse_string("1E:").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("1E:"))).parse(), "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected '+', '-', or digit after exponent; last read: '1E:'"); // unexpected end of null - CHECK_THROWS_AS(parse_string("n").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("nu").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("nul").parse(), json::parse_error); - CHECK_THROWS_WITH(parse_string("n").parse(), + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("n"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("nu"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("nul"))).parse(), json::parse_error); + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("n"))).parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - invalid literal; expected 'null'; last read: 'n'"); - CHECK_THROWS_WITH(parse_string("nu").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("nu"))).parse(), "[json.exception.parse_error.101] parse error at 3: syntax error - invalid literal; expected 'null'; last read: 'nu'"); - CHECK_THROWS_WITH(parse_string("nul").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("nul"))).parse(), "[json.exception.parse_error.101] parse error at 4: syntax error - invalid literal; expected 'null'; last read: 'nul'"); // unexpected end of true - CHECK_THROWS_AS(parse_string("t").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("tr").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("tru").parse(), json::parse_error); - CHECK_THROWS_WITH(parse_string("t").parse(), + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("t"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("tr"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("tru"))).parse(), json::parse_error); + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("t"))).parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - invalid literal; expected 'true'; last read: 't'"); - CHECK_THROWS_WITH(parse_string("tr").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("tr"))).parse(), "[json.exception.parse_error.101] parse error at 3: syntax error - invalid literal; expected 'true'; last read: 'tr'"); - CHECK_THROWS_WITH(parse_string("tru").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("tru"))).parse(), "[json.exception.parse_error.101] parse error at 4: syntax error - invalid literal; expected 'true'; last read: 'tru'"); // unexpected end of false - CHECK_THROWS_AS(parse_string("f").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("fa").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("fal").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("fals").parse(), json::parse_error); - CHECK_THROWS_WITH(parse_string("f").parse(), + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("f"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("fa"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("fal"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("fals"))).parse(), json::parse_error); + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("f"))).parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - invalid literal; expected 'false'; last read: 'f'"); - CHECK_THROWS_WITH(parse_string("fa").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("fa"))).parse(), "[json.exception.parse_error.101] parse error at 3: syntax error - invalid literal; expected 'false'; last read: 'fa'"); - CHECK_THROWS_WITH(parse_string("fal").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("fal"))).parse(), "[json.exception.parse_error.101] parse error at 4: syntax error - invalid literal; expected 'false'; last read: 'fal'"); - CHECK_THROWS_WITH(parse_string("fals").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("fals"))).parse(), "[json.exception.parse_error.101] parse error at 5: syntax error - invalid literal; expected 'false'; last read: 'fals'"); // missing/unexpected end of array - CHECK_THROWS_AS(parse_string("[").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("[1").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("[1,").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("[1,]").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("]").parse(), json::parse_error); - CHECK_THROWS_WITH(parse_string("[").parse(), + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("["))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("[1"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("[1,"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("[1,]"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("]"))).parse(), json::parse_error); + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("["))).parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected end of input"); - CHECK_THROWS_WITH(parse_string("[1").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("[1"))).parse(), "[json.exception.parse_error.101] parse error at 3: syntax error - unexpected end of input; expected ']'"); - CHECK_THROWS_WITH(parse_string("[1,").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("[1,"))).parse(), "[json.exception.parse_error.101] parse error at 4: syntax error - unexpected end of input"); - CHECK_THROWS_WITH(parse_string("[1,]").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("[1,]"))).parse(), "[json.exception.parse_error.101] parse error at 4: syntax error - unexpected ']'"); - CHECK_THROWS_WITH(parse_string("]").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("]"))).parse(), "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected ']'"); // missing/unexpected end of object - CHECK_THROWS_AS(parse_string("{").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("{\"foo\"").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("{\"foo\":").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("{\"foo\":}").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("{\"foo\":1,}").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("}").parse(), json::parse_error); - CHECK_THROWS_WITH(parse_string("{").parse(), + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("{"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("{\"foo\""))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("{\"foo\":"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("{\"foo\":}"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("{\"foo\":1,}"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("}"))).parse(), json::parse_error); + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("{"))).parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected end of input; expected string literal"); - CHECK_THROWS_WITH(parse_string("{\"foo\"").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("{\"foo\""))).parse(), "[json.exception.parse_error.101] parse error at 7: syntax error - unexpected end of input; expected ':'"); - CHECK_THROWS_WITH(parse_string("{\"foo\":").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("{\"foo\":"))).parse(), "[json.exception.parse_error.101] parse error at 8: syntax error - unexpected end of input"); - CHECK_THROWS_WITH(parse_string("{\"foo\":}").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("{\"foo\":}"))).parse(), "[json.exception.parse_error.101] parse error at 8: syntax error - unexpected '}'"); - CHECK_THROWS_WITH(parse_string("{\"foo\":1,}").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("{\"foo\":1,}"))).parse(), "[json.exception.parse_error.101] parse error at 10: syntax error - unexpected '}'; expected string literal"); - CHECK_THROWS_WITH(parse_string("}").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("}"))).parse(), "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '}'"); // missing/unexpected end of string - CHECK_THROWS_AS(parse_string("\"").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("\"\\\"").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("\"\\u\"").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("\"\\u0\"").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("\"\\u01\"").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("\"\\u012\"").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("\"\\u").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("\"\\u0").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("\"\\u01").parse(), json::parse_error); - CHECK_THROWS_AS(parse_string("\"\\u012").parse(), json::parse_error); - CHECK_THROWS_WITH(parse_string("\"").parse(), + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\""))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\\\""))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\\u\""))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\\u0\""))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\\u01\""))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\\u012\""))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\\u"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\\u0"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\\u01"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\\u012"))).parse(), json::parse_error); + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("\""))).parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: missing closing quote; last read: '\"'"); - CHECK_THROWS_WITH(parse_string("\"\\\"").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("\"\\\""))).parse(), "[json.exception.parse_error.101] parse error at 4: syntax error - invalid string: missing closing quote; last read: '\"\\\"'"); - CHECK_THROWS_WITH(parse_string("\"\\u\"").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("\"\\u\""))).parse(), "[json.exception.parse_error.101] parse error at 4: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read: '\"\\u\"'"); - CHECK_THROWS_WITH(parse_string("\"\\u0\"").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("\"\\u0\""))).parse(), "[json.exception.parse_error.101] parse error at 5: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read: '\"\\u0\"'"); - CHECK_THROWS_WITH(parse_string("\"\\u01\"").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("\"\\u01\""))).parse(), "[json.exception.parse_error.101] parse error at 6: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read: '\"\\u01\"'"); - CHECK_THROWS_WITH(parse_string("\"\\u012\"").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("\"\\u012\""))).parse(), "[json.exception.parse_error.101] parse error at 7: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read: '\"\\u012\"'"); - CHECK_THROWS_WITH(parse_string("\"\\u").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("\"\\u"))).parse(), "[json.exception.parse_error.101] parse error at 4: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read: '\"\\u'"); - CHECK_THROWS_WITH(parse_string("\"\\u0").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("\"\\u0"))).parse(), "[json.exception.parse_error.101] parse error at 5: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read: '\"\\u0'"); - CHECK_THROWS_WITH(parse_string("\"\\u01").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("\"\\u01"))).parse(), "[json.exception.parse_error.101] parse error at 6: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read: '\"\\u01'"); - CHECK_THROWS_WITH(parse_string("\"\\u012").parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("\"\\u012"))).parse(), "[json.exception.parse_error.101] parse error at 7: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read: '\"\\u012'"); // invalid escapes @@ -780,7 +773,7 @@ TEST_CASE("parser class") case ('r'): case ('t'): { - CHECK_NOTHROW(parse_string(s.c_str()).parse()); + CHECK_NOTHROW(json::parser(json::input_adapter::create(std::string(s.c_str()))).parse()); break; } @@ -793,11 +786,11 @@ TEST_CASE("parser class") // any other combination of backslash and character is invalid default: { - CHECK_THROWS_AS(parse_string(s.c_str()).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string(s.c_str()))).parse(), json::parse_error); // only check error message if c is not a control character if (c > 0x1f) { - CHECK_THROWS_WITH(parse_string(s.c_str()).parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string(s.c_str()))).parse(), "[json.exception.parse_error.101] parse error at 3: syntax error - invalid string: forbidden character after backslash; last read: '\"\\" + std::string(1, static_cast(c)) + "'"); } break; @@ -858,49 +851,49 @@ TEST_CASE("parser class") if (valid(c)) { CAPTURE(s1); - CHECK_NOTHROW(parse_string(s1.c_str()).parse()); + CHECK_NOTHROW(json::parser(json::input_adapter::create(std::string(s1.c_str()))).parse()); CAPTURE(s2); - CHECK_NOTHROW(parse_string(s2.c_str()).parse()); + CHECK_NOTHROW(json::parser(json::input_adapter::create(std::string(s2.c_str()))).parse()); CAPTURE(s3); - CHECK_NOTHROW(parse_string(s3.c_str()).parse()); + CHECK_NOTHROW(json::parser(json::input_adapter::create(std::string(s3.c_str()))).parse()); CAPTURE(s4); - CHECK_NOTHROW(parse_string(s4.c_str()).parse()); + CHECK_NOTHROW(json::parser(json::input_adapter::create(std::string(s4.c_str()))).parse()); } else { CAPTURE(s1); - CHECK_THROWS_AS(parse_string(s1.c_str()).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string(s1.c_str()))).parse(), json::parse_error); // only check error message if c is not a control character if (c > 0x1f) { - CHECK_THROWS_WITH(parse_string(s1.c_str()).parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string(s1.c_str()))).parse(), "[json.exception.parse_error.101] parse error at 7: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read: '" + s1.substr(0, 7) + "'"); } CAPTURE(s2); - CHECK_THROWS_AS(parse_string(s2.c_str()).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string(s2.c_str()))).parse(), json::parse_error); // only check error message if c is not a control character if (c > 0x1f) { - CHECK_THROWS_WITH(parse_string(s2.c_str()).parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string(s2.c_str()))).parse(), "[json.exception.parse_error.101] parse error at 6: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read: '" + s2.substr(0, 6) + "'"); } CAPTURE(s3); - CHECK_THROWS_AS(parse_string(s3.c_str()).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string(s3.c_str()))).parse(), json::parse_error); // only check error message if c is not a control character if (c > 0x1f) { - CHECK_THROWS_WITH(parse_string(s3.c_str()).parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string(s3.c_str()))).parse(), "[json.exception.parse_error.101] parse error at 5: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read: '" + s3.substr(0, 5) + "'"); } CAPTURE(s4); - CHECK_THROWS_AS(parse_string(s4.c_str()).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string(s4.c_str()))).parse(), json::parse_error); // only check error message if c is not a control character if (c > 0x1f) { - CHECK_THROWS_WITH(parse_string(s4.c_str()).parse(), + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string(s4.c_str()))).parse(), "[json.exception.parse_error.101] parse error at 4: syntax error - invalid string: '\\u' must be followed by 4 hex digits; last read: '" + s4.substr(0, 4) + "'"); } } @@ -926,63 +919,63 @@ TEST_CASE("parser class") SECTION("parse errors (accept)") { // unexpected end of number - CHECK(parse_string("0.").accept() == false); - CHECK(parse_string("-").accept() == false); - CHECK(parse_string("--").accept() == false); - CHECK(parse_string("-0.").accept() == false); - CHECK(parse_string("-.").accept() == false); - CHECK(parse_string("-:").accept() == false); - CHECK(parse_string("0.:").accept() == false); - CHECK(parse_string("e.").accept() == false); - CHECK(parse_string("1e.").accept() == false); - CHECK(parse_string("1e/").accept() == false); - CHECK(parse_string("1e:").accept() == false); - CHECK(parse_string("1E.").accept() == false); - CHECK(parse_string("1E/").accept() == false); - CHECK(parse_string("1E:").accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("0."))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("-"))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("--"))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("-0."))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("-."))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("-:"))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("0.:"))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("e."))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("1e."))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("1e/"))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("1e:"))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("1E."))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("1E/"))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("1E:"))).accept() == false); // unexpected end of null - CHECK(parse_string("n").accept() == false); - CHECK(parse_string("nu").accept() == false); - CHECK(parse_string("nul").accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("n"))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("nu"))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("nul"))).accept() == false); // unexpected end of true - CHECK(parse_string("t").accept() == false); - CHECK(parse_string("tr").accept() == false); - CHECK(parse_string("tru").accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("t"))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("tr"))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("tru"))).accept() == false); // unexpected end of false - CHECK(parse_string("f").accept() == false); - CHECK(parse_string("fa").accept() == false); - CHECK(parse_string("fal").accept() == false); - CHECK(parse_string("fals").accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("f"))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("fa"))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("fal"))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("fals"))).accept() == false); // missing/unexpected end of array - CHECK(parse_string("[").accept() == false); - CHECK(parse_string("[1").accept() == false); - CHECK(parse_string("[1,").accept() == false); - CHECK(parse_string("[1,]").accept() == false); - CHECK(parse_string("]").accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("["))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("[1"))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("[1,"))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("[1,]"))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("]"))).accept() == false); // missing/unexpected end of object - CHECK(parse_string("{").accept() == false); - CHECK(parse_string("{\"foo\"").accept() == false); - CHECK(parse_string("{\"foo\":").accept() == false); - CHECK(parse_string("{\"foo\":}").accept() == false); - CHECK(parse_string("{\"foo\":1,}").accept() == false); - CHECK(parse_string("}").accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("{"))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("{\"foo\""))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("{\"foo\":"))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("{\"foo\":}"))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("{\"foo\":1,}"))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("}"))).accept() == false); // missing/unexpected end of string - CHECK(parse_string("\"").accept() == false); - CHECK(parse_string("\"\\\"").accept() == false); - CHECK(parse_string("\"\\u\"").accept() == false); - CHECK(parse_string("\"\\u0\"").accept() == false); - CHECK(parse_string("\"\\u01\"").accept() == false); - CHECK(parse_string("\"\\u012\"").accept() == false); - CHECK(parse_string("\"\\u").accept() == false); - CHECK(parse_string("\"\\u0").accept() == false); - CHECK(parse_string("\"\\u01").accept() == false); - CHECK(parse_string("\"\\u012").accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\""))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\\""))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\u\""))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\u0\""))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\u01\""))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\u012\""))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\u"))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\u0"))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\u01"))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\u012"))).accept() == false); // invalid escapes for (int c = 1; c < 128; ++c) @@ -1001,7 +994,7 @@ TEST_CASE("parser class") case ('r'): case ('t'): { - CHECK(parse_string(s.c_str()).accept()); + CHECK(json::parser(json::input_adapter::create(std::string(s.c_str()))).accept()); break; } @@ -1014,7 +1007,7 @@ TEST_CASE("parser class") // any other combination of backslash and character is invalid default: { - CHECK(parse_string(s.c_str()).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string(s.c_str()))).accept() == false); break; } } @@ -1073,48 +1066,48 @@ TEST_CASE("parser class") if (valid(c)) { CAPTURE(s1); - CHECK(parse_string(s1.c_str()).accept()); + CHECK(json::parser(json::input_adapter::create(std::string(s1.c_str()))).accept()); CAPTURE(s2); - CHECK(parse_string(s2.c_str()).accept()); + CHECK(json::parser(json::input_adapter::create(std::string(s2.c_str()))).accept()); CAPTURE(s3); - CHECK(parse_string(s3.c_str()).accept()); + CHECK(json::parser(json::input_adapter::create(std::string(s3.c_str()))).accept()); CAPTURE(s4); - CHECK(parse_string(s4.c_str()).accept()); + CHECK(json::parser(json::input_adapter::create(std::string(s4.c_str()))).accept()); } else { CAPTURE(s1); - CHECK(parse_string(s1.c_str()).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string(s1.c_str()))).accept() == false); CAPTURE(s2); - CHECK(parse_string(s2.c_str()).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string(s2.c_str()))).accept() == false); CAPTURE(s3); - CHECK(parse_string(s3.c_str()).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string(s3.c_str()))).accept() == false); CAPTURE(s4); - CHECK(parse_string(s4.c_str()).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string(s4.c_str()))).accept() == false); } } } // missing part of a surrogate pair - CHECK(parse_string("\"\\uD80C\"").accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\uD80C\""))).accept() == false); // invalid surrogate pair - CHECK(parse_string("\"\\uD80C\\uD80C\"").accept() == false); - CHECK(parse_string("\"\\uD80C\\u0000\"").accept() == false); - CHECK(parse_string("\"\\uD80C\\uFFFF\"").accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\uD80C\\uD80C\""))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\uD80C\\u0000\""))).accept() == false); + CHECK(json::parser(json::input_adapter::create(std::string("\"\\uD80C\\uFFFF\""))).accept() == false); } SECTION("tests found by mutate++") { // test case to make sure no comma preceeds the first key - CHECK_THROWS_AS(parse_string("{,\"key\": false}").parse(), json::parse_error); - CHECK_THROWS_WITH(parse_string("{,\"key\": false}").parse(), + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("{,\"key\": false}"))).parse(), json::parse_error); + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("{,\"key\": false}"))).parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected ','; expected string literal"); // test case to make sure an object is properly closed - CHECK_THROWS_AS(parse_string("[{\"key\": false true]").parse(), json::parse_error); - CHECK_THROWS_WITH(parse_string("[{\"key\": false true]").parse(), + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("[{\"key\": false true]"))).parse(), json::parse_error); + CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("[{\"key\": false true]"))).parse(), "[json.exception.parse_error.101] parse error at 19: syntax error - unexpected true literal; expected '}'"); // test case to make sure the callback is properly evaluated after reading a key @@ -1319,7 +1312,7 @@ TEST_CASE("parser class") SECTION("from char literal") { - CHECK(parse_string("true").parse() == json(true)); + CHECK(json::parser(json::input_adapter::create(std::string("true"))).parse() == json(true)); } SECTION("from std::string") From 7c2a187258738f3b1f83c987ac333a73ec68901b Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 20 Jun 2017 21:09:26 +0200 Subject: [PATCH 250/530] :white_check_mark: improved test coverage --- test/src/unit-comparison.cpp | 9 +++++++++ test/src/unit-deserialization.cpp | 7 +++++++ 2 files changed, 16 insertions(+) diff --git a/test/src/unit-comparison.cpp b/test/src/unit-comparison.cpp index ac243674..3219d595 100644 --- a/test/src/unit-comparison.cpp +++ b/test/src/unit-comparison.cpp @@ -31,6 +31,14 @@ SOFTWARE. #include "json.hpp" using nlohmann::json; +// helper function to check std::less +// see http://en.cppreference.com/w/cpp/utility/functional/less +template > +bool f(A a, B b, U u = U()) +{ + return u(a, b); +} + TEST_CASE("lexicographical comparison operators") { SECTION("types") @@ -69,6 +77,7 @@ TEST_CASE("lexicographical comparison operators") CAPTURE(j); // check precomputed values CHECK(operator<(j_types[i], j_types[j]) == expected[i][j]); + CHECK(f(j_types[i], j_types[j]) == expected[i][j]); } } } diff --git a/test/src/unit-deserialization.cpp b/test/src/unit-deserialization.cpp index 66fb5e1b..81ee2770 100644 --- a/test/src/unit-deserialization.cpp +++ b/test/src/unit-deserialization.cpp @@ -359,6 +359,13 @@ TEST_CASE("deserialization") CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); CHECK(not json::accept(std::begin(v), std::end(v))); } + + SECTION("case 16") + { + uint8_t v[] = {'{', '1', ',', ','}; + CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); + CHECK(not json::accept(std::begin(v), std::end(v))); + } } } } From a5c27ede32e68532954c3c13d7d12b84f84e5e34 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 20 Jun 2017 21:39:36 +0200 Subject: [PATCH 251/530] :hammer: fixed a warning --- src/json.hpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/json.hpp b/src/json.hpp index 596f7136..c312dfbb 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -9029,7 +9029,7 @@ class basic_json const std::streampos start_position; /// internal buffer - std::array buffer; + std::array buffer = {}; }; /// input adapter for buffer input From 2f007ca092f1bab33a1ade0dbfe05eef46998b90 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 20 Jun 2017 22:03:36 +0200 Subject: [PATCH 252/530] :white_check_mark: improved coverage --- src/json.hpp | 4 ++-- test/src/unit-cbor.cpp | 7 +++++++ test/src/unit-msgpack.cpp | 7 +++++++ 3 files changed, 16 insertions(+), 2 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index c312dfbb..40c7e63a 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -9986,7 +9986,7 @@ class basic_json } else { - vec[i] = static_cast(current); + vec[i] = static_cast(current); // LCOV_EXCL_LINE } } @@ -10744,7 +10744,7 @@ class basic_json } else { - oa->write_character(vec[i]); + oa->write_character(vec[i]); // LCOV_EXCL_LINE } } } diff --git a/test/src/unit-cbor.cpp b/test/src/unit-cbor.cpp index debb3732..5638d490 100644 --- a/test/src/unit-cbor.cpp +++ b/test/src/unit-cbor.cpp @@ -1253,6 +1253,13 @@ TEST_CASE("CBOR") } } } + + SECTION("invalid string in map") + { + CHECK_THROWS_AS(json::from_cbor(std::vector({0xa1, 0xff, 0x01})), json::parse_error); + CHECK_THROWS_WITH(json::from_cbor(std::vector({0xa1, 0xff, 0x01})), + "[json.exception.parse_error.113] parse error at 2: expected a CBOR string; last byte: 0xff"); + } } } diff --git a/test/src/unit-msgpack.cpp b/test/src/unit-msgpack.cpp index a580913d..5b1c301d 100644 --- a/test/src/unit-msgpack.cpp +++ b/test/src/unit-msgpack.cpp @@ -1094,6 +1094,13 @@ TEST_CASE("MessagePack") } } } + + SECTION("invalid string in map") + { + CHECK_THROWS_AS(json::from_msgpack(std::vector({0x81, 0xff, 0x01})), json::parse_error); + CHECK_THROWS_WITH(json::from_msgpack(std::vector({0x81, 0xff, 0x01})), + "[json.exception.parse_error.113] parse error at 2: expected a MessagePack string; last byte: 0xff"); + } } } From 9add24ff10f3a404ba56041d29a7719db89fc5fb Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 20 Jun 2017 22:20:41 +0200 Subject: [PATCH 253/530] :hammer: fixed a warning --- src/json.hpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/json.hpp b/src/json.hpp index 40c7e63a..ebf7dc75 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -9029,7 +9029,7 @@ class basic_json const std::streampos start_position; /// internal buffer - std::array buffer = {}; + std::array buffer{{}}; }; /// input adapter for buffer input From 84f2e64a7c3edcb27465c3baa834069e6cd800e1 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 20 Jun 2017 22:35:31 +0200 Subject: [PATCH 254/530] :white_check_mark: improved coverage --- test/src/unit-deserialization.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/test/src/unit-deserialization.cpp b/test/src/unit-deserialization.cpp index 81ee2770..48ec91c0 100644 --- a/test/src/unit-deserialization.cpp +++ b/test/src/unit-deserialization.cpp @@ -362,7 +362,7 @@ TEST_CASE("deserialization") SECTION("case 16") { - uint8_t v[] = {'{', '1', ',', ','}; + uint8_t v[] = {'{', '\"', '\"', ':', '1', '1'}; CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); CHECK(not json::accept(std::begin(v), std::end(v))); } From e7695306f1038cc04e7d3b7f812b7f997ea63889 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 20 Jun 2017 22:58:02 +0200 Subject: [PATCH 255/530] :hammer: fixed a warning --- test/src/unit-iterators2.cpp | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/test/src/unit-iterators2.cpp b/test/src/unit-iterators2.cpp index ae8ce687..29d134da 100644 --- a/test/src/unit-iterators2.cpp +++ b/test/src/unit-iterators2.cpp @@ -317,7 +317,7 @@ TEST_CASE("iterators 2") auto it = j_array.begin(); it += 3; CHECK((j_array.begin() + 3) == it); - CHECK(json::iterator(3 + j_array.begin()) == it); + CHECK((3 + j_array.begin()) == it); CHECK((it - 3) == j_array.begin()); CHECK((it - j_array.begin()) == 3); CHECK(*it == json(4)); @@ -328,7 +328,7 @@ TEST_CASE("iterators 2") auto it = j_array.cbegin(); it += 3; CHECK((j_array.cbegin() + 3) == it); - CHECK(json::const_iterator(3 + j_array.cbegin()) == it); + CHECK((3 + j_array.cbegin()) == it); CHECK((it - 3) == j_array.cbegin()); CHECK((it - j_array.cbegin()) == 3); CHECK(*it == json(4)); @@ -343,7 +343,7 @@ TEST_CASE("iterators 2") auto it = j_null.begin(); it += 3; CHECK((j_null.begin() + 3) == it); - CHECK(json::iterator(3 + j_null.begin()) == it); + CHECK((3 + j_null.begin()) == it); CHECK((it - 3) == j_null.begin()); CHECK((it - j_null.begin()) == 3); CHECK(it != j_null.end()); @@ -354,7 +354,7 @@ TEST_CASE("iterators 2") auto it = j_null.cbegin(); it += 3; CHECK((j_null.cbegin() + 3) == it); - CHECK(json::const_iterator(3 + j_null.cbegin()) == it); + CHECK((3 + j_null.cbegin()) == it); CHECK((it - 3) == j_null.cbegin()); CHECK((it - j_null.cbegin()) == 3); CHECK(it != j_null.cend()); @@ -369,7 +369,7 @@ TEST_CASE("iterators 2") auto it = j_value.begin(); it += 3; CHECK((j_value.begin() + 3) == it); - CHECK(json::iterator(3 + j_value.begin()) == it); + CHECK((3 + j_value.begin()) == it); CHECK((it - 3) == j_value.begin()); CHECK((it - j_value.begin()) == 3); CHECK(it != j_value.end()); @@ -380,7 +380,7 @@ TEST_CASE("iterators 2") auto it = j_value.cbegin(); it += 3; CHECK((j_value.cbegin() + 3) == it); - CHECK(json::const_iterator(3 + j_value.cbegin()) == it); + CHECK((3 + j_value.cbegin()) == it); CHECK((it - 3) == j_value.cbegin()); CHECK((it - j_value.cbegin()) == 3); CHECK(it != j_value.cend()); From 2fd214c14bab832c498f5ed5b119d725c1297427 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 20 Jun 2017 23:05:37 +0200 Subject: [PATCH 256/530] :hammer: adjusted pedantic flags --- Makefile | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/Makefile b/Makefile index 138a85b6..5a232cf0 100644 --- a/Makefile +++ b/Makefile @@ -51,7 +51,7 @@ doctest: # -Wno-float-equal: not all comparisons in the tests can be replaced by Approx pedantic_clang: $(MAKE) json_unit CXXFLAGS="\ - -std=c++11 \ + -std=c++11 -Wno-c++98-compat -Wno-c++98-compat-pedantic \ -Werror \ -Weverything \ -Wno-documentation-unknown-command \ @@ -60,7 +60,9 @@ pedantic_clang: -Wno-deprecated-declarations \ -Wno-weak-vtables \ -Wno-range-loop-analysis \ - -Wno-float-equal" + -Wno-float-equal \ + -Wno-switch-enum -Wno-covered-switch-default \ + -Wno-padded" # calling GCC with most warnings pedantic_gcc: From 6f3bebff5c11560542bc3ac4ad27fbe9de20e6a6 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Wed, 21 Jun 2017 07:26:50 +0200 Subject: [PATCH 257/530] :hammer: removed unexpect function --- src/json.hpp | 24 ++++++++++++------------ test/src/unit-class_parser.cpp | 14 +++++++------- test/src/unit-regression.cpp | 18 +++++++++--------- 3 files changed, 28 insertions(+), 28 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index ebf7dc75..517558c2 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -11132,7 +11132,8 @@ class basic_json name_separator, ///< the name separator `:` value_separator, ///< the value separator `,` parse_error, ///< indicating a parse error - end_of_input ///< indicating the end of the input buffer + end_of_input, ///< indicating the end of the input buffer + literal_or_value ///< a literal or the begin of a value (only for diagnostics) }; /// return name of values of type token_type (only used for errors) @@ -11170,6 +11171,8 @@ class basic_json return ""; case token_type::end_of_input: return "end of input"; + case token_type::literal_or_value: + return "'[', '{', or a literal"; default: { // catch non-enum values @@ -12903,10 +12906,16 @@ scan_number_done: break; } + case lexer::token_type::parse_error: + { + // using "uninitialized" to avoid "expected" message + expect(lexer::token_type::uninitialized); + } + default: { - // the last token was unexpected - unexpect(); + // we expected a value + expect(lexer::token_type::literal_or_value); } } @@ -13060,15 +13069,6 @@ scan_number_done: } } - /*! - @throw parse_error.101 if unexpected token occurred - */ - void unexpect() - { - errored = true; - throw_exception(); - } - [[noreturn]] void throw_exception() const { std::string error_msg = "syntax error - "; diff --git a/test/src/unit-class_parser.cpp b/test/src/unit-class_parser.cpp index a95d1642..1cca2adf 100644 --- a/test/src/unit-class_parser.cpp +++ b/test/src/unit-class_parser.cpp @@ -694,15 +694,15 @@ TEST_CASE("parser class") CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("[1,]"))).parse(), json::parse_error); CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("]"))).parse(), json::parse_error); CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("["))).parse(), - "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected end of input"); + "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected end of input; expected '[', '{', or a literal"); CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("[1"))).parse(), "[json.exception.parse_error.101] parse error at 3: syntax error - unexpected end of input; expected ']'"); CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("[1,"))).parse(), - "[json.exception.parse_error.101] parse error at 4: syntax error - unexpected end of input"); + "[json.exception.parse_error.101] parse error at 4: syntax error - unexpected end of input; expected '[', '{', or a literal"); CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("[1,]"))).parse(), - "[json.exception.parse_error.101] parse error at 4: syntax error - unexpected ']'"); + "[json.exception.parse_error.101] parse error at 4: syntax error - unexpected ']'; expected '[', '{', or a literal"); CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("]"))).parse(), - "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected ']'"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected ']'; expected '[', '{', or a literal"); // missing/unexpected end of object CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("{"))).parse(), json::parse_error); @@ -716,13 +716,13 @@ TEST_CASE("parser class") CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("{\"foo\""))).parse(), "[json.exception.parse_error.101] parse error at 7: syntax error - unexpected end of input; expected ':'"); CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("{\"foo\":"))).parse(), - "[json.exception.parse_error.101] parse error at 8: syntax error - unexpected end of input"); + "[json.exception.parse_error.101] parse error at 8: syntax error - unexpected end of input; expected '[', '{', or a literal"); CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("{\"foo\":}"))).parse(), - "[json.exception.parse_error.101] parse error at 8: syntax error - unexpected '}'"); + "[json.exception.parse_error.101] parse error at 8: syntax error - unexpected '}'; expected '[', '{', or a literal"); CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("{\"foo\":1,}"))).parse(), "[json.exception.parse_error.101] parse error at 10: syntax error - unexpected '}'; expected string literal"); CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("}"))).parse(), - "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '}'"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '}'; expected '[', '{', or a literal"); // missing/unexpected end of string CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\""))).parse(), json::parse_error); diff --git a/test/src/unit-regression.cpp b/test/src/unit-regression.cpp index 4d1231c2..d4017c7a 100644 --- a/test/src/unit-regression.cpp +++ b/test/src/unit-regression.cpp @@ -597,7 +597,7 @@ TEST_CASE("regression tests") // a parse error because of the EOF. CHECK_THROWS_AS(ss >> j, json::parse_error); CHECK_THROWS_WITH(ss >> j, - "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected end of input"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected end of input; expected '[', '{', or a literal"); } SECTION("issue #367 - behavior of operator>> should more closely resemble that of built-in overloads") @@ -608,7 +608,7 @@ TEST_CASE("regression tests") json j; CHECK_THROWS_AS(ss >> j, json::parse_error); CHECK_THROWS_WITH(ss >> j, - "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected end of input"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected end of input; expected '[', '{', or a literal"); } SECTION("(whitespace)") @@ -618,7 +618,7 @@ TEST_CASE("regression tests") json j; CHECK_THROWS_AS(ss >> j, json::parse_error); CHECK_THROWS_WITH(ss >> j, - "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected end of input"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected end of input; expected '[', '{', or a literal"); } SECTION("one value") @@ -631,7 +631,7 @@ TEST_CASE("regression tests") CHECK_THROWS_AS(ss >> j, json::parse_error); CHECK_THROWS_WITH(ss >> j, - "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected end of input"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected end of input; expected '[', '{', or a literal"); } SECTION("one value + whitespace") @@ -644,7 +644,7 @@ TEST_CASE("regression tests") CHECK_THROWS_AS(ss >> j, json::parse_error); CHECK_THROWS_WITH(ss >> j, - "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected end of input"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected end of input; expected '[', '{', or a literal"); } SECTION("whitespace + one value") @@ -657,7 +657,7 @@ TEST_CASE("regression tests") CHECK_THROWS_AS(ss >> j, json::parse_error); CHECK_THROWS_WITH(ss >> j, - "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected end of input"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected end of input; expected '[', '{', or a literal"); } SECTION("three values") @@ -674,7 +674,7 @@ TEST_CASE("regression tests") CHECK_THROWS_AS(ss >> j, json::parse_error); CHECK_THROWS_WITH(ss >> j, - "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected end of input"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected end of input; expected '[', '{', or a literal"); } SECTION("literals without whitespace") @@ -693,7 +693,7 @@ TEST_CASE("regression tests") CHECK_THROWS_AS(ss >> j, json::parse_error); CHECK_THROWS_WITH(ss >> j, - "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected end of input"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected end of input; expected '[', '{', or a literal"); } SECTION("example from #529") @@ -708,7 +708,7 @@ TEST_CASE("regression tests") CHECK_THROWS_AS(ss >> j, json::parse_error); CHECK_THROWS_WITH(ss >> j, - "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected end of input"); + "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected end of input; expected '[', '{', or a literal"); } SECTION("second example from #529") From 66ffac9e8a4ee2d7210a25e0f7a9ce2e714ca814 Mon Sep 17 00:00:00 2001 From: Chocobo1 Date: Wed, 21 Jun 2017 16:17:10 +0800 Subject: [PATCH 258/530] Fix MSVC warning C4819 Replace unicode whitespaces by ASCII whitespace 0x20 --- src/json.hpp | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 517558c2..af8a48b3 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -10865,16 +10865,16 @@ class basic_json boolean | `false` | false | 0xc2 number_integer | -9223372036854775808..-2147483649 | int64 | 0xd3 number_integer | -2147483648..-32769 | int32 | 0xd2 - number_integer | -32768..-129 | int16 | 0xd1 + number_integer | -32768..-129 | int16 | 0xd1 number_integer | -128..-33 | int8 | 0xd0 number_integer | -32..-1 | negative fixint | 0xe0..0xff number_integer | 0..127 | positive fixint | 0x00..0x7f - number_integer | 128..255 | uint 8 | 0xcc + number_integer | 128..255 | uint 8 | 0xcc number_integer | 256..65535 | uint 16 | 0xcd number_integer | 65536..4294967295 | uint 32 | 0xce number_integer | 4294967296..18446744073709551615 | uint 64 | 0xcf number_unsigned | 0..127 | positive fixint | 0x00..0x7f - number_unsigned | 128..255 | uint 8 | 0xcc + number_unsigned | 128..255 | uint 8 | 0xcc number_unsigned | 256..65535 | uint 16 | 0xcd number_unsigned | 65536..4294967295 | uint 32 | 0xce number_unsigned | 4294967296..18446744073709551615 | uint 64 | 0xcf @@ -10970,7 +10970,7 @@ class basic_json map | object | 0xbb map | object | 0xbf False | `false` | 0xf4 - True | `true` | 0xf5 + True | `true` | 0xf5 Nill | `null` | 0xf6 Half-Precision Float | number_float | 0xf9 Single-Precision Float | number_float | 0xfa From 0395f3fdfb164b8a07ab287bdf3831d56aa1944e Mon Sep 17 00:00:00 2001 From: Chocobo1 Date: Wed, 21 Jun 2017 21:41:20 +0800 Subject: [PATCH 259/530] fix typo --- doc/examples/operator__notequal.cpp | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/doc/examples/operator__notequal.cpp b/doc/examples/operator__notequal.cpp index 1310a8d9..802447b0 100644 --- a/doc/examples/operator__notequal.cpp +++ b/doc/examples/operator__notequal.cpp @@ -16,8 +16,8 @@ int main() // output values and comparisons std::cout << std::boolalpha; - std::cout << array_1 << " == " << array_2 << " " << (array_1 != array_2) << '\n'; - std::cout << object_1 << " == " << object_2 << " " << (object_1 != object_2) << '\n'; - std::cout << number_1 << " == " << number_2 << " " << (number_1 != number_2) << '\n'; - std::cout << string_1 << " == " << string_2 << " " << (string_1 != string_2) << '\n'; + std::cout << array_1 << " != " << array_2 << " " << (array_1 != array_2) << '\n'; + std::cout << object_1 << " != " << object_2 << " " << (object_1 != object_2) << '\n'; + std::cout << number_1 << " != " << number_2 << " " << (number_1 != number_2) << '\n'; + std::cout << string_1 << " != " << string_2 << " " << (string_1 != string_2) << '\n'; } From 4b5433578a940b4aa54e14f19637b1f82c7220f1 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Wed, 21 Jun 2017 18:50:11 +0200 Subject: [PATCH 260/530] :white_check_mark: added progress output for Unicode tests Travis aborts tests if they do not produce output for 10 minutes. This commit shall fix this. --- src/json.hpp | 2 +- test/src/unit-unicode.cpp | 7 +++++++ 2 files changed, 8 insertions(+), 1 deletion(-) diff --git a/src/json.hpp b/src/json.hpp index 517558c2..8512c02d 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -12914,7 +12914,7 @@ scan_number_done: default: { - // we expected a value + // the last token was unexpected; we expected a value expect(lexer::token_type::literal_or_value); } } diff --git a/test/src/unit-unicode.cpp b/test/src/unit-unicode.cpp index e55a7573..70be088e 100644 --- a/test/src/unit-unicode.cpp +++ b/test/src/unit-unicode.cpp @@ -34,11 +34,18 @@ using nlohmann::json; #include +size_t calls = 0; + void check_utf8string(bool success_expected, int byte1, int byte2, int byte3, int byte4); // create and check a JSON string with up to four UTF-8 bytes void check_utf8string(bool success_expected, int byte1, int byte2 = -1, int byte3 = -1, int byte4 = -1) { + if (++calls % 100000 == 0) + { + std::cout << calls << " UTF-8 strings checked" << std::endl; + } + std::string json_string = "\""; CAPTURE(byte1); From b9a8425cc8f6a669a5b88618c3363bff5ab02ad7 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Wed, 21 Jun 2017 18:55:25 +0200 Subject: [PATCH 261/530] :memo: cleanup after #629 and #630 --- README.md | 1 + 1 file changed, 1 insertion(+) diff --git a/README.md b/README.md index 7fbcf48a..21125190 100644 --- a/README.md +++ b/README.md @@ -845,6 +845,7 @@ I deeply appreciate the help of the following people. - [Henry Lee](https://github.com/HenryRLee) fixed a warning in ICC and improved the iterator implementation. - [Vincent Thiery](https://github.com/vthiery) maintains a package for the Conan package manager. - [Steffen](https://github.com/koemeet) fixed a potential issue with MSVC and `std::min`. +- [Mike Tzou](https://github.com/Chocobo1) fixed some typos. Thanks a lot for helping out! Please [let me know](mailto:mail@nlohmann.me) if I forgot someone. From e99b783def9120b94bc7611fe6b852534e851323 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Thu, 22 Jun 2017 16:40:15 +0200 Subject: [PATCH 262/530] :hammer: simplifying scanner --- src/json.hpp | 72 +++++++++++----------------------- test/src/unit-class_parser.cpp | 22 +++++------ 2 files changed, 34 insertions(+), 60 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index d4d13b43..e53fbcf2 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -11461,17 +11461,6 @@ class basic_json return codepoint; } - /*! - @brief create diagnostic representation of a codepoint - @return string "U+XXXX" for codepoint XXXX - */ - static std::string codepoint_to_string(int codepoint) - { - std::stringstream ss; - ss << "U+" << std::setw(4) << std::uppercase << std::setfill('0') << std::hex << codepoint; - return ss.str(); - } - /*! @brief scan a string literal @@ -11497,9 +11486,7 @@ class basic_json while (true) { // get next character - get(); - - switch (current) + switch (get()) { // end of file while parsing string case std::char_traits::eof(): @@ -12396,40 +12383,24 @@ scan_number_done: return token_type::value_float; } - token_type scan_true() + /*! + @param[in] literal_text the literal text to expect + @param[in] length the length of the passed literal text + @param[in] return_type the token type to return on success + */ + token_type scan_literal(const char* literal_text, const size_t length, + token_type return_type) { - assert(current == 't'); - if (JSON_LIKELY((get() == 'r' and get() == 'u' and get() == 'e'))) + assert(current == literal_text[0]); + for (size_t i = 1; i < length; ++i) { - return token_type::literal_true; + if (JSON_UNLIKELY(get() != literal_text[i])) + { + error_message = "invalid literal"; + return token_type::parse_error; + } } - - error_message = "invalid literal; expected 'true'"; - return token_type::parse_error; - } - - token_type scan_false() - { - assert(current == 'f'); - if (JSON_LIKELY((get() == 'a' and get() == 'l' and get() == 's' and get() == 'e'))) - { - return token_type::literal_false; - } - - error_message = "invalid literal; expected 'false'"; - return token_type::parse_error; - } - - token_type scan_null() - { - assert(current == 'n'); - if (JSON_LIKELY((get() == 'u' and get() == 'l' and get() == 'l'))) - { - return token_type::literal_null; - } - - error_message = "invalid literal; expected 'null'"; - return token_type::parse_error; + return return_type; } ///////////////////// @@ -12461,6 +12432,7 @@ scan_number_done: { yytext.resize(2 * yytext.capacity(), '\0'); } + assert(yylen < yytext.size()); yytext[yylen++] = static_cast(c); } @@ -12523,7 +12495,9 @@ scan_number_done: else if ('\x00' <= c and c <= '\x1f') { // escape control characters - result += "<" + codepoint_to_string(c) + ">"; + std::stringstream ss; + ss << "(c) << ">"; + result += ss.str(); } else { @@ -12572,11 +12546,11 @@ scan_number_done: // literals case 't': - return scan_true(); + return scan_literal("true", 4, token_type::literal_true); case 'f': - return scan_false(); + return scan_literal("false", 5, token_type::literal_false); case 'n': - return scan_null(); + return scan_literal("null", 4, token_type::literal_null); // string case '\"': diff --git a/test/src/unit-class_parser.cpp b/test/src/unit-class_parser.cpp index 1cca2adf..ed55eba5 100644 --- a/test/src/unit-class_parser.cpp +++ b/test/src/unit-class_parser.cpp @@ -338,7 +338,7 @@ TEST_CASE("parser class") CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("-0e-:"))).parse(), "[json.exception.parse_error.101] parse error at 5: syntax error - invalid number; expected digit after exponent sign; last read: '-0e-:'"); CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("-0f"))).parse(), - "[json.exception.parse_error.101] parse error at 4: syntax error - invalid literal; expected 'false'; last read: '-0f'; expected end of input"); + "[json.exception.parse_error.101] parse error at 4: syntax error - invalid literal; last read: '-0f'; expected end of input"); } } } @@ -656,22 +656,22 @@ TEST_CASE("parser class") CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("nu"))).parse(), json::parse_error); CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("nul"))).parse(), json::parse_error); CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("n"))).parse(), - "[json.exception.parse_error.101] parse error at 2: syntax error - invalid literal; expected 'null'; last read: 'n'"); + "[json.exception.parse_error.101] parse error at 2: syntax error - invalid literal; last read: 'n'"); CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("nu"))).parse(), - "[json.exception.parse_error.101] parse error at 3: syntax error - invalid literal; expected 'null'; last read: 'nu'"); + "[json.exception.parse_error.101] parse error at 3: syntax error - invalid literal; last read: 'nu'"); CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("nul"))).parse(), - "[json.exception.parse_error.101] parse error at 4: syntax error - invalid literal; expected 'null'; last read: 'nul'"); + "[json.exception.parse_error.101] parse error at 4: syntax error - invalid literal; last read: 'nul'"); // unexpected end of true CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("t"))).parse(), json::parse_error); CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("tr"))).parse(), json::parse_error); CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("tru"))).parse(), json::parse_error); CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("t"))).parse(), - "[json.exception.parse_error.101] parse error at 2: syntax error - invalid literal; expected 'true'; last read: 't'"); + "[json.exception.parse_error.101] parse error at 2: syntax error - invalid literal; last read: 't'"); CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("tr"))).parse(), - "[json.exception.parse_error.101] parse error at 3: syntax error - invalid literal; expected 'true'; last read: 'tr'"); + "[json.exception.parse_error.101] parse error at 3: syntax error - invalid literal; last read: 'tr'"); CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("tru"))).parse(), - "[json.exception.parse_error.101] parse error at 4: syntax error - invalid literal; expected 'true'; last read: 'tru'"); + "[json.exception.parse_error.101] parse error at 4: syntax error - invalid literal; last read: 'tru'"); // unexpected end of false CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("f"))).parse(), json::parse_error); @@ -679,13 +679,13 @@ TEST_CASE("parser class") CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("fal"))).parse(), json::parse_error); CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("fals"))).parse(), json::parse_error); CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("f"))).parse(), - "[json.exception.parse_error.101] parse error at 2: syntax error - invalid literal; expected 'false'; last read: 'f'"); + "[json.exception.parse_error.101] parse error at 2: syntax error - invalid literal; last read: 'f'"); CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("fa"))).parse(), - "[json.exception.parse_error.101] parse error at 3: syntax error - invalid literal; expected 'false'; last read: 'fa'"); + "[json.exception.parse_error.101] parse error at 3: syntax error - invalid literal; last read: 'fa'"); CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("fal"))).parse(), - "[json.exception.parse_error.101] parse error at 4: syntax error - invalid literal; expected 'false'; last read: 'fal'"); + "[json.exception.parse_error.101] parse error at 4: syntax error - invalid literal; last read: 'fal'"); CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("fals"))).parse(), - "[json.exception.parse_error.101] parse error at 5: syntax error - invalid literal; expected 'false'; last read: 'fals'"); + "[json.exception.parse_error.101] parse error at 5: syntax error - invalid literal; last read: 'fals'"); // missing/unexpected end of array CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("["))).parse(), json::parse_error); From 32065c804598b9e46688c657e8432cc5507bc449 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Fri, 7 Jul 2017 21:16:33 +0200 Subject: [PATCH 263/530] :memo: fix for #639 Removed deprecated documentation of the comparison operator. --- README.md | 1 + src/json.hpp | 5 +---- 2 files changed, 2 insertions(+), 4 deletions(-) diff --git a/README.md b/README.md index 21125190..b3a51bdf 100644 --- a/README.md +++ b/README.md @@ -846,6 +846,7 @@ I deeply appreciate the help of the following people. - [Vincent Thiery](https://github.com/vthiery) maintains a package for the Conan package manager. - [Steffen](https://github.com/koemeet) fixed a potential issue with MSVC and `std::min`. - [Mike Tzou](https://github.com/Chocobo1) fixed some typos. +- [amrcode](https://github.com/amrcode) noted a missleading documentation about comparison of floats. Thanks a lot for helping out! Please [let me know](mailto:mail@nlohmann.me) if I forgot someone. diff --git a/src/json.hpp b/src/json.hpp index e53fbcf2..86664171 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -6273,10 +6273,7 @@ class basic_json their stored values are the same according to their respective `operator==`. - Integer and floating-point numbers are automatically converted before - comparison. Floating-point numbers are compared indirectly: two - floating-point numbers `f1` and `f2` are considered equal if neither - `f1 > f2` nor `f2 > f1` holds. Note than two NaN values are always - treated as unequal. + comparison. Note than two NaN values are always treated as unequal. - Two JSON null values are equal. @note NaN values never compare equal to themselves or to other NaN values. From 8b123107c01bd86a087dfdc9c364db8442b18ad4 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Fri, 7 Jul 2017 22:41:22 +0200 Subject: [PATCH 264/530] :hammer: fixed some pedantic GCC warnings --- Makefile | 1 + src/json.hpp | 8 +- test/src/unit-algorithms.cpp | 2 +- test/src/unit-allocator.cpp | 22 +-- test/src/unit-cbor.cpp | 38 ++-- test/src/unit-class_const_iterator.cpp | 8 +- test/src/unit-class_iterator.cpp | 8 +- test/src/unit-class_parser.cpp | 226 ++++++++++++------------ test/src/unit-constructor1.cpp | 62 +++---- test/src/unit-conversions.cpp | 92 +++++----- test/src/unit-deserialization.cpp | 46 ++--- test/src/unit-element_access1.cpp | 166 +++++++++--------- test/src/unit-element_access2.cpp | 168 +++++++++--------- test/src/unit-iterators1.cpp | 80 ++++----- test/src/unit-iterators2.cpp | 232 ++++++++++++------------- test/src/unit-json_patch.cpp | 78 ++++----- test/src/unit-json_pointer.cpp | 58 +++---- test/src/unit-modifiers.cpp | 56 +++--- test/src/unit-msgpack.cpp | 38 ++-- test/src/unit-regression.cpp | 82 ++++----- test/src/unit-testsuites.cpp | 8 +- test/src/unit-unicode.cpp | 12 +- 22 files changed, 747 insertions(+), 744 deletions(-) diff --git a/Makefile b/Makefile index 5a232cf0..95b2063b 100644 --- a/Makefile +++ b/Makefile @@ -88,6 +88,7 @@ pedantic_gcc: -Wformat-signedness \ -Wformat-truncation=2 \ -Wformat=2 \ + -Wno-ignored-qualifiers \ -Wimplicit-fallthrough=5 \ -Wlogical-op \ -Wmissing-declarations \ diff --git a/src/json.hpp b/src/json.hpp index 86664171..918a646b 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -2268,7 +2268,7 @@ class basic_json object | `{}` array | `[]` - @param[in] value_type the type of the value to create + @param[in] v the type of the value to create @complexity Constant. @@ -2277,8 +2277,8 @@ class basic_json @since version 1.0.0 */ - basic_json(const value_t value_type) - : m_type(value_type), m_value(value_type) + basic_json(const value_t v) + : m_type(v), m_value(v) { assert_invariant(); } @@ -12881,12 +12881,14 @@ scan_number_done: { // using "uninitialized" to avoid "expected" message expect(lexer::token_type::uninitialized); + break; } default: { // the last token was unexpected; we expected a value expect(lexer::token_type::literal_or_value); + break; } } diff --git a/test/src/unit-algorithms.cpp b/test/src/unit-algorithms.cpp index 494ea83e..5a106b6a 100644 --- a/test/src/unit-algorithms.cpp +++ b/test/src/unit-algorithms.cpp @@ -240,7 +240,7 @@ TEST_CASE("algorithms") SECTION("sorting an object") { json j({{"one", 1}, {"two", 2}}); - CHECK_THROWS_AS(std::sort(j.begin(), j.end()), json::invalid_iterator); + CHECK_THROWS_AS(std::sort(j.begin(), j.end()), json::invalid_iterator&); CHECK_THROWS_WITH(std::sort(j.begin(), j.end()), "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } diff --git a/test/src/unit-allocator.cpp b/test/src/unit-allocator.cpp index 903d5950..d2423a48 100644 --- a/test/src/unit-allocator.cpp +++ b/test/src/unit-allocator.cpp @@ -59,7 +59,7 @@ TEST_CASE("bad_alloc") bad_allocator>; // creating an object should throw - CHECK_THROWS_AS(bad_json(bad_json::value_t::object), std::bad_alloc); + CHECK_THROWS_AS(bad_json(bad_json::value_t::object), std::bad_alloc&); } } @@ -143,7 +143,7 @@ TEST_CASE("controlled bad_alloc") auto t = my_json::value_t::object; CHECK_NOTHROW(my_allocator_clean_up(my_json::json_value(t).object)); next_construct_fails = true; - CHECK_THROWS_AS(my_json::json_value(t), std::bad_alloc); + CHECK_THROWS_AS(my_json::json_value(t), std::bad_alloc&); next_construct_fails = false; } SECTION("array") @@ -152,7 +152,7 @@ TEST_CASE("controlled bad_alloc") auto t = my_json::value_t::array; CHECK_NOTHROW(my_allocator_clean_up(my_json::json_value(t).array)); next_construct_fails = true; - CHECK_THROWS_AS(my_json::json_value(t), std::bad_alloc); + CHECK_THROWS_AS(my_json::json_value(t), std::bad_alloc&); next_construct_fails = false; } SECTION("string") @@ -161,7 +161,7 @@ TEST_CASE("controlled bad_alloc") auto t = my_json::value_t::string; CHECK_NOTHROW(my_allocator_clean_up(my_json::json_value(t).string)); next_construct_fails = true; - CHECK_THROWS_AS(my_json::json_value(t), std::bad_alloc); + CHECK_THROWS_AS(my_json::json_value(t), std::bad_alloc&); next_construct_fails = false; } } @@ -172,7 +172,7 @@ TEST_CASE("controlled bad_alloc") my_json::string_t v("foo"); CHECK_NOTHROW(my_allocator_clean_up(my_json::json_value(v).string)); next_construct_fails = true; - CHECK_THROWS_AS(my_json::json_value(v), std::bad_alloc); + CHECK_THROWS_AS(my_json::json_value(v), std::bad_alloc&); next_construct_fails = false; } @@ -183,7 +183,7 @@ TEST_CASE("controlled bad_alloc") my_json::object_t v {{"foo", "bar"}}; CHECK_NOTHROW(my_json::json_value j(v)); next_construct_fails = true; - CHECK_THROWS_AS(my_json::json_value j(v), std::bad_alloc); + CHECK_THROWS_AS(my_json::json_value j(v), std::bad_alloc&); next_construct_fails = false; } */ @@ -194,7 +194,7 @@ TEST_CASE("controlled bad_alloc") my_json::array_t v = {"foo", "bar", "baz"}; CHECK_NOTHROW(my_json::json_value j(v)); next_construct_fails = true; - CHECK_THROWS_AS(my_json::json_value j(v), std::bad_alloc); + CHECK_THROWS_AS(my_json::json_value j(v), std::bad_alloc&); next_construct_fails = false; } */ @@ -208,7 +208,7 @@ TEST_CASE("controlled bad_alloc") std::map v {{"foo", "bar"}}; CHECK_NOTHROW(my_json(v)); next_construct_fails = true; - CHECK_THROWS_AS(my_json(v), std::bad_alloc); + CHECK_THROWS_AS(my_json(v), std::bad_alloc&); next_construct_fails = false; } @@ -218,7 +218,7 @@ TEST_CASE("controlled bad_alloc") std::vector v {"foo", "bar", "baz"}; CHECK_NOTHROW(my_json(v)); next_construct_fails = true; - CHECK_THROWS_AS(my_json(v), std::bad_alloc); + CHECK_THROWS_AS(my_json(v), std::bad_alloc&); next_construct_fails = false; } @@ -227,7 +227,7 @@ TEST_CASE("controlled bad_alloc") next_construct_fails = false; CHECK_NOTHROW(my_json("foo")); next_construct_fails = true; - CHECK_THROWS_AS(my_json("foo"), std::bad_alloc); + CHECK_THROWS_AS(my_json("foo"), std::bad_alloc&); next_construct_fails = false; } @@ -237,7 +237,7 @@ TEST_CASE("controlled bad_alloc") std::string s("foo"); CHECK_NOTHROW(my_json(s)); next_construct_fails = true; - CHECK_THROWS_AS(my_json(s), std::bad_alloc); + CHECK_THROWS_AS(my_json(s), std::bad_alloc&); next_construct_fails = false; } } diff --git a/test/src/unit-cbor.cpp b/test/src/unit-cbor.cpp index 5638d490..5ab6a2e2 100644 --- a/test/src/unit-cbor.cpp +++ b/test/src/unit-cbor.cpp @@ -1143,21 +1143,21 @@ TEST_CASE("CBOR") { SECTION("too short byte vector") { - CHECK_THROWS_AS(json::from_cbor(std::vector({0x18})), json::parse_error); - CHECK_THROWS_AS(json::from_cbor(std::vector({0x19})), json::parse_error); - CHECK_THROWS_AS(json::from_cbor(std::vector({0x19, 0x00})), json::parse_error); - CHECK_THROWS_AS(json::from_cbor(std::vector({0x1a})), json::parse_error); - CHECK_THROWS_AS(json::from_cbor(std::vector({0x1a, 0x00})), json::parse_error); - CHECK_THROWS_AS(json::from_cbor(std::vector({0x1a, 0x00, 0x00})), json::parse_error); - CHECK_THROWS_AS(json::from_cbor(std::vector({0x1a, 0x00, 0x00, 0x00})), json::parse_error); - CHECK_THROWS_AS(json::from_cbor(std::vector({0x1b})), json::parse_error); - CHECK_THROWS_AS(json::from_cbor(std::vector({0x1b, 0x00})), json::parse_error); - CHECK_THROWS_AS(json::from_cbor(std::vector({0x1b, 0x00, 0x00})), json::parse_error); - CHECK_THROWS_AS(json::from_cbor(std::vector({0x1b, 0x00, 0x00, 0x00})), json::parse_error); - CHECK_THROWS_AS(json::from_cbor(std::vector({0x1b, 0x00, 0x00, 0x00, 0x00})), json::parse_error); - CHECK_THROWS_AS(json::from_cbor(std::vector({0x1b, 0x00, 0x00, 0x00, 0x00, 0x00})), json::parse_error); - CHECK_THROWS_AS(json::from_cbor(std::vector({0x1b, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00})), json::parse_error); - CHECK_THROWS_AS(json::from_cbor(std::vector({0x1b, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00})), json::parse_error); + CHECK_THROWS_AS(json::from_cbor(std::vector({0x18})), json::parse_error&); + CHECK_THROWS_AS(json::from_cbor(std::vector({0x19})), json::parse_error&); + CHECK_THROWS_AS(json::from_cbor(std::vector({0x19, 0x00})), json::parse_error&); + CHECK_THROWS_AS(json::from_cbor(std::vector({0x1a})), json::parse_error&); + CHECK_THROWS_AS(json::from_cbor(std::vector({0x1a, 0x00})), json::parse_error&); + CHECK_THROWS_AS(json::from_cbor(std::vector({0x1a, 0x00, 0x00})), json::parse_error&); + CHECK_THROWS_AS(json::from_cbor(std::vector({0x1a, 0x00, 0x00, 0x00})), json::parse_error&); + CHECK_THROWS_AS(json::from_cbor(std::vector({0x1b})), json::parse_error&); + CHECK_THROWS_AS(json::from_cbor(std::vector({0x1b, 0x00})), json::parse_error&); + CHECK_THROWS_AS(json::from_cbor(std::vector({0x1b, 0x00, 0x00})), json::parse_error&); + CHECK_THROWS_AS(json::from_cbor(std::vector({0x1b, 0x00, 0x00, 0x00})), json::parse_error&); + CHECK_THROWS_AS(json::from_cbor(std::vector({0x1b, 0x00, 0x00, 0x00, 0x00})), json::parse_error&); + CHECK_THROWS_AS(json::from_cbor(std::vector({0x1b, 0x00, 0x00, 0x00, 0x00, 0x00})), json::parse_error&); + CHECK_THROWS_AS(json::from_cbor(std::vector({0x1b, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00})), json::parse_error&); + CHECK_THROWS_AS(json::from_cbor(std::vector({0x1b, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00})), json::parse_error&); CHECK_THROWS_WITH(json::from_cbor(std::vector({0x18})), "[json.exception.parse_error.110] parse error at 2: unexpected end of input"); @@ -1195,10 +1195,10 @@ TEST_CASE("CBOR") { SECTION("concrete examples") { - CHECK_THROWS_AS(json::from_cbor(std::vector({0x1c})), json::parse_error); + CHECK_THROWS_AS(json::from_cbor(std::vector({0x1c})), json::parse_error&); CHECK_THROWS_WITH(json::from_cbor(std::vector({0x1c})), "[json.exception.parse_error.112] parse error at 1: error reading CBOR; last byte: 0x1c"); - CHECK_THROWS_AS(json::from_cbor(std::vector({0xf8})), json::parse_error); + CHECK_THROWS_AS(json::from_cbor(std::vector({0xf8})), json::parse_error&); CHECK_THROWS_WITH(json::from_cbor(std::vector({0xf8})), "[json.exception.parse_error.112] parse error at 1: error reading CBOR; last byte: 0xf8"); } @@ -1249,14 +1249,14 @@ TEST_CASE("CBOR") 0xf8 }) { - CHECK_THROWS_AS(json::from_cbor(std::vector({static_cast(byte)})), json::parse_error); + CHECK_THROWS_AS(json::from_cbor(std::vector({static_cast(byte)})), json::parse_error&); } } } SECTION("invalid string in map") { - CHECK_THROWS_AS(json::from_cbor(std::vector({0xa1, 0xff, 0x01})), json::parse_error); + CHECK_THROWS_AS(json::from_cbor(std::vector({0xa1, 0xff, 0x01})), json::parse_error&); CHECK_THROWS_WITH(json::from_cbor(std::vector({0xa1, 0xff, 0x01})), "[json.exception.parse_error.113] parse error at 2: expected a CBOR string; last byte: 0xff"); } diff --git a/test/src/unit-class_const_iterator.cpp b/test/src/unit-class_const_iterator.cpp index 26b31265..78179882 100644 --- a/test/src/unit-class_const_iterator.cpp +++ b/test/src/unit-class_const_iterator.cpp @@ -147,7 +147,7 @@ TEST_CASE("const_iterator class") { json j(json::value_t::null); json::const_iterator it = j.cbegin(); - CHECK_THROWS_AS(*it, json::invalid_iterator); + CHECK_THROWS_AS(*it, json::invalid_iterator&); CHECK_THROWS_WITH(*it, "[json.exception.invalid_iterator.214] cannot get value"); } @@ -157,7 +157,7 @@ TEST_CASE("const_iterator class") json::const_iterator it = j.cbegin(); CHECK(*it == json(17)); it = j.cend(); - CHECK_THROWS_AS(*it, json::invalid_iterator); + CHECK_THROWS_AS(*it, json::invalid_iterator&); CHECK_THROWS_WITH(*it, "[json.exception.invalid_iterator.214] cannot get value"); } @@ -182,7 +182,7 @@ TEST_CASE("const_iterator class") { json j(json::value_t::null); json::const_iterator it = j.cbegin(); - CHECK_THROWS_AS(it->type_name(), json::invalid_iterator); + CHECK_THROWS_AS(it->type_name(), json::invalid_iterator&); CHECK_THROWS_WITH(it->type_name(), "[json.exception.invalid_iterator.214] cannot get value"); } @@ -192,7 +192,7 @@ TEST_CASE("const_iterator class") json::const_iterator it = j.cbegin(); CHECK(it->type_name() == "number"); it = j.cend(); - CHECK_THROWS_AS(it->type_name(), json::invalid_iterator); + CHECK_THROWS_AS(it->type_name(), json::invalid_iterator&); CHECK_THROWS_WITH(it->type_name(), "[json.exception.invalid_iterator.214] cannot get value"); } diff --git a/test/src/unit-class_iterator.cpp b/test/src/unit-class_iterator.cpp index 6ff3c369..7057d967 100644 --- a/test/src/unit-class_iterator.cpp +++ b/test/src/unit-class_iterator.cpp @@ -131,7 +131,7 @@ TEST_CASE("iterator class") { json j(json::value_t::null); json::iterator it = j.begin(); - CHECK_THROWS_AS(*it, json::invalid_iterator); + CHECK_THROWS_AS(*it, json::invalid_iterator&); CHECK_THROWS_WITH(*it, "[json.exception.invalid_iterator.214] cannot get value"); } @@ -141,7 +141,7 @@ TEST_CASE("iterator class") json::iterator it = j.begin(); CHECK(*it == json(17)); it = j.end(); - CHECK_THROWS_AS(*it, json::invalid_iterator); + CHECK_THROWS_AS(*it, json::invalid_iterator&); CHECK_THROWS_WITH(*it, "[json.exception.invalid_iterator.214] cannot get value"); } @@ -166,7 +166,7 @@ TEST_CASE("iterator class") { json j(json::value_t::null); json::iterator it = j.begin(); - CHECK_THROWS_AS(it->type_name(), json::invalid_iterator); + CHECK_THROWS_AS(it->type_name(), json::invalid_iterator&); CHECK_THROWS_WITH(it->type_name(), "[json.exception.invalid_iterator.214] cannot get value"); } @@ -176,7 +176,7 @@ TEST_CASE("iterator class") json::iterator it = j.begin(); CHECK(it->type_name() == "number"); it = j.end(); - CHECK_THROWS_AS(it->type_name(), json::invalid_iterator); + CHECK_THROWS_AS(it->type_name(), json::invalid_iterator&); CHECK_THROWS_WITH(it->type_name(), "[json.exception.invalid_iterator.214] cannot get value"); } diff --git a/test/src/unit-class_parser.cpp b/test/src/unit-class_parser.cpp index ed55eba5..7a8378a7 100644 --- a/test/src/unit-class_parser.cpp +++ b/test/src/unit-class_parser.cpp @@ -89,56 +89,56 @@ TEST_CASE("parser class") SECTION("errors") { // error: tab in string - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\t\""))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\t\""))).parse(), json::parse_error&); CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("\"\t\""))).parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: control character must be escaped; last read: '\"'"); // error: newline in string - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\n\""))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\r\""))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\n\""))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\r\""))).parse(), json::parse_error&); CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("\"\n\""))).parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: control character must be escaped; last read: '\"'"); CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("\"\r\""))).parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: control character must be escaped; last read: '\"'"); // error: backspace in string - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\b\""))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\b\""))).parse(), json::parse_error&); CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("\"\b\""))).parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: control character must be escaped; last read: '\"'"); // improve code coverage - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\uFF01"))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("[-4:1,]"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\uFF01"))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("[-4:1,]"))).parse(), json::parse_error&); // unescaped control characters - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x00\""))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x01\""))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x02\""))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x03\""))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x04\""))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x05\""))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x06\""))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x07\""))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x08\""))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x09\""))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x0a\""))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x0b\""))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x0c\""))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x0d\""))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x0e\""))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x0f\""))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x10\""))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x11\""))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x12\""))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x13\""))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x14\""))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x15\""))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x16\""))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x17\""))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x18\""))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x19\""))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x1a\""))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x1b\""))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x1c\""))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x1d\""))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x1e\""))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x1f\""))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x00\""))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x01\""))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x02\""))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x03\""))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x04\""))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x05\""))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x06\""))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x07\""))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x08\""))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x09\""))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x0a\""))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x0b\""))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x0c\""))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x0d\""))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x0e\""))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x0f\""))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x10\""))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x11\""))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x12\""))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x13\""))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x14\""))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x15\""))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x16\""))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x17\""))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x18\""))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x19\""))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x1a\""))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x1b\""))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x1c\""))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x1d\""))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x1e\""))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\x1f\""))).parse(), json::parse_error&); } SECTION("escaped") @@ -277,33 +277,33 @@ TEST_CASE("parser class") SECTION("overflow") { // overflows during parsing yield an exception - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("1.18973e+4932"))).parse() == json(), json::out_of_range); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("1.18973e+4932"))).parse() == json(), json::out_of_range&); CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("1.18973e+4932"))).parse() == json(), "[json.exception.out_of_range.406] number overflow parsing '1.18973e+4932'"); } SECTION("invalid numbers") { - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("01"))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("--1"))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("1."))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("1E"))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("1E-"))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("1.E1"))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("-1E"))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("-0E#"))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("-0E-#"))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("-0#"))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("-0.0:"))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("-0.0Z"))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("-0E123:"))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("-0e0-:"))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("-0e-:"))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("-0f"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("01"))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("--1"))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("1."))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("1E"))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("1E-"))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("1.E1"))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("-1E"))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("-0E#"))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("-0E-#"))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("-0#"))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("-0.0:"))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("-0.0Z"))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("-0E123:"))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("-0e0-:"))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("-0e-:"))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("-0f"))).parse(), json::parse_error&); // numbers must not begin with "+" - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("+1"))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("+0"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("+1"))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("+0"))).parse(), json::parse_error&); CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("01"))).parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected number literal; expected end of input"); @@ -608,20 +608,20 @@ TEST_CASE("parser class") SECTION("parse errors") { // unexpected end of number - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("0."))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("-"))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("--"))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("-0."))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("-."))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("-:"))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("0.:"))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("e."))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("1e."))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("1e/"))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("1e:"))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("1E."))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("1E/"))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("1E:"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("0."))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("-"))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("--"))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("-0."))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("-."))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("-:"))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("0.:"))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("e."))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("1e."))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("1e/"))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("1e:"))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("1E."))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("1E/"))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("1E:"))).parse(), json::parse_error&); CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("0."))).parse(), "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected digit after '.'; last read: '0.'"); CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("-"))).parse(), @@ -652,9 +652,9 @@ TEST_CASE("parser class") "[json.exception.parse_error.101] parse error at 3: syntax error - invalid number; expected '+', '-', or digit after exponent; last read: '1E:'"); // unexpected end of null - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("n"))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("nu"))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("nul"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("n"))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("nu"))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("nul"))).parse(), json::parse_error&); CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("n"))).parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - invalid literal; last read: 'n'"); CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("nu"))).parse(), @@ -663,9 +663,9 @@ TEST_CASE("parser class") "[json.exception.parse_error.101] parse error at 4: syntax error - invalid literal; last read: 'nul'"); // unexpected end of true - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("t"))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("tr"))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("tru"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("t"))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("tr"))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("tru"))).parse(), json::parse_error&); CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("t"))).parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - invalid literal; last read: 't'"); CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("tr"))).parse(), @@ -674,10 +674,10 @@ TEST_CASE("parser class") "[json.exception.parse_error.101] parse error at 4: syntax error - invalid literal; last read: 'tru'"); // unexpected end of false - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("f"))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("fa"))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("fal"))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("fals"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("f"))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("fa"))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("fal"))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("fals"))).parse(), json::parse_error&); CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("f"))).parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - invalid literal; last read: 'f'"); CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("fa"))).parse(), @@ -688,11 +688,11 @@ TEST_CASE("parser class") "[json.exception.parse_error.101] parse error at 5: syntax error - invalid literal; last read: 'fals'"); // missing/unexpected end of array - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("["))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("[1"))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("[1,"))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("[1,]"))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("]"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("["))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("[1"))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("[1,"))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("[1,]"))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("]"))).parse(), json::parse_error&); CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("["))).parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected end of input; expected '[', '{', or a literal"); CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("[1"))).parse(), @@ -705,12 +705,12 @@ TEST_CASE("parser class") "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected ']'; expected '[', '{', or a literal"); // missing/unexpected end of object - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("{"))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("{\"foo\""))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("{\"foo\":"))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("{\"foo\":}"))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("{\"foo\":1,}"))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("}"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("{"))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("{\"foo\""))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("{\"foo\":"))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("{\"foo\":}"))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("{\"foo\":1,}"))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("}"))).parse(), json::parse_error&); CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("{"))).parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected end of input; expected string literal"); CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("{\"foo\""))).parse(), @@ -725,16 +725,16 @@ TEST_CASE("parser class") "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected '}'; expected '[', '{', or a literal"); // missing/unexpected end of string - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\""))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\\\""))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\\u\""))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\\u0\""))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\\u01\""))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\\u012\""))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\\u"))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\\u0"))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\\u01"))).parse(), json::parse_error); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\\u012"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\""))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\\\""))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\\u\""))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\\u0\""))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\\u01\""))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\\u012\""))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\\u"))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\\u0"))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\\u01"))).parse(), json::parse_error&); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("\"\\u012"))).parse(), json::parse_error&); CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("\""))).parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - invalid string: missing closing quote; last read: '\"'"); CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("\"\\\""))).parse(), @@ -786,7 +786,7 @@ TEST_CASE("parser class") // any other combination of backslash and character is invalid default: { - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string(s.c_str()))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string(s.c_str()))).parse(), json::parse_error&); // only check error message if c is not a control character if (c > 0x1f) { @@ -862,7 +862,7 @@ TEST_CASE("parser class") else { CAPTURE(s1); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string(s1.c_str()))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string(s1.c_str()))).parse(), json::parse_error&); // only check error message if c is not a control character if (c > 0x1f) { @@ -871,7 +871,7 @@ TEST_CASE("parser class") } CAPTURE(s2); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string(s2.c_str()))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string(s2.c_str()))).parse(), json::parse_error&); // only check error message if c is not a control character if (c > 0x1f) { @@ -880,7 +880,7 @@ TEST_CASE("parser class") } CAPTURE(s3); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string(s3.c_str()))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string(s3.c_str()))).parse(), json::parse_error&); // only check error message if c is not a control character if (c > 0x1f) { @@ -889,7 +889,7 @@ TEST_CASE("parser class") } CAPTURE(s4); - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string(s4.c_str()))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string(s4.c_str()))).parse(), json::parse_error&); // only check error message if c is not a control character if (c > 0x1f) { @@ -901,13 +901,13 @@ TEST_CASE("parser class") } // missing part of a surrogate pair - CHECK_THROWS_AS(json::parse("\"\\uD80C\""), json::parse_error); + CHECK_THROWS_AS(json::parse("\"\\uD80C\""), json::parse_error&); CHECK_THROWS_WITH(json::parse("\"\\uD80C\""), "[json.exception.parse_error.101] parse error at 8: syntax error - invalid string: surrogate U+DC00..U+DFFF must be followed by U+DC00..U+DFFF; last read: '\"\\uD80C\"'"); // invalid surrogate pair - CHECK_THROWS_AS(json::parse("\"\\uD80C\\uD80C\""), json::parse_error); - CHECK_THROWS_AS(json::parse("\"\\uD80C\\u0000\""), json::parse_error); - CHECK_THROWS_AS(json::parse("\"\\uD80C\\uFFFF\""), json::parse_error); + CHECK_THROWS_AS(json::parse("\"\\uD80C\\uD80C\""), json::parse_error&); + CHECK_THROWS_AS(json::parse("\"\\uD80C\\u0000\""), json::parse_error&); + CHECK_THROWS_AS(json::parse("\"\\uD80C\\uFFFF\""), json::parse_error&); CHECK_THROWS_WITH(json::parse("\"\\uD80C\\uD80C\""), "[json.exception.parse_error.101] parse error at 13: syntax error - invalid string: surrogate U+DC00..U+DFFF must be followed by U+DC00..U+DFFF; last read: '\"\\uD80C\\uD80C'"); CHECK_THROWS_WITH(json::parse("\"\\uD80C\\u0000\""), @@ -1102,11 +1102,11 @@ TEST_CASE("parser class") SECTION("tests found by mutate++") { // test case to make sure no comma preceeds the first key - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("{,\"key\": false}"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("{,\"key\": false}"))).parse(), json::parse_error&); CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("{,\"key\": false}"))).parse(), "[json.exception.parse_error.101] parse error at 2: syntax error - unexpected ','; expected string literal"); // test case to make sure an object is properly closed - CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("[{\"key\": false true]"))).parse(), json::parse_error); + CHECK_THROWS_AS(json::parser(json::input_adapter::create(std::string("[{\"key\": false true]"))).parse(), json::parse_error&); CHECK_THROWS_WITH(json::parser(json::input_adapter::create(std::string("[{\"key\": false true]"))).parse(), "[json.exception.parse_error.101] parse error at 19: syntax error - unexpected true literal; expected '}'"); diff --git a/test/src/unit-constructor1.cpp b/test/src/unit-constructor1.cpp index 9dbf4540..7822b634 100644 --- a/test/src/unit-constructor1.cpp +++ b/test/src/unit-constructor1.cpp @@ -1017,7 +1017,7 @@ TEST_CASE("constructors") SECTION("object with error") { CHECK_THROWS_AS(json::object({ {"one", 1}, {"two", 1u}, {"three", 2.2}, {"four", false}, 13 }), - json::type_error); + json::type_error&); CHECK_THROWS_WITH(json::object({ {"one", 1}, {"two", 1u}, {"three", 2.2}, {"four", false}, 13 }), "[json.exception.type_error.301] cannot create object from initializer list"); } @@ -1091,16 +1091,16 @@ TEST_CASE("constructors") { json jobject = {{"a", "a"}, {"b", 1}, {"c", 17u}, {"d", false}, {"e", true}}; json jobject2 = {{"a", "a"}, {"b", 1}, {"c", 17u}}; - CHECK_THROWS_AS(json(jobject.begin(), jobject2.end()), json::invalid_iterator); - CHECK_THROWS_AS(json(jobject2.begin(), jobject.end()), json::invalid_iterator); + CHECK_THROWS_AS(json(jobject.begin(), jobject2.end()), json::invalid_iterator&); + CHECK_THROWS_AS(json(jobject2.begin(), jobject.end()), json::invalid_iterator&); CHECK_THROWS_WITH(json(jobject.begin(), jobject2.end()), "[json.exception.invalid_iterator.201] iterators are not compatible"); CHECK_THROWS_WITH(json(jobject2.begin(), jobject.end()), "[json.exception.invalid_iterator.201] iterators are not compatible"); } { json jobject = {{"a", "a"}, {"b", 1}, {"c", 17u}, {"d", false}, {"e", true}}; json jobject2 = {{"a", "a"}, {"b", 1}, {"c", 17u}}; - CHECK_THROWS_AS(json(jobject.cbegin(), jobject2.cend()), json::invalid_iterator); - CHECK_THROWS_AS(json(jobject2.cbegin(), jobject.cend()), json::invalid_iterator); + CHECK_THROWS_AS(json(jobject.cbegin(), jobject2.cend()), json::invalid_iterator&); + CHECK_THROWS_AS(json(jobject2.cbegin(), jobject.cend()), json::invalid_iterator&); CHECK_THROWS_WITH(json(jobject.cbegin(), jobject2.cend()), "[json.exception.invalid_iterator.201] iterators are not compatible"); CHECK_THROWS_WITH(json(jobject2.cbegin(), jobject.cend()), "[json.exception.invalid_iterator.201] iterators are not compatible"); } @@ -1156,16 +1156,16 @@ TEST_CASE("constructors") { json jarray = {1, 2, 3, 4}; json jarray2 = {2, 3, 4, 5}; - CHECK_THROWS_AS(json(jarray.begin(), jarray2.end()), json::invalid_iterator); - CHECK_THROWS_AS(json(jarray2.begin(), jarray.end()), json::invalid_iterator); + CHECK_THROWS_AS(json(jarray.begin(), jarray2.end()), json::invalid_iterator&); + CHECK_THROWS_AS(json(jarray2.begin(), jarray.end()), json::invalid_iterator&); CHECK_THROWS_WITH(json(jarray.begin(), jarray2.end()), "[json.exception.invalid_iterator.201] iterators are not compatible"); CHECK_THROWS_WITH(json(jarray2.begin(), jarray.end()), "[json.exception.invalid_iterator.201] iterators are not compatible"); } { json jarray = {1, 2, 3, 4}; json jarray2 = {2, 3, 4, 5}; - CHECK_THROWS_AS(json(jarray.cbegin(), jarray2.cend()), json::invalid_iterator); - CHECK_THROWS_AS(json(jarray2.cbegin(), jarray.cend()), json::invalid_iterator); + CHECK_THROWS_AS(json(jarray.cbegin(), jarray2.cend()), json::invalid_iterator&); + CHECK_THROWS_AS(json(jarray2.cbegin(), jarray.cend()), json::invalid_iterator&); CHECK_THROWS_WITH(json(jarray.cbegin(), jarray2.cend()), "[json.exception.invalid_iterator.201] iterators are not compatible"); CHECK_THROWS_WITH(json(jarray2.cbegin(), jarray.cend()), "[json.exception.invalid_iterator.201] iterators are not compatible"); } @@ -1180,13 +1180,13 @@ TEST_CASE("constructors") { { json j; - CHECK_THROWS_AS(json(j.begin(), j.end()), json::invalid_iterator); + CHECK_THROWS_AS(json(j.begin(), j.end()), json::invalid_iterator&); CHECK_THROWS_WITH(json(j.begin(), j.end()), "[json.exception.invalid_iterator.206] cannot construct with iterators from null"); } { json j; - CHECK_THROWS_AS(json(j.cbegin(), j.cend()), json::invalid_iterator); + CHECK_THROWS_AS(json(j.cbegin(), j.cend()), json::invalid_iterator&); CHECK_THROWS_WITH(json(j.cbegin(), j.cend()), "[json.exception.invalid_iterator.206] cannot construct with iterators from null"); } @@ -1269,15 +1269,15 @@ TEST_CASE("constructors") { { json j = "foo"; - CHECK_THROWS_AS(json(j.end(), j.end()), json::invalid_iterator); - CHECK_THROWS_AS(json(j.begin(), j.begin()), json::invalid_iterator); + CHECK_THROWS_AS(json(j.end(), j.end()), json::invalid_iterator&); + CHECK_THROWS_AS(json(j.begin(), j.begin()), json::invalid_iterator&); CHECK_THROWS_WITH(json(j.end(), j.end()), "[json.exception.invalid_iterator.204] iterators out of range"); CHECK_THROWS_WITH(json(j.begin(), j.begin()), "[json.exception.invalid_iterator.204] iterators out of range"); } { json j = "bar"; - CHECK_THROWS_AS(json(j.cend(), j.cend()), json::invalid_iterator); - CHECK_THROWS_AS(json(j.cbegin(), j.cbegin()), json::invalid_iterator); + CHECK_THROWS_AS(json(j.cend(), j.cend()), json::invalid_iterator&); + CHECK_THROWS_AS(json(j.cbegin(), j.cbegin()), json::invalid_iterator&); CHECK_THROWS_WITH(json(j.cend(), j.cend()), "[json.exception.invalid_iterator.204] iterators out of range"); CHECK_THROWS_WITH(json(j.cbegin(), j.cbegin()), "[json.exception.invalid_iterator.204] iterators out of range"); } @@ -1287,15 +1287,15 @@ TEST_CASE("constructors") { { json j = false; - CHECK_THROWS_AS(json(j.end(), j.end()), json::invalid_iterator); - CHECK_THROWS_AS(json(j.begin(), j.begin()), json::invalid_iterator); + CHECK_THROWS_AS(json(j.end(), j.end()), json::invalid_iterator&); + CHECK_THROWS_AS(json(j.begin(), j.begin()), json::invalid_iterator&); CHECK_THROWS_WITH(json(j.end(), j.end()), "[json.exception.invalid_iterator.204] iterators out of range"); CHECK_THROWS_WITH(json(j.begin(), j.begin()), "[json.exception.invalid_iterator.204] iterators out of range"); } { json j = true; - CHECK_THROWS_AS(json(j.cend(), j.cend()), json::invalid_iterator); - CHECK_THROWS_AS(json(j.cbegin(), j.cbegin()), json::invalid_iterator); + CHECK_THROWS_AS(json(j.cend(), j.cend()), json::invalid_iterator&); + CHECK_THROWS_AS(json(j.cbegin(), j.cbegin()), json::invalid_iterator&); CHECK_THROWS_WITH(json(j.cend(), j.cend()), "[json.exception.invalid_iterator.204] iterators out of range"); CHECK_THROWS_WITH(json(j.cbegin(), j.cbegin()), "[json.exception.invalid_iterator.204] iterators out of range"); } @@ -1305,15 +1305,15 @@ TEST_CASE("constructors") { { json j = 17; - CHECK_THROWS_AS(json(j.end(), j.end()), json::invalid_iterator); - CHECK_THROWS_AS(json(j.begin(), j.begin()), json::invalid_iterator); + CHECK_THROWS_AS(json(j.end(), j.end()), json::invalid_iterator&); + CHECK_THROWS_AS(json(j.begin(), j.begin()), json::invalid_iterator&); CHECK_THROWS_WITH(json(j.end(), j.end()), "[json.exception.invalid_iterator.204] iterators out of range"); CHECK_THROWS_WITH(json(j.begin(), j.begin()), "[json.exception.invalid_iterator.204] iterators out of range"); } { json j = 17; - CHECK_THROWS_AS(json(j.cend(), j.cend()), json::invalid_iterator); - CHECK_THROWS_AS(json(j.cbegin(), j.cbegin()), json::invalid_iterator); + CHECK_THROWS_AS(json(j.cend(), j.cend()), json::invalid_iterator&); + CHECK_THROWS_AS(json(j.cbegin(), j.cbegin()), json::invalid_iterator&); CHECK_THROWS_WITH(json(j.cend(), j.cend()), "[json.exception.invalid_iterator.204] iterators out of range"); CHECK_THROWS_WITH(json(j.cbegin(), j.cbegin()), "[json.exception.invalid_iterator.204] iterators out of range"); } @@ -1323,15 +1323,15 @@ TEST_CASE("constructors") { { json j = 17u; - CHECK_THROWS_AS(json(j.end(), j.end()), json::invalid_iterator); - CHECK_THROWS_AS(json(j.begin(), j.begin()), json::invalid_iterator); + CHECK_THROWS_AS(json(j.end(), j.end()), json::invalid_iterator&); + CHECK_THROWS_AS(json(j.begin(), j.begin()), json::invalid_iterator&); CHECK_THROWS_WITH(json(j.end(), j.end()), "[json.exception.invalid_iterator.204] iterators out of range"); CHECK_THROWS_WITH(json(j.begin(), j.begin()), "[json.exception.invalid_iterator.204] iterators out of range"); } { json j = 17u; - CHECK_THROWS_AS(json(j.cend(), j.cend()), json::invalid_iterator); - CHECK_THROWS_AS(json(j.cbegin(), j.cbegin()), json::invalid_iterator); + CHECK_THROWS_AS(json(j.cend(), j.cend()), json::invalid_iterator&); + CHECK_THROWS_AS(json(j.cbegin(), j.cbegin()), json::invalid_iterator&); CHECK_THROWS_WITH(json(j.cend(), j.cend()), "[json.exception.invalid_iterator.204] iterators out of range"); CHECK_THROWS_WITH(json(j.cbegin(), j.cbegin()), "[json.exception.invalid_iterator.204] iterators out of range"); } @@ -1341,15 +1341,15 @@ TEST_CASE("constructors") { { json j = 23.42; - CHECK_THROWS_AS(json(j.end(), j.end()), json::invalid_iterator); - CHECK_THROWS_AS(json(j.begin(), j.begin()), json::invalid_iterator); + CHECK_THROWS_AS(json(j.end(), j.end()), json::invalid_iterator&); + CHECK_THROWS_AS(json(j.begin(), j.begin()), json::invalid_iterator&); CHECK_THROWS_WITH(json(j.end(), j.end()), "[json.exception.invalid_iterator.204] iterators out of range"); CHECK_THROWS_WITH(json(j.begin(), j.begin()), "[json.exception.invalid_iterator.204] iterators out of range"); } { json j = 23.42; - CHECK_THROWS_AS(json(j.cend(), j.cend()), json::invalid_iterator); - CHECK_THROWS_AS(json(j.cbegin(), j.cbegin()), json::invalid_iterator); + CHECK_THROWS_AS(json(j.cend(), j.cend()), json::invalid_iterator&); + CHECK_THROWS_AS(json(j.cbegin(), j.cbegin()), json::invalid_iterator&); CHECK_THROWS_WITH(json(j.cend(), j.cend()), "[json.exception.invalid_iterator.204] iterators out of range"); CHECK_THROWS_WITH(json(j.cbegin(), j.cbegin()), "[json.exception.invalid_iterator.204] iterators out of range"); } diff --git a/test/src/unit-conversions.cpp b/test/src/unit-conversions.cpp index 0940ec51..59c8e53f 100644 --- a/test/src/unit-conversions.cpp +++ b/test/src/unit-conversions.cpp @@ -78,13 +78,13 @@ TEST_CASE("value conversion") SECTION("exception in case of a non-object type") { - CHECK_THROWS_AS(json(json::value_t::null).get(), json::type_error); - CHECK_THROWS_AS(json(json::value_t::array).get(), json::type_error); - CHECK_THROWS_AS(json(json::value_t::string).get(), json::type_error); - CHECK_THROWS_AS(json(json::value_t::boolean).get(), json::type_error); - CHECK_THROWS_AS(json(json::value_t::number_integer).get(), json::type_error); - CHECK_THROWS_AS(json(json::value_t::number_unsigned).get(), json::type_error); - CHECK_THROWS_AS(json(json::value_t::number_float).get(), json::type_error); + CHECK_THROWS_AS(json(json::value_t::null).get(), json::type_error&); + CHECK_THROWS_AS(json(json::value_t::array).get(), json::type_error&); + CHECK_THROWS_AS(json(json::value_t::string).get(), json::type_error&); + CHECK_THROWS_AS(json(json::value_t::boolean).get(), json::type_error&); + CHECK_THROWS_AS(json(json::value_t::number_integer).get(), json::type_error&); + CHECK_THROWS_AS(json(json::value_t::number_unsigned).get(), json::type_error&); + CHECK_THROWS_AS(json(json::value_t::number_float).get(), json::type_error&); CHECK_THROWS_WITH(json(json::value_t::null).get(), "[json.exception.type_error.302] type must be object, but is null"); @@ -161,7 +161,7 @@ TEST_CASE("value conversion") std::forward_list a = j.get>(); CHECK(json(a) == j); - CHECK_THROWS_AS(json(json::value_t::null).get>(), json::type_error); + CHECK_THROWS_AS(json(json::value_t::null).get>(), json::type_error&); CHECK_THROWS_WITH(json(json::value_t::null).get>(), "[json.exception.type_error.302] type must be array, but is null"); } @@ -171,7 +171,7 @@ TEST_CASE("value conversion") std::vector a = j.get>(); CHECK(json(a) == j); - CHECK_THROWS_AS(json(json::value_t::null).get>(), json::type_error); + CHECK_THROWS_AS(json(json::value_t::null).get>(), json::type_error&); CHECK_THROWS_WITH(json(json::value_t::null).get>(), "[json.exception.type_error.302] type must be array, but is null"); @@ -180,7 +180,7 @@ TEST_CASE("value conversion") { // making the call to from_json throw in order to check capacity std::vector v; - CHECK_THROWS_AS(nlohmann::from_json(j, v), json::type_error); + CHECK_THROWS_AS(nlohmann::from_json(j, v), json::type_error&); CHECK(v.capacity() == j.size()); // make sure all values are properly copied @@ -212,13 +212,13 @@ TEST_CASE("value conversion") SECTION("exception in case of a non-array type") { - CHECK_THROWS_AS(json(json::value_t::null).get(), json::type_error); - CHECK_THROWS_AS(json(json::value_t::object).get(), json::type_error); - CHECK_THROWS_AS(json(json::value_t::string).get(), json::type_error); - CHECK_THROWS_AS(json(json::value_t::boolean).get(), json::type_error); - CHECK_THROWS_AS(json(json::value_t::number_integer).get(), json::type_error); - CHECK_THROWS_AS(json(json::value_t::number_unsigned).get(), json::type_error); - CHECK_THROWS_AS(json(json::value_t::number_float).get(), json::type_error); + CHECK_THROWS_AS(json(json::value_t::null).get(), json::type_error&); + CHECK_THROWS_AS(json(json::value_t::object).get(), json::type_error&); + CHECK_THROWS_AS(json(json::value_t::string).get(), json::type_error&); + CHECK_THROWS_AS(json(json::value_t::boolean).get(), json::type_error&); + CHECK_THROWS_AS(json(json::value_t::number_integer).get(), json::type_error&); + CHECK_THROWS_AS(json(json::value_t::number_unsigned).get(), json::type_error&); + CHECK_THROWS_AS(json(json::value_t::number_float).get(), json::type_error&); CHECK_THROWS_WITH(json(json::value_t::object).get>(), "[json.exception.type_error.302] type must be array, but is object"); @@ -294,13 +294,13 @@ TEST_CASE("value conversion") SECTION("exception in case of a non-string type") { - CHECK_THROWS_AS(json(json::value_t::null).get(), json::type_error); - CHECK_THROWS_AS(json(json::value_t::object).get(), json::type_error); - CHECK_THROWS_AS(json(json::value_t::array).get(), json::type_error); - CHECK_THROWS_AS(json(json::value_t::boolean).get(), json::type_error); - CHECK_THROWS_AS(json(json::value_t::number_integer).get(), json::type_error); - CHECK_THROWS_AS(json(json::value_t::number_unsigned).get(), json::type_error); - CHECK_THROWS_AS(json(json::value_t::number_float).get(), json::type_error); + CHECK_THROWS_AS(json(json::value_t::null).get(), json::type_error&); + CHECK_THROWS_AS(json(json::value_t::object).get(), json::type_error&); + CHECK_THROWS_AS(json(json::value_t::array).get(), json::type_error&); + CHECK_THROWS_AS(json(json::value_t::boolean).get(), json::type_error&); + CHECK_THROWS_AS(json(json::value_t::number_integer).get(), json::type_error&); + CHECK_THROWS_AS(json(json::value_t::number_unsigned).get(), json::type_error&); + CHECK_THROWS_AS(json(json::value_t::number_float).get(), json::type_error&); CHECK_THROWS_WITH(json(json::value_t::null).get(), "[json.exception.type_error.302] type must be string, but is null"); @@ -356,13 +356,13 @@ TEST_CASE("value conversion") SECTION("exception in case of a non-string type") { - CHECK_THROWS_AS(json(json::value_t::null).get(), json::type_error); - CHECK_THROWS_AS(json(json::value_t::object).get(), json::type_error); - CHECK_THROWS_AS(json(json::value_t::array).get(), json::type_error); - CHECK_THROWS_AS(json(json::value_t::string).get(), json::type_error); - CHECK_THROWS_AS(json(json::value_t::number_integer).get(), json::type_error); - CHECK_THROWS_AS(json(json::value_t::number_unsigned).get(), json::type_error); - CHECK_THROWS_AS(json(json::value_t::number_float).get(), json::type_error); + CHECK_THROWS_AS(json(json::value_t::null).get(), json::type_error&); + CHECK_THROWS_AS(json(json::value_t::object).get(), json::type_error&); + CHECK_THROWS_AS(json(json::value_t::array).get(), json::type_error&); + CHECK_THROWS_AS(json(json::value_t::string).get(), json::type_error&); + CHECK_THROWS_AS(json(json::value_t::number_integer).get(), json::type_error&); + CHECK_THROWS_AS(json(json::value_t::number_unsigned).get(), json::type_error&); + CHECK_THROWS_AS(json(json::value_t::number_float).get(), json::type_error&); CHECK_THROWS_WITH(json(json::value_t::null).get(), "[json.exception.type_error.302] type must be boolean, but is null"); @@ -612,11 +612,11 @@ TEST_CASE("value conversion") SECTION("exception in case of a non-number type") { - CHECK_THROWS_AS(json(json::value_t::null).get(), json::type_error); - CHECK_THROWS_AS(json(json::value_t::object).get(), json::type_error); - CHECK_THROWS_AS(json(json::value_t::array).get(), json::type_error); - CHECK_THROWS_AS(json(json::value_t::string).get(), json::type_error); - CHECK_THROWS_AS(json(json::value_t::boolean).get(), json::type_error); + CHECK_THROWS_AS(json(json::value_t::null).get(), json::type_error&); + CHECK_THROWS_AS(json(json::value_t::object).get(), json::type_error&); + CHECK_THROWS_AS(json(json::value_t::array).get(), json::type_error&); + CHECK_THROWS_AS(json(json::value_t::string).get(), json::type_error&); + CHECK_THROWS_AS(json(json::value_t::boolean).get(), json::type_error&); CHECK_THROWS_WITH(json(json::value_t::null).get(), "[json.exception.type_error.302] type must be number, but is null"); @@ -871,11 +871,11 @@ TEST_CASE("value conversion") SECTION("exception in case of a non-string type") { - CHECK_THROWS_AS(json(json::value_t::null).get(), json::type_error); - CHECK_THROWS_AS(json(json::value_t::object).get(), json::type_error); - CHECK_THROWS_AS(json(json::value_t::array).get(), json::type_error); - CHECK_THROWS_AS(json(json::value_t::string).get(), json::type_error); - CHECK_THROWS_AS(json(json::value_t::boolean).get(), json::type_error); + CHECK_THROWS_AS(json(json::value_t::null).get(), json::type_error&); + CHECK_THROWS_AS(json(json::value_t::object).get(), json::type_error&); + CHECK_THROWS_AS(json(json::value_t::array).get(), json::type_error&); + CHECK_THROWS_AS(json(json::value_t::string).get(), json::type_error&); + CHECK_THROWS_AS(json(json::value_t::boolean).get(), json::type_error&); CHECK_THROWS_WITH(json(json::value_t::null).get(), "[json.exception.type_error.302] type must be number, but is null"); @@ -977,7 +977,7 @@ TEST_CASE("value conversion") SECTION("exception in case of a non-object type") { - CHECK_THROWS_AS((json().get>()), json::type_error); + CHECK_THROWS_AS((json().get>()), json::type_error&); CHECK_THROWS_WITH((json().get>()), "[json.exception.type_error.302] type must be object, but is null"); } } @@ -1046,10 +1046,10 @@ TEST_CASE("value conversion") SECTION("exception in case of a non-object type") { - CHECK_THROWS_AS((json().get>()), json::type_error); - CHECK_THROWS_AS((json().get>()), json::type_error); - CHECK_THROWS_AS((json().get>()), json::type_error); - CHECK_THROWS_AS((json().get>()), json::type_error); + CHECK_THROWS_AS((json().get>()), json::type_error&); + CHECK_THROWS_AS((json().get>()), json::type_error&); + CHECK_THROWS_AS((json().get>()), json::type_error&); + CHECK_THROWS_AS((json().get>()), json::type_error&); // does type really must be an array? or it rather must not be null? // that's what I thought when other test like this one broke diff --git a/test/src/unit-deserialization.cpp b/test/src/unit-deserialization.cpp index 48ec91c0..3de8f156 100644 --- a/test/src/unit-deserialization.cpp +++ b/test/src/unit-deserialization.cpp @@ -95,7 +95,7 @@ TEST_CASE("deserialization") ss1 << "[\"foo\",1,2,3,false,{\"one\":1}"; ss2 << "[\"foo\",1,2,3,false,{\"one\":1}"; ss3 << "[\"foo\",1,2,3,false,{\"one\":1}"; - CHECK_THROWS_AS(json::parse(ss1), json::parse_error); + CHECK_THROWS_AS(json::parse(ss1), json::parse_error&); CHECK_THROWS_WITH(json::parse(ss2), "[json.exception.parse_error.101] parse error at 29: syntax error - unexpected end of input; expected ']'"); CHECK(not json::accept(ss3)); @@ -104,7 +104,7 @@ TEST_CASE("deserialization") SECTION("string") { json::string_t s = "[\"foo\",1,2,3,false,{\"one\":1}"; - CHECK_THROWS_AS(json::parse(s), json::parse_error); + CHECK_THROWS_AS(json::parse(s), json::parse_error&); CHECK_THROWS_WITH(json::parse(s), "[json.exception.parse_error.101] parse error at 29: syntax error - unexpected end of input; expected ']'"); CHECK(not json::accept(s)); @@ -116,7 +116,7 @@ TEST_CASE("deserialization") ss1 << "[\"foo\",1,2,3,false,{\"one\":1}"; ss2 << "[\"foo\",1,2,3,false,{\"one\":1}"; json j; - CHECK_THROWS_AS(j << ss1, json::parse_error); + CHECK_THROWS_AS(j << ss1, json::parse_error&); CHECK_THROWS_WITH(j << ss2, "[json.exception.parse_error.101] parse error at 29: syntax error - unexpected end of input; expected ']'"); } @@ -127,14 +127,14 @@ TEST_CASE("deserialization") ss1 << "[\"foo\",1,2,3,false,{\"one\":1}"; ss2 << "[\"foo\",1,2,3,false,{\"one\":1}"; json j; - CHECK_THROWS_AS(ss1 >> j, json::parse_error); + CHECK_THROWS_AS(ss1 >> j, json::parse_error&); CHECK_THROWS_WITH(ss2 >> j, "[json.exception.parse_error.101] parse error at 29: syntax error - unexpected end of input; expected ']'"); } SECTION("user-defined string literal") { - CHECK_THROWS_AS("[\"foo\",1,2,3,false,{\"one\":1}"_json, json::parse_error); + CHECK_THROWS_AS("[\"foo\",1,2,3,false,{\"one\":1}"_json, json::parse_error&); CHECK_THROWS_WITH("[\"foo\",1,2,3,false,{\"one\":1}"_json, "[json.exception.parse_error.101] parse error at 29: syntax error - unexpected end of input; expected ']'"); } @@ -195,7 +195,7 @@ TEST_CASE("deserialization") SECTION("empty container") { std::vector v; - CHECK_THROWS_AS(json::parse(v), json::parse_error); + CHECK_THROWS_AS(json::parse(v), json::parse_error&); CHECK(not json::accept(v)); } } @@ -247,7 +247,7 @@ TEST_CASE("deserialization") SECTION("with empty range") { std::vector v; - CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); + CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error&); CHECK(not json::accept(std::begin(v), std::end(v))); } } @@ -258,112 +258,112 @@ TEST_CASE("deserialization") SECTION("case 1") { uint8_t v[] = {'\"', 'a', 'a', 'a', 'a', 'a', 'a', '\\', 'u'}; - CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); + CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error&); CHECK(not json::accept(std::begin(v), std::end(v))); } SECTION("case 2") { uint8_t v[] = {'\"', 'a', 'a', 'a', 'a', 'a', 'a', '\\', 'u', '1'}; - CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); + CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error&); CHECK(not json::accept(std::begin(v), std::end(v))); } SECTION("case 3") { uint8_t v[] = {'\"', 'a', 'a', 'a', 'a', 'a', 'a', '\\', 'u', '1', '1', '1', '1', '1', '1', '1', '1'}; - CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); + CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error&); CHECK(not json::accept(std::begin(v), std::end(v))); } SECTION("case 4") { uint8_t v[] = {'\"', 'a', 'a', 'a', 'a', 'a', 'a', 'u', '1', '1', '1', '1', '1', '1', '1', '1', '\\'}; - CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); + CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error&); CHECK(not json::accept(std::begin(v), std::end(v))); } SECTION("case 5") { uint8_t v[] = {'\"', 0x7F, 0xC1}; - CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); + CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error&); CHECK(not json::accept(std::begin(v), std::end(v))); } SECTION("case 6") { uint8_t v[] = {'\"', 0x7F, 0xDF, 0x7F}; - CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); + CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error&); CHECK(not json::accept(std::begin(v), std::end(v))); } SECTION("case 7") { uint8_t v[] = {'\"', 0x7F, 0xDF, 0xC0}; - CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); + CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error&); CHECK(not json::accept(std::begin(v), std::end(v))); } SECTION("case 8") { uint8_t v[] = {'\"', 0x7F, 0xE0, 0x9F}; - CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); + CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error&); CHECK(not json::accept(std::begin(v), std::end(v))); } SECTION("case 9") { uint8_t v[] = {'\"', 0x7F, 0xEF, 0xC0}; - CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); + CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error&); CHECK(not json::accept(std::begin(v), std::end(v))); } SECTION("case 10") { uint8_t v[] = {'\"', 0x7F, 0xED, 0x7F}; - CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); + CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error&); CHECK(not json::accept(std::begin(v), std::end(v))); } SECTION("case 11") { uint8_t v[] = {'\"', 0x7F, 0xF0, 0x8F}; - CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); + CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error&); CHECK(not json::accept(std::begin(v), std::end(v))); } SECTION("case 12") { uint8_t v[] = {'\"', 0x7F, 0xF0, 0xC0}; - CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); + CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error&); CHECK(not json::accept(std::begin(v), std::end(v))); } SECTION("case 13") { uint8_t v[] = {'\"', 0x7F, 0xF3, 0x7F}; - CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); + CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error&); CHECK(not json::accept(std::begin(v), std::end(v))); } SECTION("case 14") { uint8_t v[] = {'\"', 0x7F, 0xF3, 0xC0}; - CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); + CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error&); CHECK(not json::accept(std::begin(v), std::end(v))); } SECTION("case 15") { uint8_t v[] = {'\"', 0x7F, 0xF4, 0x7F}; - CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); + CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error&); CHECK(not json::accept(std::begin(v), std::end(v))); } SECTION("case 16") { uint8_t v[] = {'{', '\"', '\"', ':', '1', '1'}; - CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error); + CHECK_THROWS_AS(json::parse(std::begin(v), std::end(v)), json::parse_error&); CHECK(not json::accept(std::begin(v), std::end(v))); } } diff --git a/test/src/unit-element_access1.cpp b/test/src/unit-element_access1.cpp index eb4a92cb..630b1a5e 100644 --- a/test/src/unit-element_access1.cpp +++ b/test/src/unit-element_access1.cpp @@ -63,8 +63,8 @@ TEST_CASE("element access 1") SECTION("access outside bounds") { - CHECK_THROWS_AS(j.at(8), json::out_of_range); - CHECK_THROWS_AS(j_const.at(8), json::out_of_range); + CHECK_THROWS_AS(j.at(8), json::out_of_range&); + CHECK_THROWS_AS(j_const.at(8), json::out_of_range&); CHECK_THROWS_WITH(j.at(8), "[json.exception.out_of_range.401] array index 8 is out of range"); @@ -78,8 +78,8 @@ TEST_CASE("element access 1") { json j_nonarray(json::value_t::null); const json j_nonarray_const(j_nonarray); - CHECK_THROWS_AS(j_nonarray.at(0), json::type_error); - CHECK_THROWS_AS(j_nonarray_const.at(0), json::type_error); + CHECK_THROWS_AS(j_nonarray.at(0), json::type_error&); + CHECK_THROWS_AS(j_nonarray_const.at(0), json::type_error&); CHECK_THROWS_WITH(j_nonarray.at(0), "[json.exception.type_error.304] cannot use at() with null"); CHECK_THROWS_WITH(j_nonarray_const.at(0), "[json.exception.type_error.304] cannot use at() with null"); @@ -89,8 +89,8 @@ TEST_CASE("element access 1") { json j_nonarray(json::value_t::boolean); const json j_nonarray_const(j_nonarray); - CHECK_THROWS_AS(j_nonarray.at(0), json::type_error); - CHECK_THROWS_AS(j_nonarray_const.at(0), json::type_error); + CHECK_THROWS_AS(j_nonarray.at(0), json::type_error&); + CHECK_THROWS_AS(j_nonarray_const.at(0), json::type_error&); CHECK_THROWS_WITH(j_nonarray.at(0), "[json.exception.type_error.304] cannot use at() with boolean"); CHECK_THROWS_WITH(j_nonarray_const.at(0), "[json.exception.type_error.304] cannot use at() with boolean"); @@ -100,8 +100,8 @@ TEST_CASE("element access 1") { json j_nonarray(json::value_t::string); const json j_nonarray_const(j_nonarray); - CHECK_THROWS_AS(j_nonarray.at(0), json::type_error); - CHECK_THROWS_AS(j_nonarray_const.at(0), json::type_error); + CHECK_THROWS_AS(j_nonarray.at(0), json::type_error&); + CHECK_THROWS_AS(j_nonarray_const.at(0), json::type_error&); CHECK_THROWS_WITH(j_nonarray.at(0), "[json.exception.type_error.304] cannot use at() with string"); CHECK_THROWS_WITH(j_nonarray_const.at(0), "[json.exception.type_error.304] cannot use at() with string"); @@ -111,8 +111,8 @@ TEST_CASE("element access 1") { json j_nonarray(json::value_t::object); const json j_nonarray_const(j_nonarray); - CHECK_THROWS_AS(j_nonarray.at(0), json::type_error); - CHECK_THROWS_AS(j_nonarray_const.at(0), json::type_error); + CHECK_THROWS_AS(j_nonarray.at(0), json::type_error&); + CHECK_THROWS_AS(j_nonarray_const.at(0), json::type_error&); CHECK_THROWS_WITH(j_nonarray.at(0), "[json.exception.type_error.304] cannot use at() with object"); CHECK_THROWS_WITH(j_nonarray_const.at(0), "[json.exception.type_error.304] cannot use at() with object"); @@ -122,8 +122,8 @@ TEST_CASE("element access 1") { json j_nonarray(json::value_t::number_integer); const json j_nonarray_const(j_nonarray); - CHECK_THROWS_AS(j_nonarray.at(0), json::type_error); - CHECK_THROWS_AS(j_nonarray_const.at(0), json::type_error); + CHECK_THROWS_AS(j_nonarray.at(0), json::type_error&); + CHECK_THROWS_AS(j_nonarray_const.at(0), json::type_error&); CHECK_THROWS_WITH(j_nonarray.at(0), "[json.exception.type_error.304] cannot use at() with number"); CHECK_THROWS_WITH(j_nonarray_const.at(0), "[json.exception.type_error.304] cannot use at() with number"); @@ -133,8 +133,8 @@ TEST_CASE("element access 1") { json j_nonarray(json::value_t::number_unsigned); const json j_nonarray_const(j_nonarray); - CHECK_THROWS_AS(j_nonarray.at(0), json::type_error); - CHECK_THROWS_AS(j_nonarray_const.at(0), json::type_error); + CHECK_THROWS_AS(j_nonarray.at(0), json::type_error&); + CHECK_THROWS_AS(j_nonarray_const.at(0), json::type_error&); CHECK_THROWS_WITH(j_nonarray.at(0), "[json.exception.type_error.304] cannot use at() with number"); CHECK_THROWS_WITH(j_nonarray_const.at(0), "[json.exception.type_error.304] cannot use at() with number"); @@ -144,8 +144,8 @@ TEST_CASE("element access 1") { json j_nonarray(json::value_t::number_float); const json j_nonarray_const(j_nonarray); - CHECK_THROWS_AS(j_nonarray.at(0), json::type_error); - CHECK_THROWS_AS(j_nonarray_const.at(0), json::type_error); + CHECK_THROWS_AS(j_nonarray.at(0), json::type_error&); + CHECK_THROWS_AS(j_nonarray_const.at(0), json::type_error&); CHECK_THROWS_WITH(j_nonarray.at(0), "[json.exception.type_error.304] cannot use at() with number"); CHECK_THROWS_WITH(j_nonarray_const.at(0), "[json.exception.type_error.304] cannot use at() with number"); @@ -193,7 +193,7 @@ TEST_CASE("element access 1") json j_nonarray(json::value_t::null); const json j_nonarray_const(j_nonarray); CHECK_NOTHROW(j_nonarray[0]); - CHECK_THROWS_AS(j_nonarray_const[0], json::type_error); + CHECK_THROWS_AS(j_nonarray_const[0], json::type_error&); CHECK_THROWS_WITH(j_nonarray_const[0], "[json.exception.type_error.305] cannot use operator[] with null"); } @@ -209,8 +209,8 @@ TEST_CASE("element access 1") { json j_nonarray(json::value_t::boolean); const json j_nonarray_const(j_nonarray); - CHECK_THROWS_AS(j_nonarray[0], json::type_error); - CHECK_THROWS_AS(j_nonarray_const[0], json::type_error); + CHECK_THROWS_AS(j_nonarray[0], json::type_error&); + CHECK_THROWS_AS(j_nonarray_const[0], json::type_error&); CHECK_THROWS_WITH(j_nonarray[0], "[json.exception.type_error.305] cannot use operator[] with boolean"); CHECK_THROWS_WITH(j_nonarray_const[0], "[json.exception.type_error.305] cannot use operator[] with boolean"); } @@ -219,8 +219,8 @@ TEST_CASE("element access 1") { json j_nonarray(json::value_t::string); const json j_nonarray_const(j_nonarray); - CHECK_THROWS_AS(j_nonarray[0], json::type_error); - CHECK_THROWS_AS(j_nonarray_const[0], json::type_error); + CHECK_THROWS_AS(j_nonarray[0], json::type_error&); + CHECK_THROWS_AS(j_nonarray_const[0], json::type_error&); CHECK_THROWS_WITH(j_nonarray[0], "[json.exception.type_error.305] cannot use operator[] with string"); CHECK_THROWS_WITH(j_nonarray_const[0], "[json.exception.type_error.305] cannot use operator[] with string"); } @@ -229,8 +229,8 @@ TEST_CASE("element access 1") { json j_nonarray(json::value_t::object); const json j_nonarray_const(j_nonarray); - CHECK_THROWS_AS(j_nonarray[0], json::type_error); - CHECK_THROWS_AS(j_nonarray_const[0], json::type_error); + CHECK_THROWS_AS(j_nonarray[0], json::type_error&); + CHECK_THROWS_AS(j_nonarray_const[0], json::type_error&); CHECK_THROWS_WITH(j_nonarray[0], "[json.exception.type_error.305] cannot use operator[] with object"); CHECK_THROWS_WITH(j_nonarray_const[0], "[json.exception.type_error.305] cannot use operator[] with object"); } @@ -239,8 +239,8 @@ TEST_CASE("element access 1") { json j_nonarray(json::value_t::number_integer); const json j_nonarray_const(j_nonarray); - CHECK_THROWS_AS(j_nonarray[0], json::type_error); - CHECK_THROWS_AS(j_nonarray_const[0], json::type_error); + CHECK_THROWS_AS(j_nonarray[0], json::type_error&); + CHECK_THROWS_AS(j_nonarray_const[0], json::type_error&); CHECK_THROWS_WITH(j_nonarray[0], "[json.exception.type_error.305] cannot use operator[] with number"); CHECK_THROWS_WITH(j_nonarray_const[0], "[json.exception.type_error.305] cannot use operator[] with number"); } @@ -249,8 +249,8 @@ TEST_CASE("element access 1") { json j_nonarray(json::value_t::number_unsigned); const json j_nonarray_const(j_nonarray); - CHECK_THROWS_AS(j_nonarray[0], json::type_error); - CHECK_THROWS_AS(j_nonarray_const[0], json::type_error); + CHECK_THROWS_AS(j_nonarray[0], json::type_error&); + CHECK_THROWS_AS(j_nonarray_const[0], json::type_error&); CHECK_THROWS_WITH(j_nonarray[0], "[json.exception.type_error.305] cannot use operator[] with number"); CHECK_THROWS_WITH(j_nonarray_const[0], "[json.exception.type_error.305] cannot use operator[] with number"); } @@ -259,8 +259,8 @@ TEST_CASE("element access 1") { json j_nonarray(json::value_t::number_float); const json j_nonarray_const(j_nonarray); - CHECK_THROWS_AS(j_nonarray[0], json::type_error); - CHECK_THROWS_AS(j_nonarray_const[0], json::type_error); + CHECK_THROWS_AS(j_nonarray[0], json::type_error&); + CHECK_THROWS_AS(j_nonarray_const[0], json::type_error&); CHECK_THROWS_WITH(j_nonarray[0], "[json.exception.type_error.305] cannot use operator[] with number"); CHECK_THROWS_WITH(j_nonarray_const[0], "[json.exception.type_error.305] cannot use operator[] with number"); } @@ -313,7 +313,7 @@ TEST_CASE("element access 1") } { json jarray = {1, 1u, true, nullptr, "string", 42.23, json::object(), {1, 2, 3}}; - CHECK_THROWS_AS(jarray.erase(8), json::out_of_range); + CHECK_THROWS_AS(jarray.erase(8), json::out_of_range&); CHECK_THROWS_WITH(jarray.erase(8), "[json.exception.out_of_range.401] array index 8 is out of range"); } @@ -408,10 +408,10 @@ TEST_CASE("element access 1") { json jarray = {1, 1u, true, nullptr, "string", 42.23, json::object(), {1, 2, 3}}; json jarray2 = {"foo", "bar"}; - CHECK_THROWS_AS(jarray.erase(jarray2.begin()), json::invalid_iterator); - CHECK_THROWS_AS(jarray.erase(jarray.begin(), jarray2.end()), json::invalid_iterator); - CHECK_THROWS_AS(jarray.erase(jarray2.begin(), jarray.end()), json::invalid_iterator); - CHECK_THROWS_AS(jarray.erase(jarray2.begin(), jarray2.end()), json::invalid_iterator); + CHECK_THROWS_AS(jarray.erase(jarray2.begin()), json::invalid_iterator&); + CHECK_THROWS_AS(jarray.erase(jarray.begin(), jarray2.end()), json::invalid_iterator&); + CHECK_THROWS_AS(jarray.erase(jarray2.begin(), jarray.end()), json::invalid_iterator&); + CHECK_THROWS_AS(jarray.erase(jarray2.begin(), jarray2.end()), json::invalid_iterator&); CHECK_THROWS_WITH(jarray.erase(jarray2.begin()), "[json.exception.invalid_iterator.202] iterator does not fit current value"); @@ -425,10 +425,10 @@ TEST_CASE("element access 1") { json jarray = {1, 1u, true, nullptr, "string", 42.23, json::object(), {1, 2, 3}}; json jarray2 = {"foo", "bar"}; - CHECK_THROWS_AS(jarray.erase(jarray2.cbegin()), json::invalid_iterator); - CHECK_THROWS_AS(jarray.erase(jarray.cbegin(), jarray2.cend()), json::invalid_iterator); - CHECK_THROWS_AS(jarray.erase(jarray2.cbegin(), jarray.cend()), json::invalid_iterator); - CHECK_THROWS_AS(jarray.erase(jarray2.cbegin(), jarray2.cend()), json::invalid_iterator); + CHECK_THROWS_AS(jarray.erase(jarray2.cbegin()), json::invalid_iterator&); + CHECK_THROWS_AS(jarray.erase(jarray.cbegin(), jarray2.cend()), json::invalid_iterator&); + CHECK_THROWS_AS(jarray.erase(jarray2.cbegin(), jarray.cend()), json::invalid_iterator&); + CHECK_THROWS_AS(jarray.erase(jarray2.cbegin(), jarray2.cend()), json::invalid_iterator&); CHECK_THROWS_WITH(jarray.erase(jarray2.cbegin()), "[json.exception.invalid_iterator.202] iterator does not fit current value"); @@ -447,7 +447,7 @@ TEST_CASE("element access 1") SECTION("null") { json j_nonobject(json::value_t::null); - CHECK_THROWS_AS(j_nonobject.erase(0), json::type_error); + CHECK_THROWS_AS(j_nonobject.erase(0), json::type_error&); CHECK_THROWS_WITH(j_nonobject.erase(0), "[json.exception.type_error.307] cannot use erase() with null"); } @@ -455,7 +455,7 @@ TEST_CASE("element access 1") SECTION("boolean") { json j_nonobject(json::value_t::boolean); - CHECK_THROWS_AS(j_nonobject.erase(0), json::type_error); + CHECK_THROWS_AS(j_nonobject.erase(0), json::type_error&); CHECK_THROWS_WITH(j_nonobject.erase(0), "[json.exception.type_error.307] cannot use erase() with boolean"); } @@ -463,7 +463,7 @@ TEST_CASE("element access 1") SECTION("string") { json j_nonobject(json::value_t::string); - CHECK_THROWS_AS(j_nonobject.erase(0), json::type_error); + CHECK_THROWS_AS(j_nonobject.erase(0), json::type_error&); CHECK_THROWS_WITH(j_nonobject.erase(0), "[json.exception.type_error.307] cannot use erase() with string"); } @@ -471,7 +471,7 @@ TEST_CASE("element access 1") SECTION("object") { json j_nonobject(json::value_t::object); - CHECK_THROWS_AS(j_nonobject.erase(0), json::type_error); + CHECK_THROWS_AS(j_nonobject.erase(0), json::type_error&); CHECK_THROWS_WITH(j_nonobject.erase(0), "[json.exception.type_error.307] cannot use erase() with object"); } @@ -479,7 +479,7 @@ TEST_CASE("element access 1") SECTION("number (integer)") { json j_nonobject(json::value_t::number_integer); - CHECK_THROWS_AS(j_nonobject.erase(0), json::type_error); + CHECK_THROWS_AS(j_nonobject.erase(0), json::type_error&); CHECK_THROWS_WITH(j_nonobject.erase(0), "[json.exception.type_error.307] cannot use erase() with number"); } @@ -487,7 +487,7 @@ TEST_CASE("element access 1") SECTION("number (unsigned)") { json j_nonobject(json::value_t::number_unsigned); - CHECK_THROWS_AS(j_nonobject.erase(0), json::type_error); + CHECK_THROWS_AS(j_nonobject.erase(0), json::type_error&); CHECK_THROWS_WITH(j_nonobject.erase(0), "[json.exception.type_error.307] cannot use erase() with number"); } @@ -495,7 +495,7 @@ TEST_CASE("element access 1") SECTION("number (floating-point)") { json j_nonobject(json::value_t::number_float); - CHECK_THROWS_AS(j_nonobject.erase(0), json::type_error); + CHECK_THROWS_AS(j_nonobject.erase(0), json::type_error&); CHECK_THROWS_WITH(j_nonobject.erase(0), "[json.exception.type_error.307] cannot use erase() with number"); } @@ -511,15 +511,15 @@ TEST_CASE("element access 1") { { json j; - CHECK_THROWS_AS(j.front(), json::invalid_iterator); - CHECK_THROWS_AS(j.back(), json::invalid_iterator); + CHECK_THROWS_AS(j.front(), json::invalid_iterator&); + CHECK_THROWS_AS(j.back(), json::invalid_iterator&); CHECK_THROWS_WITH(j.front(), "[json.exception.invalid_iterator.214] cannot get value"); CHECK_THROWS_WITH(j.back(), "[json.exception.invalid_iterator.214] cannot get value"); } { const json j{}; - CHECK_THROWS_AS(j.front(), json::invalid_iterator); - CHECK_THROWS_AS(j.back(), json::invalid_iterator); + CHECK_THROWS_AS(j.front(), json::invalid_iterator&); + CHECK_THROWS_AS(j.back(), json::invalid_iterator&); CHECK_THROWS_WITH(j.front(), "[json.exception.invalid_iterator.214] cannot get value"); CHECK_THROWS_WITH(j.back(), "[json.exception.invalid_iterator.214] cannot get value"); } @@ -602,13 +602,13 @@ TEST_CASE("element access 1") { { json j; - CHECK_THROWS_AS(j.erase(j.begin()), json::type_error); + CHECK_THROWS_AS(j.erase(j.begin()), json::type_error&); CHECK_THROWS_WITH(j.erase(j.begin()), "[json.exception.type_error.307] cannot use erase() with null"); } { json j; - CHECK_THROWS_AS(j.erase(j.cbegin()), json::type_error); + CHECK_THROWS_AS(j.erase(j.cbegin()), json::type_error&); CHECK_THROWS_WITH(j.erase(j.begin()), "[json.exception.type_error.307] cannot use erase() with null"); } @@ -701,13 +701,13 @@ TEST_CASE("element access 1") { { json j = "foo"; - CHECK_THROWS_AS(j.erase(j.end()), json::invalid_iterator); + CHECK_THROWS_AS(j.erase(j.end()), json::invalid_iterator&); CHECK_THROWS_WITH(j.erase(j.end()), "[json.exception.invalid_iterator.205] iterator out of range"); } { json j = "bar"; - CHECK_THROWS_AS(j.erase(j.cend()), json::invalid_iterator); + CHECK_THROWS_AS(j.erase(j.cend()), json::invalid_iterator&); CHECK_THROWS_WITH(j.erase(j.cend()), "[json.exception.invalid_iterator.205] iterator out of range"); } @@ -717,13 +717,13 @@ TEST_CASE("element access 1") { { json j = false; - CHECK_THROWS_AS(j.erase(j.end()), json::invalid_iterator); + CHECK_THROWS_AS(j.erase(j.end()), json::invalid_iterator&); CHECK_THROWS_WITH(j.erase(j.end()), "[json.exception.invalid_iterator.205] iterator out of range"); } { json j = true; - CHECK_THROWS_AS(j.erase(j.cend()), json::invalid_iterator); + CHECK_THROWS_AS(j.erase(j.cend()), json::invalid_iterator&); CHECK_THROWS_WITH(j.erase(j.cend()), "[json.exception.invalid_iterator.205] iterator out of range"); } @@ -733,13 +733,13 @@ TEST_CASE("element access 1") { { json j = 17; - CHECK_THROWS_AS(j.erase(j.end()), json::invalid_iterator); + CHECK_THROWS_AS(j.erase(j.end()), json::invalid_iterator&); CHECK_THROWS_WITH(j.erase(j.end()), "[json.exception.invalid_iterator.205] iterator out of range"); } { json j = 17; - CHECK_THROWS_AS(j.erase(j.cend()), json::invalid_iterator); + CHECK_THROWS_AS(j.erase(j.cend()), json::invalid_iterator&); CHECK_THROWS_WITH(j.erase(j.cend()), "[json.exception.invalid_iterator.205] iterator out of range"); } @@ -749,13 +749,13 @@ TEST_CASE("element access 1") { { json j = 17u; - CHECK_THROWS_AS(j.erase(j.end()), json::invalid_iterator); + CHECK_THROWS_AS(j.erase(j.end()), json::invalid_iterator&); CHECK_THROWS_WITH(j.erase(j.end()), "[json.exception.invalid_iterator.205] iterator out of range"); } { json j = 17u; - CHECK_THROWS_AS(j.erase(j.cend()), json::invalid_iterator); + CHECK_THROWS_AS(j.erase(j.cend()), json::invalid_iterator&); CHECK_THROWS_WITH(j.erase(j.cend()), "[json.exception.invalid_iterator.205] iterator out of range"); } @@ -765,13 +765,13 @@ TEST_CASE("element access 1") { { json j = 23.42; - CHECK_THROWS_AS(j.erase(j.end()), json::invalid_iterator); + CHECK_THROWS_AS(j.erase(j.end()), json::invalid_iterator&); CHECK_THROWS_WITH(j.erase(j.end()), "[json.exception.invalid_iterator.205] iterator out of range"); } { json j = 23.42; - CHECK_THROWS_AS(j.erase(j.cend()), json::invalid_iterator); + CHECK_THROWS_AS(j.erase(j.cend()), json::invalid_iterator&); CHECK_THROWS_WITH(j.erase(j.cend()), "[json.exception.invalid_iterator.205] iterator out of range"); } @@ -784,13 +784,13 @@ TEST_CASE("element access 1") { { json j; - CHECK_THROWS_AS(j.erase(j.begin(), j.end()), json::type_error); + CHECK_THROWS_AS(j.erase(j.begin(), j.end()), json::type_error&); CHECK_THROWS_WITH(j.erase(j.begin(), j.end()), "[json.exception.type_error.307] cannot use erase() with null"); } { json j; - CHECK_THROWS_AS(j.erase(j.cbegin(), j.cend()), json::type_error); + CHECK_THROWS_AS(j.erase(j.cbegin(), j.cend()), json::type_error&); CHECK_THROWS_WITH(j.erase(j.cbegin(), j.cend()), "[json.exception.type_error.307] cannot use erase() with null"); } @@ -883,15 +883,15 @@ TEST_CASE("element access 1") { { json j = "foo"; - CHECK_THROWS_AS(j.erase(j.end(), j.end()), json::invalid_iterator); - CHECK_THROWS_AS(j.erase(j.begin(), j.begin()), json::invalid_iterator); + CHECK_THROWS_AS(j.erase(j.end(), j.end()), json::invalid_iterator&); + CHECK_THROWS_AS(j.erase(j.begin(), j.begin()), json::invalid_iterator&); CHECK_THROWS_WITH(j.erase(j.end(), j.end()), "[json.exception.invalid_iterator.204] iterators out of range"); CHECK_THROWS_WITH(j.erase(j.begin(), j.begin()), "[json.exception.invalid_iterator.204] iterators out of range"); } { json j = "bar"; - CHECK_THROWS_AS(j.erase(j.cend(), j.cend()), json::invalid_iterator); - CHECK_THROWS_AS(j.erase(j.cbegin(), j.cbegin()), json::invalid_iterator); + CHECK_THROWS_AS(j.erase(j.cend(), j.cend()), json::invalid_iterator&); + CHECK_THROWS_AS(j.erase(j.cbegin(), j.cbegin()), json::invalid_iterator&); CHECK_THROWS_WITH(j.erase(j.cend(), j.cend()), "[json.exception.invalid_iterator.204] iterators out of range"); CHECK_THROWS_WITH(j.erase(j.cbegin(), j.cbegin()), "[json.exception.invalid_iterator.204] iterators out of range"); } @@ -901,15 +901,15 @@ TEST_CASE("element access 1") { { json j = false; - CHECK_THROWS_AS(j.erase(j.end(), j.end()), json::invalid_iterator); - CHECK_THROWS_AS(j.erase(j.begin(), j.begin()), json::invalid_iterator); + CHECK_THROWS_AS(j.erase(j.end(), j.end()), json::invalid_iterator&); + CHECK_THROWS_AS(j.erase(j.begin(), j.begin()), json::invalid_iterator&); CHECK_THROWS_WITH(j.erase(j.end(), j.end()), "[json.exception.invalid_iterator.204] iterators out of range"); CHECK_THROWS_WITH(j.erase(j.begin(), j.begin()), "[json.exception.invalid_iterator.204] iterators out of range"); } { json j = true; - CHECK_THROWS_AS(j.erase(j.cend(), j.cend()), json::invalid_iterator); - CHECK_THROWS_AS(j.erase(j.cbegin(), j.cbegin()), json::invalid_iterator); + CHECK_THROWS_AS(j.erase(j.cend(), j.cend()), json::invalid_iterator&); + CHECK_THROWS_AS(j.erase(j.cbegin(), j.cbegin()), json::invalid_iterator&); CHECK_THROWS_WITH(j.erase(j.cend(), j.cend()), "[json.exception.invalid_iterator.204] iterators out of range"); CHECK_THROWS_WITH(j.erase(j.cbegin(), j.cbegin()), "[json.exception.invalid_iterator.204] iterators out of range"); } @@ -919,15 +919,15 @@ TEST_CASE("element access 1") { { json j = 17; - CHECK_THROWS_AS(j.erase(j.end(), j.end()), json::invalid_iterator); - CHECK_THROWS_AS(j.erase(j.begin(), j.begin()), json::invalid_iterator); + CHECK_THROWS_AS(j.erase(j.end(), j.end()), json::invalid_iterator&); + CHECK_THROWS_AS(j.erase(j.begin(), j.begin()), json::invalid_iterator&); CHECK_THROWS_WITH(j.erase(j.end(), j.end()), "[json.exception.invalid_iterator.204] iterators out of range"); CHECK_THROWS_WITH(j.erase(j.begin(), j.begin()), "[json.exception.invalid_iterator.204] iterators out of range"); } { json j = 17; - CHECK_THROWS_AS(j.erase(j.cend(), j.cend()), json::invalid_iterator); - CHECK_THROWS_AS(j.erase(j.cbegin(), j.cbegin()), json::invalid_iterator); + CHECK_THROWS_AS(j.erase(j.cend(), j.cend()), json::invalid_iterator&); + CHECK_THROWS_AS(j.erase(j.cbegin(), j.cbegin()), json::invalid_iterator&); CHECK_THROWS_WITH(j.erase(j.cend(), j.cend()), "[json.exception.invalid_iterator.204] iterators out of range"); CHECK_THROWS_WITH(j.erase(j.cbegin(), j.cbegin()), "[json.exception.invalid_iterator.204] iterators out of range"); } @@ -937,15 +937,15 @@ TEST_CASE("element access 1") { { json j = 17u; - CHECK_THROWS_AS(j.erase(j.end(), j.end()), json::invalid_iterator); - CHECK_THROWS_AS(j.erase(j.begin(), j.begin()), json::invalid_iterator); + CHECK_THROWS_AS(j.erase(j.end(), j.end()), json::invalid_iterator&); + CHECK_THROWS_AS(j.erase(j.begin(), j.begin()), json::invalid_iterator&); CHECK_THROWS_WITH(j.erase(j.end(), j.end()), "[json.exception.invalid_iterator.204] iterators out of range"); CHECK_THROWS_WITH(j.erase(j.begin(), j.begin()), "[json.exception.invalid_iterator.204] iterators out of range"); } { json j = 17u; - CHECK_THROWS_AS(j.erase(j.cend(), j.cend()), json::invalid_iterator); - CHECK_THROWS_AS(j.erase(j.cbegin(), j.cbegin()), json::invalid_iterator); + CHECK_THROWS_AS(j.erase(j.cend(), j.cend()), json::invalid_iterator&); + CHECK_THROWS_AS(j.erase(j.cbegin(), j.cbegin()), json::invalid_iterator&); CHECK_THROWS_WITH(j.erase(j.cend(), j.cend()), "[json.exception.invalid_iterator.204] iterators out of range"); CHECK_THROWS_WITH(j.erase(j.cbegin(), j.cbegin()), "[json.exception.invalid_iterator.204] iterators out of range"); } @@ -955,15 +955,15 @@ TEST_CASE("element access 1") { { json j = 23.42; - CHECK_THROWS_AS(j.erase(j.end(), j.end()), json::invalid_iterator); - CHECK_THROWS_AS(j.erase(j.begin(), j.begin()), json::invalid_iterator); + CHECK_THROWS_AS(j.erase(j.end(), j.end()), json::invalid_iterator&); + CHECK_THROWS_AS(j.erase(j.begin(), j.begin()), json::invalid_iterator&); CHECK_THROWS_WITH(j.erase(j.end(), j.end()), "[json.exception.invalid_iterator.204] iterators out of range"); CHECK_THROWS_WITH(j.erase(j.begin(), j.begin()), "[json.exception.invalid_iterator.204] iterators out of range"); } { json j = 23.42; - CHECK_THROWS_AS(j.erase(j.cend(), j.cend()), json::invalid_iterator); - CHECK_THROWS_AS(j.erase(j.cbegin(), j.cbegin()), json::invalid_iterator); + CHECK_THROWS_AS(j.erase(j.cend(), j.cend()), json::invalid_iterator&); + CHECK_THROWS_AS(j.erase(j.cbegin(), j.cbegin()), json::invalid_iterator&); CHECK_THROWS_WITH(j.erase(j.cend(), j.cend()), "[json.exception.invalid_iterator.204] iterators out of range"); CHECK_THROWS_WITH(j.erase(j.cbegin(), j.cbegin()), "[json.exception.invalid_iterator.204] iterators out of range"); } diff --git a/test/src/unit-element_access2.cpp b/test/src/unit-element_access2.cpp index ec03507e..5950349b 100644 --- a/test/src/unit-element_access2.cpp +++ b/test/src/unit-element_access2.cpp @@ -63,8 +63,8 @@ TEST_CASE("element access 2") SECTION("access outside bounds") { - CHECK_THROWS_AS(j.at("foo"), json::out_of_range); - CHECK_THROWS_AS(j_const.at("foo"), json::out_of_range); + CHECK_THROWS_AS(j.at("foo"), json::out_of_range&); + CHECK_THROWS_AS(j_const.at("foo"), json::out_of_range&); CHECK_THROWS_WITH(j.at("foo"), "[json.exception.out_of_range.403] key 'foo' not found"); CHECK_THROWS_WITH(j_const.at("foo"), @@ -77,8 +77,8 @@ TEST_CASE("element access 2") { json j_nonobject(json::value_t::null); const json j_nonobject_const(j_nonobject); - CHECK_THROWS_AS(j_nonobject.at("foo"), json::type_error); - CHECK_THROWS_AS(j_nonobject_const.at("foo"), json::type_error); + CHECK_THROWS_AS(j_nonobject.at("foo"), json::type_error&); + CHECK_THROWS_AS(j_nonobject_const.at("foo"), json::type_error&); CHECK_THROWS_WITH(j_nonobject.at("foo"), "[json.exception.type_error.304] cannot use at() with null"); CHECK_THROWS_WITH(j_nonobject_const.at("foo"), "[json.exception.type_error.304] cannot use at() with null"); } @@ -87,8 +87,8 @@ TEST_CASE("element access 2") { json j_nonobject(json::value_t::boolean); const json j_nonobject_const(j_nonobject); - CHECK_THROWS_AS(j_nonobject.at("foo"), json::type_error); - CHECK_THROWS_AS(j_nonobject_const.at("foo"), json::type_error); + CHECK_THROWS_AS(j_nonobject.at("foo"), json::type_error&); + CHECK_THROWS_AS(j_nonobject_const.at("foo"), json::type_error&); CHECK_THROWS_WITH(j_nonobject.at("foo"), "[json.exception.type_error.304] cannot use at() with boolean"); CHECK_THROWS_WITH(j_nonobject_const.at("foo"), "[json.exception.type_error.304] cannot use at() with boolean"); } @@ -97,8 +97,8 @@ TEST_CASE("element access 2") { json j_nonobject(json::value_t::string); const json j_nonobject_const(j_nonobject); - CHECK_THROWS_AS(j_nonobject.at("foo"), json::type_error); - CHECK_THROWS_AS(j_nonobject_const.at("foo"), json::type_error); + CHECK_THROWS_AS(j_nonobject.at("foo"), json::type_error&); + CHECK_THROWS_AS(j_nonobject_const.at("foo"), json::type_error&); CHECK_THROWS_WITH(j_nonobject.at("foo"), "[json.exception.type_error.304] cannot use at() with string"); CHECK_THROWS_WITH(j_nonobject_const.at("foo"), "[json.exception.type_error.304] cannot use at() with string"); } @@ -107,8 +107,8 @@ TEST_CASE("element access 2") { json j_nonobject(json::value_t::array); const json j_nonobject_const(j_nonobject); - CHECK_THROWS_AS(j_nonobject.at("foo"), json::type_error); - CHECK_THROWS_AS(j_nonobject_const.at("foo"), json::type_error); + CHECK_THROWS_AS(j_nonobject.at("foo"), json::type_error&); + CHECK_THROWS_AS(j_nonobject_const.at("foo"), json::type_error&); CHECK_THROWS_WITH(j_nonobject.at("foo"), "[json.exception.type_error.304] cannot use at() with array"); CHECK_THROWS_WITH(j_nonobject_const.at("foo"), "[json.exception.type_error.304] cannot use at() with array"); } @@ -117,8 +117,8 @@ TEST_CASE("element access 2") { json j_nonobject(json::value_t::number_integer); const json j_nonobject_const(j_nonobject); - CHECK_THROWS_AS(j_nonobject.at("foo"), json::type_error); - CHECK_THROWS_AS(j_nonobject_const.at("foo"), json::type_error); + CHECK_THROWS_AS(j_nonobject.at("foo"), json::type_error&); + CHECK_THROWS_AS(j_nonobject_const.at("foo"), json::type_error&); CHECK_THROWS_WITH(j_nonobject.at("foo"), "[json.exception.type_error.304] cannot use at() with number"); CHECK_THROWS_WITH(j_nonobject_const.at("foo"), "[json.exception.type_error.304] cannot use at() with number"); } @@ -127,8 +127,8 @@ TEST_CASE("element access 2") { json j_nonobject(json::value_t::number_unsigned); const json j_nonobject_const(j_nonobject); - CHECK_THROWS_AS(j_nonobject.at("foo"), json::type_error); - CHECK_THROWS_AS(j_nonobject_const.at("foo"), json::type_error); + CHECK_THROWS_AS(j_nonobject.at("foo"), json::type_error&); + CHECK_THROWS_AS(j_nonobject_const.at("foo"), json::type_error&); CHECK_THROWS_WITH(j_nonobject.at("foo"), "[json.exception.type_error.304] cannot use at() with number"); CHECK_THROWS_WITH(j_nonobject_const.at("foo"), "[json.exception.type_error.304] cannot use at() with number"); } @@ -137,8 +137,8 @@ TEST_CASE("element access 2") { json j_nonobject(json::value_t::number_float); const json j_nonobject_const(j_nonobject); - CHECK_THROWS_AS(j_nonobject.at("foo"), json::type_error); - CHECK_THROWS_AS(j_nonobject_const.at("foo"), json::type_error); + CHECK_THROWS_AS(j_nonobject.at("foo"), json::type_error&); + CHECK_THROWS_AS(j_nonobject_const.at("foo"), json::type_error&); CHECK_THROWS_WITH(j_nonobject.at("foo"), "[json.exception.type_error.304] cannot use at() with number"); CHECK_THROWS_WITH(j_nonobject_const.at("foo"), "[json.exception.type_error.304] cannot use at() with number"); } @@ -202,8 +202,8 @@ TEST_CASE("element access 2") { json j_nonobject(json::value_t::null); const json j_nonobject_const(j_nonobject); - CHECK_THROWS_AS(j_nonobject.value("foo", 1), json::type_error); - CHECK_THROWS_AS(j_nonobject_const.value("foo", 1), json::type_error); + CHECK_THROWS_AS(j_nonobject.value("foo", 1), json::type_error&); + CHECK_THROWS_AS(j_nonobject_const.value("foo", 1), json::type_error&); CHECK_THROWS_WITH(j_nonobject.value("foo", 1), "[json.exception.type_error.306] cannot use value() with null"); CHECK_THROWS_WITH(j_nonobject_const.value("foo", 1), @@ -214,8 +214,8 @@ TEST_CASE("element access 2") { json j_nonobject(json::value_t::boolean); const json j_nonobject_const(j_nonobject); - CHECK_THROWS_AS(j_nonobject.value("foo", 1), json::type_error); - CHECK_THROWS_AS(j_nonobject_const.value("foo", 1), json::type_error); + CHECK_THROWS_AS(j_nonobject.value("foo", 1), json::type_error&); + CHECK_THROWS_AS(j_nonobject_const.value("foo", 1), json::type_error&); CHECK_THROWS_WITH(j_nonobject.value("foo", 1), "[json.exception.type_error.306] cannot use value() with boolean"); CHECK_THROWS_WITH(j_nonobject_const.value("foo", 1), @@ -226,8 +226,8 @@ TEST_CASE("element access 2") { json j_nonobject(json::value_t::string); const json j_nonobject_const(j_nonobject); - CHECK_THROWS_AS(j_nonobject.value("foo", 1), json::type_error); - CHECK_THROWS_AS(j_nonobject_const.value("foo", 1), json::type_error); + CHECK_THROWS_AS(j_nonobject.value("foo", 1), json::type_error&); + CHECK_THROWS_AS(j_nonobject_const.value("foo", 1), json::type_error&); CHECK_THROWS_WITH(j_nonobject.value("foo", 1), "[json.exception.type_error.306] cannot use value() with string"); CHECK_THROWS_WITH(j_nonobject_const.value("foo", 1), @@ -238,8 +238,8 @@ TEST_CASE("element access 2") { json j_nonobject(json::value_t::array); const json j_nonobject_const(j_nonobject); - CHECK_THROWS_AS(j_nonobject.value("foo", 1), json::type_error); - CHECK_THROWS_AS(j_nonobject_const.value("foo", 1), json::type_error); + CHECK_THROWS_AS(j_nonobject.value("foo", 1), json::type_error&); + CHECK_THROWS_AS(j_nonobject_const.value("foo", 1), json::type_error&); CHECK_THROWS_WITH(j_nonobject.value("foo", 1), "[json.exception.type_error.306] cannot use value() with array"); CHECK_THROWS_WITH(j_nonobject_const.value("foo", 1), @@ -250,8 +250,8 @@ TEST_CASE("element access 2") { json j_nonobject(json::value_t::number_integer); const json j_nonobject_const(j_nonobject); - CHECK_THROWS_AS(j_nonobject.value("foo", 1), json::type_error); - CHECK_THROWS_AS(j_nonobject_const.value("foo", 1), json::type_error); + CHECK_THROWS_AS(j_nonobject.value("foo", 1), json::type_error&); + CHECK_THROWS_AS(j_nonobject_const.value("foo", 1), json::type_error&); CHECK_THROWS_WITH(j_nonobject.value("foo", 1), "[json.exception.type_error.306] cannot use value() with number"); CHECK_THROWS_WITH(j_nonobject_const.value("foo", 1), @@ -262,8 +262,8 @@ TEST_CASE("element access 2") { json j_nonobject(json::value_t::number_unsigned); const json j_nonobject_const(j_nonobject); - CHECK_THROWS_AS(j_nonobject.value("foo", 1), json::type_error); - CHECK_THROWS_AS(j_nonobject_const.value("foo", 1), json::type_error); + CHECK_THROWS_AS(j_nonobject.value("foo", 1), json::type_error&); + CHECK_THROWS_AS(j_nonobject_const.value("foo", 1), json::type_error&); CHECK_THROWS_WITH(j_nonobject.value("foo", 1), "[json.exception.type_error.306] cannot use value() with number"); CHECK_THROWS_WITH(j_nonobject_const.value("foo", 1), @@ -274,8 +274,8 @@ TEST_CASE("element access 2") { json j_nonobject(json::value_t::number_float); const json j_nonobject_const(j_nonobject); - CHECK_THROWS_AS(j_nonobject.value("foo", 1), json::type_error); - CHECK_THROWS_AS(j_nonobject_const.value("foo", 1), json::type_error); + CHECK_THROWS_AS(j_nonobject.value("foo", 1), json::type_error&); + CHECK_THROWS_AS(j_nonobject_const.value("foo", 1), json::type_error&); CHECK_THROWS_WITH(j_nonobject.value("foo", 1), "[json.exception.type_error.306] cannot use value() with number"); CHECK_THROWS_WITH(j_nonobject_const.value("foo", 1), @@ -320,8 +320,8 @@ TEST_CASE("element access 2") { json j_nonobject(json::value_t::null); const json j_nonobject_const(j_nonobject); - CHECK_THROWS_AS(j_nonobject.value("/foo"_json_pointer, 1), json::type_error); - CHECK_THROWS_AS(j_nonobject_const.value("/foo"_json_pointer, 1), json::type_error); + CHECK_THROWS_AS(j_nonobject.value("/foo"_json_pointer, 1), json::type_error&); + CHECK_THROWS_AS(j_nonobject_const.value("/foo"_json_pointer, 1), json::type_error&); CHECK_THROWS_WITH(j_nonobject.value("/foo"_json_pointer, 1), "[json.exception.type_error.306] cannot use value() with null"); CHECK_THROWS_WITH(j_nonobject_const.value("/foo"_json_pointer, 1), @@ -332,8 +332,8 @@ TEST_CASE("element access 2") { json j_nonobject(json::value_t::boolean); const json j_nonobject_const(j_nonobject); - CHECK_THROWS_AS(j_nonobject.value("/foo"_json_pointer, 1), json::type_error); - CHECK_THROWS_AS(j_nonobject_const.value("/foo"_json_pointer, 1), json::type_error); + CHECK_THROWS_AS(j_nonobject.value("/foo"_json_pointer, 1), json::type_error&); + CHECK_THROWS_AS(j_nonobject_const.value("/foo"_json_pointer, 1), json::type_error&); CHECK_THROWS_WITH(j_nonobject.value("/foo"_json_pointer, 1), "[json.exception.type_error.306] cannot use value() with boolean"); CHECK_THROWS_WITH(j_nonobject_const.value("/foo"_json_pointer, 1), @@ -344,8 +344,8 @@ TEST_CASE("element access 2") { json j_nonobject(json::value_t::string); const json j_nonobject_const(j_nonobject); - CHECK_THROWS_AS(j_nonobject.value("/foo"_json_pointer, 1), json::type_error); - CHECK_THROWS_AS(j_nonobject_const.value("/foo"_json_pointer, 1), json::type_error); + CHECK_THROWS_AS(j_nonobject.value("/foo"_json_pointer, 1), json::type_error&); + CHECK_THROWS_AS(j_nonobject_const.value("/foo"_json_pointer, 1), json::type_error&); CHECK_THROWS_WITH(j_nonobject.value("/foo"_json_pointer, 1), "[json.exception.type_error.306] cannot use value() with string"); CHECK_THROWS_WITH(j_nonobject_const.value("/foo"_json_pointer, 1), @@ -356,8 +356,8 @@ TEST_CASE("element access 2") { json j_nonobject(json::value_t::array); const json j_nonobject_const(j_nonobject); - CHECK_THROWS_AS(j_nonobject.value("/foo"_json_pointer, 1), json::type_error); - CHECK_THROWS_AS(j_nonobject_const.value("/foo"_json_pointer, 1), json::type_error); + CHECK_THROWS_AS(j_nonobject.value("/foo"_json_pointer, 1), json::type_error&); + CHECK_THROWS_AS(j_nonobject_const.value("/foo"_json_pointer, 1), json::type_error&); CHECK_THROWS_WITH(j_nonobject.value("/foo"_json_pointer, 1), "[json.exception.type_error.306] cannot use value() with array"); CHECK_THROWS_WITH(j_nonobject_const.value("/foo"_json_pointer, 1), @@ -368,8 +368,8 @@ TEST_CASE("element access 2") { json j_nonobject(json::value_t::number_integer); const json j_nonobject_const(j_nonobject); - CHECK_THROWS_AS(j_nonobject.value("/foo"_json_pointer, 1), json::type_error); - CHECK_THROWS_AS(j_nonobject_const.value("/foo"_json_pointer, 1), json::type_error); + CHECK_THROWS_AS(j_nonobject.value("/foo"_json_pointer, 1), json::type_error&); + CHECK_THROWS_AS(j_nonobject_const.value("/foo"_json_pointer, 1), json::type_error&); CHECK_THROWS_WITH(j_nonobject.value("/foo"_json_pointer, 1), "[json.exception.type_error.306] cannot use value() with number"); CHECK_THROWS_WITH(j_nonobject_const.value("/foo"_json_pointer, 1), @@ -380,8 +380,8 @@ TEST_CASE("element access 2") { json j_nonobject(json::value_t::number_unsigned); const json j_nonobject_const(j_nonobject); - CHECK_THROWS_AS(j_nonobject.value("/foo"_json_pointer, 1), json::type_error); - CHECK_THROWS_AS(j_nonobject_const.value("/foo"_json_pointer, 1), json::type_error); + CHECK_THROWS_AS(j_nonobject.value("/foo"_json_pointer, 1), json::type_error&); + CHECK_THROWS_AS(j_nonobject_const.value("/foo"_json_pointer, 1), json::type_error&); CHECK_THROWS_WITH(j_nonobject.value("/foo"_json_pointer, 1), "[json.exception.type_error.306] cannot use value() with number"); CHECK_THROWS_WITH(j_nonobject_const.value("/foo"_json_pointer, 1), @@ -392,8 +392,8 @@ TEST_CASE("element access 2") { json j_nonobject(json::value_t::number_float); const json j_nonobject_const(j_nonobject); - CHECK_THROWS_AS(j_nonobject.value("/foo"_json_pointer, 1), json::type_error); - CHECK_THROWS_AS(j_nonobject_const.value("/foo"_json_pointer, 1), json::type_error); + CHECK_THROWS_AS(j_nonobject.value("/foo"_json_pointer, 1), json::type_error&); + CHECK_THROWS_AS(j_nonobject_const.value("/foo"_json_pointer, 1), json::type_error&); CHECK_THROWS_WITH(j_nonobject.value("/foo"_json_pointer, 1), "[json.exception.type_error.306] cannot use value() with number"); CHECK_THROWS_WITH(j_nonobject_const.value("/foo"_json_pointer, 1), @@ -472,8 +472,8 @@ TEST_CASE("element access 2") const json j_const_nonobject(j_nonobject); CHECK_NOTHROW(j_nonobject["foo"]); CHECK_NOTHROW(j_nonobject2[json::object_t::key_type("foo")]); - CHECK_THROWS_AS(j_const_nonobject["foo"], json::type_error); - CHECK_THROWS_AS(j_const_nonobject[json::object_t::key_type("foo")], json::type_error); + CHECK_THROWS_AS(j_const_nonobject["foo"], json::type_error&); + CHECK_THROWS_AS(j_const_nonobject[json::object_t::key_type("foo")], json::type_error&); CHECK_THROWS_WITH(j_const_nonobject["foo"], "[json.exception.type_error.305] cannot use operator[] with null"); CHECK_THROWS_WITH(j_const_nonobject[json::object_t::key_type("foo")], "[json.exception.type_error.305] cannot use operator[] with null"); @@ -483,10 +483,10 @@ TEST_CASE("element access 2") { json j_nonobject(json::value_t::boolean); const json j_const_nonobject(j_nonobject); - CHECK_THROWS_AS(j_nonobject["foo"], json::type_error); - CHECK_THROWS_AS(j_nonobject[json::object_t::key_type("foo")], json::type_error); - CHECK_THROWS_AS(j_const_nonobject["foo"], json::type_error); - CHECK_THROWS_AS(j_const_nonobject[json::object_t::key_type("foo")], json::type_error); + CHECK_THROWS_AS(j_nonobject["foo"], json::type_error&); + CHECK_THROWS_AS(j_nonobject[json::object_t::key_type("foo")], json::type_error&); + CHECK_THROWS_AS(j_const_nonobject["foo"], json::type_error&); + CHECK_THROWS_AS(j_const_nonobject[json::object_t::key_type("foo")], json::type_error&); CHECK_THROWS_WITH(j_nonobject["foo"], "[json.exception.type_error.305] cannot use operator[] with boolean"); CHECK_THROWS_WITH(j_nonobject[json::object_t::key_type("foo")], @@ -501,10 +501,10 @@ TEST_CASE("element access 2") { json j_nonobject(json::value_t::string); const json j_const_nonobject(j_nonobject); - CHECK_THROWS_AS(j_nonobject["foo"], json::type_error); - CHECK_THROWS_AS(j_nonobject[json::object_t::key_type("foo")], json::type_error); - CHECK_THROWS_AS(j_const_nonobject["foo"], json::type_error); - CHECK_THROWS_AS(j_const_nonobject[json::object_t::key_type("foo")], json::type_error); + CHECK_THROWS_AS(j_nonobject["foo"], json::type_error&); + CHECK_THROWS_AS(j_nonobject[json::object_t::key_type("foo")], json::type_error&); + CHECK_THROWS_AS(j_const_nonobject["foo"], json::type_error&); + CHECK_THROWS_AS(j_const_nonobject[json::object_t::key_type("foo")], json::type_error&); CHECK_THROWS_WITH(j_nonobject["foo"], "[json.exception.type_error.305] cannot use operator[] with string"); CHECK_THROWS_WITH(j_nonobject[json::object_t::key_type("foo")], @@ -519,10 +519,10 @@ TEST_CASE("element access 2") { json j_nonobject(json::value_t::array); const json j_const_nonobject(j_nonobject); - CHECK_THROWS_AS(j_nonobject["foo"], json::type_error); - CHECK_THROWS_AS(j_nonobject[json::object_t::key_type("foo")], json::type_error); - CHECK_THROWS_AS(j_const_nonobject["foo"], json::type_error); - CHECK_THROWS_AS(j_const_nonobject[json::object_t::key_type("foo")], json::type_error); + CHECK_THROWS_AS(j_nonobject["foo"], json::type_error&); + CHECK_THROWS_AS(j_nonobject[json::object_t::key_type("foo")], json::type_error&); + CHECK_THROWS_AS(j_const_nonobject["foo"], json::type_error&); + CHECK_THROWS_AS(j_const_nonobject[json::object_t::key_type("foo")], json::type_error&); CHECK_THROWS_WITH(j_nonobject["foo"], "[json.exception.type_error.305] cannot use operator[] with array"); CHECK_THROWS_WITH(j_nonobject[json::object_t::key_type("foo")], "[json.exception.type_error.305] cannot use operator[] with array"); @@ -536,10 +536,10 @@ TEST_CASE("element access 2") { json j_nonobject(json::value_t::number_integer); const json j_const_nonobject(j_nonobject); - CHECK_THROWS_AS(j_nonobject["foo"], json::type_error); - CHECK_THROWS_AS(j_nonobject[json::object_t::key_type("foo")], json::type_error); - CHECK_THROWS_AS(j_const_nonobject["foo"], json::type_error); - CHECK_THROWS_AS(j_const_nonobject[json::object_t::key_type("foo")], json::type_error); + CHECK_THROWS_AS(j_nonobject["foo"], json::type_error&); + CHECK_THROWS_AS(j_nonobject[json::object_t::key_type("foo")], json::type_error&); + CHECK_THROWS_AS(j_const_nonobject["foo"], json::type_error&); + CHECK_THROWS_AS(j_const_nonobject[json::object_t::key_type("foo")], json::type_error&); CHECK_THROWS_WITH(j_nonobject["foo"], "[json.exception.type_error.305] cannot use operator[] with number"); CHECK_THROWS_WITH(j_nonobject[json::object_t::key_type("foo")], @@ -554,10 +554,10 @@ TEST_CASE("element access 2") { json j_nonobject(json::value_t::number_unsigned); const json j_const_nonobject(j_nonobject); - CHECK_THROWS_AS(j_nonobject["foo"], json::type_error); - CHECK_THROWS_AS(j_nonobject[json::object_t::key_type("foo")], json::type_error); - CHECK_THROWS_AS(j_const_nonobject["foo"], json::type_error); - CHECK_THROWS_AS(j_const_nonobject[json::object_t::key_type("foo")], json::type_error); + CHECK_THROWS_AS(j_nonobject["foo"], json::type_error&); + CHECK_THROWS_AS(j_nonobject[json::object_t::key_type("foo")], json::type_error&); + CHECK_THROWS_AS(j_const_nonobject["foo"], json::type_error&); + CHECK_THROWS_AS(j_const_nonobject[json::object_t::key_type("foo")], json::type_error&); CHECK_THROWS_WITH(j_nonobject["foo"], "[json.exception.type_error.305] cannot use operator[] with number"); CHECK_THROWS_WITH(j_nonobject[json::object_t::key_type("foo")], @@ -572,10 +572,10 @@ TEST_CASE("element access 2") { json j_nonobject(json::value_t::number_float); const json j_const_nonobject(j_nonobject); - CHECK_THROWS_AS(j_nonobject["foo"], json::type_error); - CHECK_THROWS_AS(j_nonobject[json::object_t::key_type("foo")], json::type_error); - CHECK_THROWS_AS(j_const_nonobject["foo"], json::type_error); - CHECK_THROWS_AS(j_const_nonobject[json::object_t::key_type("foo")], json::type_error); + CHECK_THROWS_AS(j_nonobject["foo"], json::type_error&); + CHECK_THROWS_AS(j_nonobject[json::object_t::key_type("foo")], json::type_error&); + CHECK_THROWS_AS(j_const_nonobject["foo"], json::type_error&); + CHECK_THROWS_AS(j_const_nonobject[json::object_t::key_type("foo")], json::type_error&); CHECK_THROWS_WITH(j_nonobject["foo"], "[json.exception.type_error.305] cannot use operator[] with number"); CHECK_THROWS_WITH(j_nonobject[json::object_t::key_type("foo")], @@ -722,10 +722,10 @@ TEST_CASE("element access 2") { json jobject = {{"a", "a"}, {"b", 1}, {"c", 17u}, {"d", false}, {"e", true}}; json jobject2 = {{"a", "a"}, {"b", 1}, {"c", 17u}}; - CHECK_THROWS_AS(jobject.erase(jobject2.begin()), json::invalid_iterator); - CHECK_THROWS_AS(jobject.erase(jobject.begin(), jobject2.end()), json::invalid_iterator); - CHECK_THROWS_AS(jobject.erase(jobject2.begin(), jobject.end()), json::invalid_iterator); - CHECK_THROWS_AS(jobject.erase(jobject2.begin(), jobject2.end()), json::invalid_iterator); + CHECK_THROWS_AS(jobject.erase(jobject2.begin()), json::invalid_iterator&); + CHECK_THROWS_AS(jobject.erase(jobject.begin(), jobject2.end()), json::invalid_iterator&); + CHECK_THROWS_AS(jobject.erase(jobject2.begin(), jobject.end()), json::invalid_iterator&); + CHECK_THROWS_AS(jobject.erase(jobject2.begin(), jobject2.end()), json::invalid_iterator&); CHECK_THROWS_WITH(jobject.erase(jobject2.begin()), "[json.exception.invalid_iterator.202] iterator does not fit current value"); CHECK_THROWS_WITH(jobject.erase(jobject.begin(), jobject2.end()), @@ -738,10 +738,10 @@ TEST_CASE("element access 2") { json jobject = {{"a", "a"}, {"b", 1}, {"c", 17u}, {"d", false}, {"e", true}}; json jobject2 = {{"a", "a"}, {"b", 1}, {"c", 17u}}; - CHECK_THROWS_AS(jobject.erase(jobject2.cbegin()), json::invalid_iterator); - CHECK_THROWS_AS(jobject.erase(jobject.cbegin(), jobject2.cend()), json::invalid_iterator); - CHECK_THROWS_AS(jobject.erase(jobject2.cbegin(), jobject.cend()), json::invalid_iterator); - CHECK_THROWS_AS(jobject.erase(jobject2.cbegin(), jobject2.cend()), json::invalid_iterator); + CHECK_THROWS_AS(jobject.erase(jobject2.cbegin()), json::invalid_iterator&); + CHECK_THROWS_AS(jobject.erase(jobject.cbegin(), jobject2.cend()), json::invalid_iterator&); + CHECK_THROWS_AS(jobject.erase(jobject2.cbegin(), jobject.cend()), json::invalid_iterator&); + CHECK_THROWS_AS(jobject.erase(jobject2.cbegin(), jobject2.cend()), json::invalid_iterator&); CHECK_THROWS_WITH(jobject.erase(jobject2.cbegin()), "[json.exception.invalid_iterator.202] iterator does not fit current value"); CHECK_THROWS_WITH(jobject.erase(jobject.cbegin(), jobject2.cend()), @@ -759,7 +759,7 @@ TEST_CASE("element access 2") SECTION("null") { json j_nonobject(json::value_t::null); - CHECK_THROWS_AS(j_nonobject.erase("foo"), json::type_error); + CHECK_THROWS_AS(j_nonobject.erase("foo"), json::type_error&); CHECK_THROWS_WITH(j_nonobject.erase("foo"), "[json.exception.type_error.307] cannot use erase() with null"); } @@ -767,7 +767,7 @@ TEST_CASE("element access 2") SECTION("boolean") { json j_nonobject(json::value_t::boolean); - CHECK_THROWS_AS(j_nonobject.erase("foo"), json::type_error); + CHECK_THROWS_AS(j_nonobject.erase("foo"), json::type_error&); CHECK_THROWS_WITH(j_nonobject.erase("foo"), "[json.exception.type_error.307] cannot use erase() with boolean"); } @@ -775,7 +775,7 @@ TEST_CASE("element access 2") SECTION("string") { json j_nonobject(json::value_t::string); - CHECK_THROWS_AS(j_nonobject.erase("foo"), json::type_error); + CHECK_THROWS_AS(j_nonobject.erase("foo"), json::type_error&); CHECK_THROWS_WITH(j_nonobject.erase("foo"), "[json.exception.type_error.307] cannot use erase() with string"); } @@ -783,7 +783,7 @@ TEST_CASE("element access 2") SECTION("array") { json j_nonobject(json::value_t::array); - CHECK_THROWS_AS(j_nonobject.erase("foo"), json::type_error); + CHECK_THROWS_AS(j_nonobject.erase("foo"), json::type_error&); CHECK_THROWS_WITH(j_nonobject.erase("foo"), "[json.exception.type_error.307] cannot use erase() with array"); } @@ -791,7 +791,7 @@ TEST_CASE("element access 2") SECTION("number (integer)") { json j_nonobject(json::value_t::number_integer); - CHECK_THROWS_AS(j_nonobject.erase("foo"), json::type_error); + CHECK_THROWS_AS(j_nonobject.erase("foo"), json::type_error&); CHECK_THROWS_WITH(j_nonobject.erase("foo"), "[json.exception.type_error.307] cannot use erase() with number"); } @@ -799,7 +799,7 @@ TEST_CASE("element access 2") SECTION("number (floating-point)") { json j_nonobject(json::value_t::number_float); - CHECK_THROWS_AS(j_nonobject.erase("foo"), json::type_error); + CHECK_THROWS_AS(j_nonobject.erase("foo"), json::type_error&); CHECK_THROWS_WITH(j_nonobject.erase("foo"), "[json.exception.type_error.307] cannot use erase() with number"); } diff --git a/test/src/unit-iterators1.cpp b/test/src/unit-iterators1.cpp index 783bf850..78607bc2 100644 --- a/test/src/unit-iterators1.cpp +++ b/test/src/unit-iterators1.cpp @@ -229,19 +229,19 @@ TEST_CASE("iterators 1") { auto it = j.begin(); auto cit = j_const.cbegin(); - CHECK_THROWS_AS(it.key(), json::invalid_iterator); + CHECK_THROWS_AS(it.key(), json::invalid_iterator&); CHECK_THROWS_WITH(it.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); CHECK(it.value() == json(true)); - CHECK_THROWS_AS(cit.key(), json::invalid_iterator); + CHECK_THROWS_AS(cit.key(), json::invalid_iterator&); CHECK_THROWS_WITH(cit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); CHECK(cit.value() == json(true)); auto rit = j.rend(); auto crit = j.crend(); - CHECK_THROWS_AS(rit.key(), json::invalid_iterator); - CHECK_THROWS_AS(rit.value(), json::invalid_iterator); - CHECK_THROWS_AS(crit.key(), json::invalid_iterator); - CHECK_THROWS_AS(crit.value(), json::invalid_iterator); + CHECK_THROWS_AS(rit.key(), json::invalid_iterator&); + CHECK_THROWS_AS(rit.value(), json::invalid_iterator&); + CHECK_THROWS_AS(crit.key(), json::invalid_iterator&); + CHECK_THROWS_AS(crit.value(), json::invalid_iterator&); CHECK_THROWS_WITH(rit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); CHECK_THROWS_WITH(rit.value(), "[json.exception.invalid_iterator.214] cannot get value"); CHECK_THROWS_WITH(crit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); @@ -433,19 +433,19 @@ TEST_CASE("iterators 1") { auto it = j.begin(); auto cit = j_const.cbegin(); - CHECK_THROWS_AS(it.key(), json::invalid_iterator); + CHECK_THROWS_AS(it.key(), json::invalid_iterator&); CHECK_THROWS_WITH(it.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); CHECK(it.value() == json("hello world")); - CHECK_THROWS_AS(cit.key(), json::invalid_iterator); + CHECK_THROWS_AS(cit.key(), json::invalid_iterator&); CHECK_THROWS_WITH(cit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); CHECK(cit.value() == json("hello world")); auto rit = j.rend(); auto crit = j.crend(); - CHECK_THROWS_AS(rit.key(), json::invalid_iterator); - CHECK_THROWS_AS(rit.value(), json::invalid_iterator); - CHECK_THROWS_AS(crit.key(), json::invalid_iterator); - CHECK_THROWS_AS(crit.value(), json::invalid_iterator); + CHECK_THROWS_AS(rit.key(), json::invalid_iterator&); + CHECK_THROWS_AS(rit.value(), json::invalid_iterator&); + CHECK_THROWS_AS(crit.key(), json::invalid_iterator&); + CHECK_THROWS_AS(crit.value(), json::invalid_iterator&); CHECK_THROWS_WITH(rit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); CHECK_THROWS_WITH(rit.value(), "[json.exception.invalid_iterator.214] cannot get value"); CHECK_THROWS_WITH(crit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); @@ -630,10 +630,10 @@ TEST_CASE("iterators 1") { auto it = j.begin(); auto cit = j_const.cbegin(); - CHECK_THROWS_AS(it.key(), json::invalid_iterator); + CHECK_THROWS_AS(it.key(), json::invalid_iterator&); CHECK_THROWS_WITH(it.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); CHECK(it.value() == json(1)); - CHECK_THROWS_AS(cit.key(), json::invalid_iterator); + CHECK_THROWS_AS(cit.key(), json::invalid_iterator&); CHECK_THROWS_WITH(cit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); CHECK(cit.value() == json(1)); } @@ -1007,19 +1007,19 @@ TEST_CASE("iterators 1") { auto it = j.begin(); auto cit = j_const.cbegin(); - CHECK_THROWS_AS(it.key(), json::invalid_iterator); + CHECK_THROWS_AS(it.key(), json::invalid_iterator&); CHECK_THROWS_WITH(it.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); CHECK(it.value() == json(23)); - CHECK_THROWS_AS(cit.key(), json::invalid_iterator); + CHECK_THROWS_AS(cit.key(), json::invalid_iterator&); CHECK_THROWS_WITH(cit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); CHECK(cit.value() == json(23)); auto rit = j.rend(); auto crit = j.crend(); - CHECK_THROWS_AS(rit.key(), json::invalid_iterator); - CHECK_THROWS_AS(rit.value(), json::invalid_iterator); - CHECK_THROWS_AS(crit.key(), json::invalid_iterator); - CHECK_THROWS_AS(crit.value(), json::invalid_iterator); + CHECK_THROWS_AS(rit.key(), json::invalid_iterator&); + CHECK_THROWS_AS(rit.value(), json::invalid_iterator&); + CHECK_THROWS_AS(crit.key(), json::invalid_iterator&); + CHECK_THROWS_AS(crit.value(), json::invalid_iterator&); CHECK_THROWS_WITH(rit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); CHECK_THROWS_WITH(rit.value(), "[json.exception.invalid_iterator.214] cannot get value"); CHECK_THROWS_WITH(crit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); @@ -1211,19 +1211,19 @@ TEST_CASE("iterators 1") { auto it = j.begin(); auto cit = j_const.cbegin(); - CHECK_THROWS_AS(it.key(), json::invalid_iterator); + CHECK_THROWS_AS(it.key(), json::invalid_iterator&); CHECK_THROWS_WITH(it.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); CHECK(it.value() == json(23)); - CHECK_THROWS_AS(cit.key(), json::invalid_iterator); + CHECK_THROWS_AS(cit.key(), json::invalid_iterator&); CHECK_THROWS_WITH(cit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); CHECK(cit.value() == json(23)); auto rit = j.rend(); auto crit = j.crend(); - CHECK_THROWS_AS(rit.key(), json::invalid_iterator); - CHECK_THROWS_AS(rit.value(), json::invalid_iterator); - CHECK_THROWS_AS(crit.key(), json::invalid_iterator); - CHECK_THROWS_AS(crit.value(), json::invalid_iterator); + CHECK_THROWS_AS(rit.key(), json::invalid_iterator&); + CHECK_THROWS_AS(rit.value(), json::invalid_iterator&); + CHECK_THROWS_AS(crit.key(), json::invalid_iterator&); + CHECK_THROWS_AS(crit.value(), json::invalid_iterator&); CHECK_THROWS_WITH(rit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); CHECK_THROWS_WITH(rit.value(), "[json.exception.invalid_iterator.214] cannot get value"); CHECK_THROWS_WITH(crit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); @@ -1415,19 +1415,19 @@ TEST_CASE("iterators 1") { auto it = j.begin(); auto cit = j_const.cbegin(); - CHECK_THROWS_AS(it.key(), json::invalid_iterator); + CHECK_THROWS_AS(it.key(), json::invalid_iterator&); CHECK_THROWS_WITH(it.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); CHECK(it.value() == json(23.42)); - CHECK_THROWS_AS(cit.key(), json::invalid_iterator); + CHECK_THROWS_AS(cit.key(), json::invalid_iterator&); CHECK_THROWS_WITH(cit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); CHECK(cit.value() == json(23.42)); auto rit = j.rend(); auto crit = j.crend(); - CHECK_THROWS_AS(rit.key(), json::invalid_iterator); - CHECK_THROWS_AS(rit.value(), json::invalid_iterator); - CHECK_THROWS_AS(crit.key(), json::invalid_iterator); - CHECK_THROWS_AS(crit.value(), json::invalid_iterator); + CHECK_THROWS_AS(rit.key(), json::invalid_iterator&); + CHECK_THROWS_AS(rit.value(), json::invalid_iterator&); + CHECK_THROWS_AS(crit.key(), json::invalid_iterator&); + CHECK_THROWS_AS(crit.value(), json::invalid_iterator&); CHECK_THROWS_WITH(rit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); CHECK_THROWS_WITH(rit.value(), "[json.exception.invalid_iterator.214] cannot get value"); CHECK_THROWS_WITH(crit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); @@ -1489,10 +1489,10 @@ TEST_CASE("iterators 1") { auto it = j.begin(); auto cit = j_const.cbegin(); - CHECK_THROWS_AS(it.key(), json::invalid_iterator); - CHECK_THROWS_AS(it.value(), json::invalid_iterator); - CHECK_THROWS_AS(cit.key(), json::invalid_iterator); - CHECK_THROWS_AS(cit.value(), json::invalid_iterator); + CHECK_THROWS_AS(it.key(), json::invalid_iterator&); + CHECK_THROWS_AS(it.value(), json::invalid_iterator&); + CHECK_THROWS_AS(cit.key(), json::invalid_iterator&); + CHECK_THROWS_AS(cit.value(), json::invalid_iterator&); CHECK_THROWS_WITH(it.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); CHECK_THROWS_WITH(it.value(), "[json.exception.invalid_iterator.214] cannot get value"); CHECK_THROWS_WITH(cit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); @@ -1500,10 +1500,10 @@ TEST_CASE("iterators 1") auto rit = j.rend(); auto crit = j.crend(); - CHECK_THROWS_AS(rit.key(), json::invalid_iterator); - CHECK_THROWS_AS(rit.value(), json::invalid_iterator); - CHECK_THROWS_AS(crit.key(), json::invalid_iterator); - CHECK_THROWS_AS(crit.value(), json::invalid_iterator); + CHECK_THROWS_AS(rit.key(), json::invalid_iterator&); + CHECK_THROWS_AS(rit.value(), json::invalid_iterator&); + CHECK_THROWS_AS(crit.key(), json::invalid_iterator&); + CHECK_THROWS_AS(crit.value(), json::invalid_iterator&); CHECK_THROWS_WITH(rit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); CHECK_THROWS_WITH(rit.value(), "[json.exception.invalid_iterator.214] cannot get value"); CHECK_THROWS_WITH(crit.key(), "[json.exception.invalid_iterator.207] cannot use key() for non-object iterators"); diff --git a/test/src/unit-iterators2.cpp b/test/src/unit-iterators2.cpp index 29d134da..be4e2770 100644 --- a/test/src/unit-iterators2.cpp +++ b/test/src/unit-iterators2.cpp @@ -81,14 +81,14 @@ TEST_CASE("iterators 2") { if (j.type() == json::value_t::object) { - CHECK_THROWS_AS(it1 < it1, json::invalid_iterator); - CHECK_THROWS_AS(it1 < it2, json::invalid_iterator); - CHECK_THROWS_AS(it2 < it3, json::invalid_iterator); - CHECK_THROWS_AS(it1 < it3, json::invalid_iterator); - CHECK_THROWS_AS(it1_c < it1_c, json::invalid_iterator); - CHECK_THROWS_AS(it1_c < it2_c, json::invalid_iterator); - CHECK_THROWS_AS(it2_c < it3_c, json::invalid_iterator); - CHECK_THROWS_AS(it1_c < it3_c, json::invalid_iterator); + CHECK_THROWS_AS(it1 < it1, json::invalid_iterator&); + CHECK_THROWS_AS(it1 < it2, json::invalid_iterator&); + CHECK_THROWS_AS(it2 < it3, json::invalid_iterator&); + CHECK_THROWS_AS(it1 < it3, json::invalid_iterator&); + CHECK_THROWS_AS(it1_c < it1_c, json::invalid_iterator&); + CHECK_THROWS_AS(it1_c < it2_c, json::invalid_iterator&); + CHECK_THROWS_AS(it2_c < it3_c, json::invalid_iterator&); + CHECK_THROWS_AS(it1_c < it3_c, json::invalid_iterator&); CHECK_THROWS_WITH(it1 < it1, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); CHECK_THROWS_WITH(it1 < it2, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); CHECK_THROWS_WITH(it2 < it3, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); @@ -115,14 +115,14 @@ TEST_CASE("iterators 2") { if (j.type() == json::value_t::object) { - CHECK_THROWS_AS(it1 <= it1, json::invalid_iterator); - CHECK_THROWS_AS(it1 <= it2, json::invalid_iterator); - CHECK_THROWS_AS(it2 <= it3, json::invalid_iterator); - CHECK_THROWS_AS(it1 <= it3, json::invalid_iterator); - CHECK_THROWS_AS(it1_c <= it1_c, json::invalid_iterator); - CHECK_THROWS_AS(it1_c <= it2_c, json::invalid_iterator); - CHECK_THROWS_AS(it2_c <= it3_c, json::invalid_iterator); - CHECK_THROWS_AS(it1_c <= it3_c, json::invalid_iterator); + CHECK_THROWS_AS(it1 <= it1, json::invalid_iterator&); + CHECK_THROWS_AS(it1 <= it2, json::invalid_iterator&); + CHECK_THROWS_AS(it2 <= it3, json::invalid_iterator&); + CHECK_THROWS_AS(it1 <= it3, json::invalid_iterator&); + CHECK_THROWS_AS(it1_c <= it1_c, json::invalid_iterator&); + CHECK_THROWS_AS(it1_c <= it2_c, json::invalid_iterator&); + CHECK_THROWS_AS(it2_c <= it3_c, json::invalid_iterator&); + CHECK_THROWS_AS(it1_c <= it3_c, json::invalid_iterator&); CHECK_THROWS_WITH(it1 <= it1, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); CHECK_THROWS_WITH(it1 <= it2, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); CHECK_THROWS_WITH(it2 <= it3, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); @@ -150,14 +150,14 @@ TEST_CASE("iterators 2") { if (j.type() == json::value_t::object) { - CHECK_THROWS_AS(it1 > it1, json::invalid_iterator); - CHECK_THROWS_AS(it1 > it2, json::invalid_iterator); - CHECK_THROWS_AS(it2 > it3, json::invalid_iterator); - CHECK_THROWS_AS(it1 > it3, json::invalid_iterator); - CHECK_THROWS_AS(it1_c > it1_c, json::invalid_iterator); - CHECK_THROWS_AS(it1_c > it2_c, json::invalid_iterator); - CHECK_THROWS_AS(it2_c > it3_c, json::invalid_iterator); - CHECK_THROWS_AS(it1_c > it3_c, json::invalid_iterator); + CHECK_THROWS_AS(it1 > it1, json::invalid_iterator&); + CHECK_THROWS_AS(it1 > it2, json::invalid_iterator&); + CHECK_THROWS_AS(it2 > it3, json::invalid_iterator&); + CHECK_THROWS_AS(it1 > it3, json::invalid_iterator&); + CHECK_THROWS_AS(it1_c > it1_c, json::invalid_iterator&); + CHECK_THROWS_AS(it1_c > it2_c, json::invalid_iterator&); + CHECK_THROWS_AS(it2_c > it3_c, json::invalid_iterator&); + CHECK_THROWS_AS(it1_c > it3_c, json::invalid_iterator&); CHECK_THROWS_WITH(it1 > it1, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); CHECK_THROWS_WITH(it1 > it2, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); CHECK_THROWS_WITH(it2 > it3, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); @@ -185,14 +185,14 @@ TEST_CASE("iterators 2") { if (j.type() == json::value_t::object) { - CHECK_THROWS_AS(it1 >= it1, json::invalid_iterator); - CHECK_THROWS_AS(it1 >= it2, json::invalid_iterator); - CHECK_THROWS_AS(it2 >= it3, json::invalid_iterator); - CHECK_THROWS_AS(it1 >= it3, json::invalid_iterator); - CHECK_THROWS_AS(it1_c >= it1_c, json::invalid_iterator); - CHECK_THROWS_AS(it1_c >= it2_c, json::invalid_iterator); - CHECK_THROWS_AS(it2_c >= it3_c, json::invalid_iterator); - CHECK_THROWS_AS(it1_c >= it3_c, json::invalid_iterator); + CHECK_THROWS_AS(it1 >= it1, json::invalid_iterator&); + CHECK_THROWS_AS(it1 >= it2, json::invalid_iterator&); + CHECK_THROWS_AS(it2 >= it3, json::invalid_iterator&); + CHECK_THROWS_AS(it1 >= it3, json::invalid_iterator&); + CHECK_THROWS_AS(it1_c >= it1_c, json::invalid_iterator&); + CHECK_THROWS_AS(it1_c >= it2_c, json::invalid_iterator&); + CHECK_THROWS_AS(it2_c >= it3_c, json::invalid_iterator&); + CHECK_THROWS_AS(it1_c >= it3_c, json::invalid_iterator&); CHECK_THROWS_WITH(it1 >= it1, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); CHECK_THROWS_WITH(it1 >= it2, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); CHECK_THROWS_WITH(it2 >= it3, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); @@ -224,13 +224,13 @@ TEST_CASE("iterators 2") { if (j != k) { - CHECK_THROWS_AS(j.begin() == k.begin(), json::invalid_iterator); - CHECK_THROWS_AS(j.cbegin() == k.cbegin(), json::invalid_iterator); + CHECK_THROWS_AS(j.begin() == k.begin(), json::invalid_iterator&); + CHECK_THROWS_AS(j.cbegin() == k.cbegin(), json::invalid_iterator&); CHECK_THROWS_WITH(j.begin() == k.begin(), "[json.exception.invalid_iterator.212] cannot compare iterators of different containers"); CHECK_THROWS_WITH(j.cbegin() == k.cbegin(), "[json.exception.invalid_iterator.212] cannot compare iterators of different containers"); - CHECK_THROWS_AS(j.begin() < k.begin(), json::invalid_iterator); - CHECK_THROWS_AS(j.cbegin() < k.cbegin(), json::invalid_iterator); + CHECK_THROWS_AS(j.begin() < k.begin(), json::invalid_iterator&); + CHECK_THROWS_AS(j.cbegin() < k.cbegin(), json::invalid_iterator&); CHECK_THROWS_WITH(j.begin() < k.begin(), "[json.exception.invalid_iterator.212] cannot compare iterators of different containers"); CHECK_THROWS_WITH(j.cbegin() < k.cbegin(), "[json.exception.invalid_iterator.212] cannot compare iterators of different containers"); } @@ -251,62 +251,62 @@ TEST_CASE("iterators 2") { { auto it = j_object.begin(); - CHECK_THROWS_AS(it += 1, json::invalid_iterator); + CHECK_THROWS_AS(it += 1, json::invalid_iterator&); CHECK_THROWS_WITH(it += 1, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } { auto it = j_object.cbegin(); - CHECK_THROWS_AS(it += 1, json::invalid_iterator); + CHECK_THROWS_AS(it += 1, json::invalid_iterator&); CHECK_THROWS_WITH(it += 1, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } { auto it = j_object.begin(); - CHECK_THROWS_AS(it + 1, json::invalid_iterator); + CHECK_THROWS_AS(it + 1, json::invalid_iterator&); CHECK_THROWS_WITH(it + 1, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } { auto it = j_object.cbegin(); - CHECK_THROWS_AS(it + 1, json::invalid_iterator); + CHECK_THROWS_AS(it + 1, json::invalid_iterator&); CHECK_THROWS_WITH(it + 1, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } { auto it = j_object.begin(); - CHECK_THROWS_AS(1 + it, json::invalid_iterator); + CHECK_THROWS_AS(1 + it, json::invalid_iterator&); CHECK_THROWS_WITH(1 + it, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } { auto it = j_object.cbegin(); - CHECK_THROWS_AS(1 + it, json::invalid_iterator); + CHECK_THROWS_AS(1 + it, json::invalid_iterator&); CHECK_THROWS_WITH(1 + it, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } { auto it = j_object.begin(); - CHECK_THROWS_AS(it -= 1, json::invalid_iterator); + CHECK_THROWS_AS(it -= 1, json::invalid_iterator&); CHECK_THROWS_WITH(it -= 1, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } { auto it = j_object.cbegin(); - CHECK_THROWS_AS(it -= 1, json::invalid_iterator); + CHECK_THROWS_AS(it -= 1, json::invalid_iterator&); CHECK_THROWS_WITH(it -= 1, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } { auto it = j_object.begin(); - CHECK_THROWS_AS(it - 1, json::invalid_iterator); + CHECK_THROWS_AS(it - 1, json::invalid_iterator&); CHECK_THROWS_WITH(it - 1, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } { auto it = j_object.cbegin(); - CHECK_THROWS_AS(it - 1, json::invalid_iterator); + CHECK_THROWS_AS(it - 1, json::invalid_iterator&); CHECK_THROWS_WITH(it - 1, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } { auto it = j_object.begin(); - CHECK_THROWS_AS(it - it, json::invalid_iterator); + CHECK_THROWS_AS(it - it, json::invalid_iterator&); CHECK_THROWS_WITH(it - it, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } { auto it = j_object.cbegin(); - CHECK_THROWS_AS(it - it, json::invalid_iterator); + CHECK_THROWS_AS(it - it, json::invalid_iterator&); CHECK_THROWS_WITH(it - it, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } } @@ -396,15 +396,15 @@ TEST_CASE("iterators 2") { { auto it = j_object.begin(); - CHECK_THROWS_AS(it[0], json::invalid_iterator); - CHECK_THROWS_AS(it[1], json::invalid_iterator); + CHECK_THROWS_AS(it[0], json::invalid_iterator&); + CHECK_THROWS_AS(it[1], json::invalid_iterator&); CHECK_THROWS_WITH(it[0], "[json.exception.invalid_iterator.208] cannot use operator[] for object iterators"); CHECK_THROWS_WITH(it[1], "[json.exception.invalid_iterator.208] cannot use operator[] for object iterators"); } { auto it = j_object.cbegin(); - CHECK_THROWS_AS(it[0], json::invalid_iterator); - CHECK_THROWS_AS(it[1], json::invalid_iterator); + CHECK_THROWS_AS(it[0], json::invalid_iterator&); + CHECK_THROWS_AS(it[1], json::invalid_iterator&); CHECK_THROWS_WITH(it[0], "[json.exception.invalid_iterator.208] cannot use operator[] for object iterators"); CHECK_THROWS_WITH(it[1], "[json.exception.invalid_iterator.208] cannot use operator[] for object iterators"); } @@ -436,15 +436,15 @@ TEST_CASE("iterators 2") { { auto it = j_null.begin(); - CHECK_THROWS_AS(it[0], json::invalid_iterator); - CHECK_THROWS_AS(it[1], json::invalid_iterator); + CHECK_THROWS_AS(it[0], json::invalid_iterator&); + CHECK_THROWS_AS(it[1], json::invalid_iterator&); CHECK_THROWS_WITH(it[0], "[json.exception.invalid_iterator.214] cannot get value"); CHECK_THROWS_WITH(it[1], "[json.exception.invalid_iterator.214] cannot get value"); } { auto it = j_null.cbegin(); - CHECK_THROWS_AS(it[0], json::invalid_iterator); - CHECK_THROWS_AS(it[1], json::invalid_iterator); + CHECK_THROWS_AS(it[0], json::invalid_iterator&); + CHECK_THROWS_AS(it[1], json::invalid_iterator&); CHECK_THROWS_WITH(it[0], "[json.exception.invalid_iterator.214] cannot get value"); CHECK_THROWS_WITH(it[1], "[json.exception.invalid_iterator.214] cannot get value"); } @@ -455,13 +455,13 @@ TEST_CASE("iterators 2") { auto it = j_value.begin(); CHECK(it[0] == json(42)); - CHECK_THROWS_AS(it[1], json::invalid_iterator); + CHECK_THROWS_AS(it[1], json::invalid_iterator&); CHECK_THROWS_WITH(it[1], "[json.exception.invalid_iterator.214] cannot get value"); } { auto it = j_value.cbegin(); CHECK(it[0] == json(42)); - CHECK_THROWS_AS(it[1], json::invalid_iterator); + CHECK_THROWS_AS(it[1], json::invalid_iterator&); CHECK_THROWS_WITH(it[1], "[json.exception.invalid_iterator.214] cannot get value"); } } @@ -516,14 +516,14 @@ TEST_CASE("iterators 2") { if (j.type() == json::value_t::object) { - CHECK_THROWS_AS(it1 < it1, json::invalid_iterator); - CHECK_THROWS_AS(it1 < it2, json::invalid_iterator); - CHECK_THROWS_AS(it2 < it3, json::invalid_iterator); - CHECK_THROWS_AS(it1 < it3, json::invalid_iterator); - CHECK_THROWS_AS(it1_c < it1_c, json::invalid_iterator); - CHECK_THROWS_AS(it1_c < it2_c, json::invalid_iterator); - CHECK_THROWS_AS(it2_c < it3_c, json::invalid_iterator); - CHECK_THROWS_AS(it1_c < it3_c, json::invalid_iterator); + CHECK_THROWS_AS(it1 < it1, json::invalid_iterator&); + CHECK_THROWS_AS(it1 < it2, json::invalid_iterator&); + CHECK_THROWS_AS(it2 < it3, json::invalid_iterator&); + CHECK_THROWS_AS(it1 < it3, json::invalid_iterator&); + CHECK_THROWS_AS(it1_c < it1_c, json::invalid_iterator&); + CHECK_THROWS_AS(it1_c < it2_c, json::invalid_iterator&); + CHECK_THROWS_AS(it2_c < it3_c, json::invalid_iterator&); + CHECK_THROWS_AS(it1_c < it3_c, json::invalid_iterator&); CHECK_THROWS_WITH(it1 < it1, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); CHECK_THROWS_WITH(it1 < it2, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); CHECK_THROWS_WITH(it2 < it3, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); @@ -550,14 +550,14 @@ TEST_CASE("iterators 2") { if (j.type() == json::value_t::object) { - CHECK_THROWS_AS(it1 <= it1, json::invalid_iterator); - CHECK_THROWS_AS(it1 <= it2, json::invalid_iterator); - CHECK_THROWS_AS(it2 <= it3, json::invalid_iterator); - CHECK_THROWS_AS(it1 <= it3, json::invalid_iterator); - CHECK_THROWS_AS(it1_c <= it1_c, json::invalid_iterator); - CHECK_THROWS_AS(it1_c <= it2_c, json::invalid_iterator); - CHECK_THROWS_AS(it2_c <= it3_c, json::invalid_iterator); - CHECK_THROWS_AS(it1_c <= it3_c, json::invalid_iterator); + CHECK_THROWS_AS(it1 <= it1, json::invalid_iterator&); + CHECK_THROWS_AS(it1 <= it2, json::invalid_iterator&); + CHECK_THROWS_AS(it2 <= it3, json::invalid_iterator&); + CHECK_THROWS_AS(it1 <= it3, json::invalid_iterator&); + CHECK_THROWS_AS(it1_c <= it1_c, json::invalid_iterator&); + CHECK_THROWS_AS(it1_c <= it2_c, json::invalid_iterator&); + CHECK_THROWS_AS(it2_c <= it3_c, json::invalid_iterator&); + CHECK_THROWS_AS(it1_c <= it3_c, json::invalid_iterator&); CHECK_THROWS_WITH(it1 <= it1, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); CHECK_THROWS_WITH(it1 <= it2, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); CHECK_THROWS_WITH(it2 <= it3, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); @@ -585,14 +585,14 @@ TEST_CASE("iterators 2") { if (j.type() == json::value_t::object) { - CHECK_THROWS_AS(it1 > it1, json::invalid_iterator); - CHECK_THROWS_AS(it1 > it2, json::invalid_iterator); - CHECK_THROWS_AS(it2 > it3, json::invalid_iterator); - CHECK_THROWS_AS(it1 > it3, json::invalid_iterator); - CHECK_THROWS_AS(it1_c > it1_c, json::invalid_iterator); - CHECK_THROWS_AS(it1_c > it2_c, json::invalid_iterator); - CHECK_THROWS_AS(it2_c > it3_c, json::invalid_iterator); - CHECK_THROWS_AS(it1_c > it3_c, json::invalid_iterator); + CHECK_THROWS_AS(it1 > it1, json::invalid_iterator&); + CHECK_THROWS_AS(it1 > it2, json::invalid_iterator&); + CHECK_THROWS_AS(it2 > it3, json::invalid_iterator&); + CHECK_THROWS_AS(it1 > it3, json::invalid_iterator&); + CHECK_THROWS_AS(it1_c > it1_c, json::invalid_iterator&); + CHECK_THROWS_AS(it1_c > it2_c, json::invalid_iterator&); + CHECK_THROWS_AS(it2_c > it3_c, json::invalid_iterator&); + CHECK_THROWS_AS(it1_c > it3_c, json::invalid_iterator&); CHECK_THROWS_WITH(it1 > it1, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); CHECK_THROWS_WITH(it1 > it2, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); CHECK_THROWS_WITH(it2 > it3, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); @@ -620,14 +620,14 @@ TEST_CASE("iterators 2") { if (j.type() == json::value_t::object) { - CHECK_THROWS_AS(it1 >= it1, json::invalid_iterator); - CHECK_THROWS_AS(it1 >= it2, json::invalid_iterator); - CHECK_THROWS_AS(it2 >= it3, json::invalid_iterator); - CHECK_THROWS_AS(it1 >= it3, json::invalid_iterator); - CHECK_THROWS_AS(it1_c >= it1_c, json::invalid_iterator); - CHECK_THROWS_AS(it1_c >= it2_c, json::invalid_iterator); - CHECK_THROWS_AS(it2_c >= it3_c, json::invalid_iterator); - CHECK_THROWS_AS(it1_c >= it3_c, json::invalid_iterator); + CHECK_THROWS_AS(it1 >= it1, json::invalid_iterator&); + CHECK_THROWS_AS(it1 >= it2, json::invalid_iterator&); + CHECK_THROWS_AS(it2 >= it3, json::invalid_iterator&); + CHECK_THROWS_AS(it1 >= it3, json::invalid_iterator&); + CHECK_THROWS_AS(it1_c >= it1_c, json::invalid_iterator&); + CHECK_THROWS_AS(it1_c >= it2_c, json::invalid_iterator&); + CHECK_THROWS_AS(it2_c >= it3_c, json::invalid_iterator&); + CHECK_THROWS_AS(it1_c >= it3_c, json::invalid_iterator&); CHECK_THROWS_WITH(it1 >= it1, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); CHECK_THROWS_WITH(it1 >= it2, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); CHECK_THROWS_WITH(it2 >= it3, "[json.exception.invalid_iterator.213] cannot compare order of object iterators"); @@ -659,13 +659,13 @@ TEST_CASE("iterators 2") { if (j != k) { - CHECK_THROWS_AS(j.rbegin() == k.rbegin(), json::invalid_iterator); - CHECK_THROWS_AS(j.crbegin() == k.crbegin(), json::invalid_iterator); + CHECK_THROWS_AS(j.rbegin() == k.rbegin(), json::invalid_iterator&); + CHECK_THROWS_AS(j.crbegin() == k.crbegin(), json::invalid_iterator&); CHECK_THROWS_WITH(j.rbegin() == k.rbegin(), "[json.exception.invalid_iterator.212] cannot compare iterators of different containers"); CHECK_THROWS_WITH(j.crbegin() == k.crbegin(), "[json.exception.invalid_iterator.212] cannot compare iterators of different containers"); - CHECK_THROWS_AS(j.rbegin() < k.rbegin(), json::invalid_iterator); - CHECK_THROWS_AS(j.crbegin() < k.crbegin(), json::invalid_iterator); + CHECK_THROWS_AS(j.rbegin() < k.rbegin(), json::invalid_iterator&); + CHECK_THROWS_AS(j.crbegin() < k.crbegin(), json::invalid_iterator&); CHECK_THROWS_WITH(j.rbegin() < k.rbegin(), "[json.exception.invalid_iterator.212] cannot compare iterators of different containers"); CHECK_THROWS_WITH(j.crbegin() < k.crbegin(), "[json.exception.invalid_iterator.212] cannot compare iterators of different containers"); } @@ -686,62 +686,62 @@ TEST_CASE("iterators 2") { { auto it = j_object.rbegin(); - CHECK_THROWS_AS(it += 1, json::invalid_iterator); + CHECK_THROWS_AS(it += 1, json::invalid_iterator&); CHECK_THROWS_WITH(it += 1, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } { auto it = j_object.crbegin(); - CHECK_THROWS_AS(it += 1, json::invalid_iterator); + CHECK_THROWS_AS(it += 1, json::invalid_iterator&); CHECK_THROWS_WITH(it += 1, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } { auto it = j_object.rbegin(); - CHECK_THROWS_AS(it + 1, json::invalid_iterator); + CHECK_THROWS_AS(it + 1, json::invalid_iterator&); CHECK_THROWS_WITH(it + 1, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } { auto it = j_object.crbegin(); - CHECK_THROWS_AS(it + 1, json::invalid_iterator); + CHECK_THROWS_AS(it + 1, json::invalid_iterator&); CHECK_THROWS_WITH(it + 1, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } { auto it = j_object.rbegin(); - CHECK_THROWS_AS(1 + it, json::invalid_iterator); + CHECK_THROWS_AS(1 + it, json::invalid_iterator&); CHECK_THROWS_WITH(1 + it, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } { auto it = j_object.crbegin(); - CHECK_THROWS_AS(1 + it, json::invalid_iterator); + CHECK_THROWS_AS(1 + it, json::invalid_iterator&); CHECK_THROWS_WITH(1 + it, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } { auto it = j_object.rbegin(); - CHECK_THROWS_AS(it -= 1, json::invalid_iterator); + CHECK_THROWS_AS(it -= 1, json::invalid_iterator&); CHECK_THROWS_WITH(it -= 1, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } { auto it = j_object.crbegin(); - CHECK_THROWS_AS(it -= 1, json::invalid_iterator); + CHECK_THROWS_AS(it -= 1, json::invalid_iterator&); CHECK_THROWS_WITH(it -= 1, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } { auto it = j_object.rbegin(); - CHECK_THROWS_AS(it - 1, json::invalid_iterator); + CHECK_THROWS_AS(it - 1, json::invalid_iterator&); CHECK_THROWS_WITH(it - 1, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } { auto it = j_object.crbegin(); - CHECK_THROWS_AS(it - 1, json::invalid_iterator); + CHECK_THROWS_AS(it - 1, json::invalid_iterator&); CHECK_THROWS_WITH(it - 1, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } { auto it = j_object.rbegin(); - CHECK_THROWS_AS(it - it, json::invalid_iterator); + CHECK_THROWS_AS(it - it, json::invalid_iterator&); CHECK_THROWS_WITH(it - it, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } { auto it = j_object.crbegin(); - CHECK_THROWS_AS(it - it, json::invalid_iterator); + CHECK_THROWS_AS(it - it, json::invalid_iterator&); CHECK_THROWS_WITH(it - it, "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } } @@ -831,15 +831,15 @@ TEST_CASE("iterators 2") { { auto it = j_object.rbegin(); - CHECK_THROWS_AS(it[0], json::invalid_iterator); - CHECK_THROWS_AS(it[1], json::invalid_iterator); + CHECK_THROWS_AS(it[0], json::invalid_iterator&); + CHECK_THROWS_AS(it[1], json::invalid_iterator&); CHECK_THROWS_WITH(it[0], "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); CHECK_THROWS_WITH(it[1], "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } { auto it = j_object.crbegin(); - CHECK_THROWS_AS(it[0], json::invalid_iterator); - CHECK_THROWS_AS(it[1], json::invalid_iterator); + CHECK_THROWS_AS(it[0], json::invalid_iterator&); + CHECK_THROWS_AS(it[1], json::invalid_iterator&); CHECK_THROWS_WITH(it[0], "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); CHECK_THROWS_WITH(it[1], "[json.exception.invalid_iterator.209] cannot use offsets with object iterators"); } @@ -871,15 +871,15 @@ TEST_CASE("iterators 2") { { auto it = j_null.rbegin(); - CHECK_THROWS_AS(it[0], json::invalid_iterator); - CHECK_THROWS_AS(it[1], json::invalid_iterator); + CHECK_THROWS_AS(it[0], json::invalid_iterator&); + CHECK_THROWS_AS(it[1], json::invalid_iterator&); CHECK_THROWS_WITH(it[0], "[json.exception.invalid_iterator.214] cannot get value"); CHECK_THROWS_WITH(it[1], "[json.exception.invalid_iterator.214] cannot get value"); } { auto it = j_null.crbegin(); - CHECK_THROWS_AS(it[0], json::invalid_iterator); - CHECK_THROWS_AS(it[1], json::invalid_iterator); + CHECK_THROWS_AS(it[0], json::invalid_iterator&); + CHECK_THROWS_AS(it[1], json::invalid_iterator&); CHECK_THROWS_WITH(it[0], "[json.exception.invalid_iterator.214] cannot get value"); CHECK_THROWS_WITH(it[1], "[json.exception.invalid_iterator.214] cannot get value"); } @@ -890,13 +890,13 @@ TEST_CASE("iterators 2") { auto it = j_value.rbegin(); CHECK(it[0] == json(42)); - CHECK_THROWS_AS(it[1], json::invalid_iterator); + CHECK_THROWS_AS(it[1], json::invalid_iterator&); CHECK_THROWS_WITH(it[1], "[json.exception.invalid_iterator.214] cannot get value"); } { auto it = j_value.crbegin(); CHECK(it[0] == json(42)); - CHECK_THROWS_AS(it[1], json::invalid_iterator); + CHECK_THROWS_AS(it[1], json::invalid_iterator&); CHECK_THROWS_WITH(it[1], "[json.exception.invalid_iterator.214] cannot get value"); } } diff --git a/test/src/unit-json_patch.cpp b/test/src/unit-json_patch.cpp index 9ab86ea4..636fa2bc 100644 --- a/test/src/unit-json_patch.cpp +++ b/test/src/unit-json_patch.cpp @@ -75,7 +75,7 @@ TEST_CASE("JSON patch") json doc2 = R"({ "q": { "bar": 2 } })"_json; // because "a" does not exist. - CHECK_THROWS_AS(doc2.patch(patch), json::out_of_range); + CHECK_THROWS_AS(doc2.patch(patch), json::out_of_range&); CHECK_THROWS_WITH(doc2.patch(patch), "[json.exception.out_of_range.403] key 'a' not found"); } @@ -337,7 +337,7 @@ TEST_CASE("JSON patch") )"_json; // check that evaluation throws - CHECK_THROWS_AS(doc.patch(patch), json::other_error); + CHECK_THROWS_AS(doc.patch(patch), json::other_error&); CHECK_THROWS_WITH(doc.patch(patch), "[json.exception.other_error.501] unsuccessful: " + patch[0].dump()); } @@ -421,7 +421,7 @@ TEST_CASE("JSON patch") // references neither the root of the document, nor a member of // an existing object, nor a member of an existing array. - CHECK_THROWS_AS(doc.patch(patch), json::out_of_range); + CHECK_THROWS_AS(doc.patch(patch), json::out_of_range&); CHECK_THROWS_WITH(doc.patch(patch), "[json.exception.out_of_range.403] key 'baz' not found"); } @@ -478,7 +478,7 @@ TEST_CASE("JSON patch") )"_json; // check that evaluation throws - CHECK_THROWS_AS(doc.patch(patch), json::other_error); + CHECK_THROWS_AS(doc.patch(patch), json::other_error&); CHECK_THROWS_WITH(doc.patch(patch), "[json.exception.other_error.501] unsuccessful: " + patch[0].dump()); } @@ -668,7 +668,7 @@ TEST_CASE("JSON patch") { json j; json patch = {{"op", "add"}, {"path", ""}, {"value", 1}}; - CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_AS(j.patch(patch), json::parse_error&); CHECK_THROWS_WITH(j.patch(patch), "[json.exception.parse_error.104] parse error: JSON patch must be an array of objects"); } @@ -677,7 +677,7 @@ TEST_CASE("JSON patch") { json j; json patch = {"op", "add", "path", "", "value", 1}; - CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_AS(j.patch(patch), json::parse_error&); CHECK_THROWS_WITH(j.patch(patch), "[json.exception.parse_error.104] parse error: JSON patch must be an array of objects"); } @@ -686,7 +686,7 @@ TEST_CASE("JSON patch") { json j; json patch = {{{"foo", "bar"}}}; - CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_AS(j.patch(patch), json::parse_error&); CHECK_THROWS_WITH(j.patch(patch), "[json.exception.parse_error.105] parse error: operation must have member 'op'"); } @@ -695,7 +695,7 @@ TEST_CASE("JSON patch") { json j; json patch = {{{"op", 1}}}; - CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_AS(j.patch(patch), json::parse_error&); CHECK_THROWS_WITH(j.patch(patch), "[json.exception.parse_error.105] parse error: operation must have string member 'op'"); } @@ -704,7 +704,7 @@ TEST_CASE("JSON patch") { json j; json patch = {{{"op", "foo"}, {"path", ""}}}; - CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_AS(j.patch(patch), json::parse_error&); CHECK_THROWS_WITH(j.patch(patch), "[json.exception.parse_error.105] parse error: operation value 'foo' is invalid"); } @@ -716,7 +716,7 @@ TEST_CASE("JSON patch") { json j; json patch = {{{"op", "add"}}}; - CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_AS(j.patch(patch), json::parse_error&); CHECK_THROWS_WITH(j.patch(patch), "[json.exception.parse_error.105] parse error: operation 'add' must have member 'path'"); } @@ -725,7 +725,7 @@ TEST_CASE("JSON patch") { json j; json patch = {{{"op", "add"}, {"path", 1}}}; - CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_AS(j.patch(patch), json::parse_error&); CHECK_THROWS_WITH(j.patch(patch), "[json.exception.parse_error.105] parse error: operation 'add' must have string member 'path'"); } @@ -734,7 +734,7 @@ TEST_CASE("JSON patch") { json j; json patch = {{{"op", "add"}, {"path", ""}}}; - CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_AS(j.patch(patch), json::parse_error&); CHECK_THROWS_WITH(j.patch(patch), "[json.exception.parse_error.105] parse error: operation 'add' must have member 'value'"); } @@ -743,7 +743,7 @@ TEST_CASE("JSON patch") { json j = {1, 2}; json patch = {{{"op", "add"}, {"path", "/4"}, {"value", 4}}}; - CHECK_THROWS_AS(j.patch(patch), json::out_of_range); + CHECK_THROWS_AS(j.patch(patch), json::out_of_range&); CHECK_THROWS_WITH(j.patch(patch), "[json.exception.out_of_range.401] array index 4 is out of range"); } @@ -755,7 +755,7 @@ TEST_CASE("JSON patch") { json j; json patch = {{{"op", "remove"}}}; - CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_AS(j.patch(patch), json::parse_error&); CHECK_THROWS_WITH(j.patch(patch), "[json.exception.parse_error.105] parse error: operation 'remove' must have member 'path'"); } @@ -764,7 +764,7 @@ TEST_CASE("JSON patch") { json j; json patch = {{{"op", "remove"}, {"path", 1}}}; - CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_AS(j.patch(patch), json::parse_error&); CHECK_THROWS_WITH(j.patch(patch), "[json.exception.parse_error.105] parse error: operation 'remove' must have string member 'path'"); } @@ -773,7 +773,7 @@ TEST_CASE("JSON patch") { json j = {1, 2, 3}; json patch = {{{"op", "remove"}, {"path", "/17"}}}; - CHECK_THROWS_AS(j.patch(patch), json::out_of_range); + CHECK_THROWS_AS(j.patch(patch), json::out_of_range&); CHECK_THROWS_WITH(j.patch(patch), "[json.exception.out_of_range.401] array index 17 is out of range"); } @@ -782,7 +782,7 @@ TEST_CASE("JSON patch") { json j = {{"foo", 1}, {"bar", 2}}; json patch = {{{"op", "remove"}, {"path", "/baz"}}}; - CHECK_THROWS_AS(j.patch(patch), json::out_of_range); + CHECK_THROWS_AS(j.patch(patch), json::out_of_range&); CHECK_THROWS_WITH(j.patch(patch), "[json.exception.out_of_range.403] key 'baz' not found"); } @@ -791,7 +791,7 @@ TEST_CASE("JSON patch") { json j = "string"; json patch = {{{"op", "remove"}, {"path", ""}}}; - CHECK_THROWS_AS(j.patch(patch), json::out_of_range); + CHECK_THROWS_AS(j.patch(patch), json::out_of_range&); CHECK_THROWS_WITH(j.patch(patch), "[json.exception.out_of_range.405] JSON pointer has no parent"); } @@ -803,7 +803,7 @@ TEST_CASE("JSON patch") { json j; json patch = {{{"op", "replace"}}}; - CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_AS(j.patch(patch), json::parse_error&); CHECK_THROWS_WITH(j.patch(patch), "[json.exception.parse_error.105] parse error: operation 'replace' must have member 'path'"); } @@ -812,7 +812,7 @@ TEST_CASE("JSON patch") { json j; json patch = {{{"op", "replace"}, {"path", 1}}}; - CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_AS(j.patch(patch), json::parse_error&); CHECK_THROWS_WITH(j.patch(patch), "[json.exception.parse_error.105] parse error: operation 'replace' must have string member 'path'"); } @@ -821,7 +821,7 @@ TEST_CASE("JSON patch") { json j; json patch = {{{"op", "replace"}, {"path", ""}}}; - CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_AS(j.patch(patch), json::parse_error&); CHECK_THROWS_WITH(j.patch(patch), "[json.exception.parse_error.105] parse error: operation 'replace' must have member 'value'"); } @@ -830,7 +830,7 @@ TEST_CASE("JSON patch") { json j = {1, 2, 3}; json patch = {{{"op", "replace"}, {"path", "/17"}, {"value", 19}}}; - CHECK_THROWS_AS(j.patch(patch), json::out_of_range); + CHECK_THROWS_AS(j.patch(patch), json::out_of_range&); CHECK_THROWS_WITH(j.patch(patch), "[json.exception.out_of_range.401] array index 17 is out of range"); } @@ -839,7 +839,7 @@ TEST_CASE("JSON patch") { json j = {{"foo", 1}, {"bar", 2}}; json patch = {{{"op", "replace"}, {"path", "/baz"}, {"value", 3}}}; - CHECK_THROWS_AS(j.patch(patch), json::out_of_range); + CHECK_THROWS_AS(j.patch(patch), json::out_of_range&); CHECK_THROWS_WITH(j.patch(patch), "[json.exception.out_of_range.403] key 'baz' not found"); } @@ -851,7 +851,7 @@ TEST_CASE("JSON patch") { json j; json patch = {{{"op", "move"}}}; - CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_AS(j.patch(patch), json::parse_error&); CHECK_THROWS_WITH(j.patch(patch), "[json.exception.parse_error.105] parse error: operation 'move' must have member 'path'"); } @@ -860,7 +860,7 @@ TEST_CASE("JSON patch") { json j; json patch = {{{"op", "move"}, {"path", 1}}}; - CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_AS(j.patch(patch), json::parse_error&); CHECK_THROWS_WITH(j.patch(patch), "[json.exception.parse_error.105] parse error: operation 'move' must have string member 'path'"); } @@ -869,7 +869,7 @@ TEST_CASE("JSON patch") { json j; json patch = {{{"op", "move"}, {"path", ""}}}; - CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_AS(j.patch(patch), json::parse_error&); CHECK_THROWS_WITH(j.patch(patch), "[json.exception.parse_error.105] parse error: operation 'move' must have member 'from'"); } @@ -878,7 +878,7 @@ TEST_CASE("JSON patch") { json j; json patch = {{{"op", "move"}, {"path", ""}, {"from", 1}}}; - CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_AS(j.patch(patch), json::parse_error&); CHECK_THROWS_WITH(j.patch(patch), "[json.exception.parse_error.105] parse error: operation 'move' must have string member 'from'"); } @@ -887,7 +887,7 @@ TEST_CASE("JSON patch") { json j = {1, 2, 3}; json patch = {{{"op", "move"}, {"path", "/0"}, {"from", "/5"}}}; - CHECK_THROWS_AS(j.patch(patch), json::out_of_range); + CHECK_THROWS_AS(j.patch(patch), json::out_of_range&); CHECK_THROWS_WITH(j.patch(patch), "[json.exception.out_of_range.401] array index 5 is out of range"); } @@ -896,7 +896,7 @@ TEST_CASE("JSON patch") { json j = {{"foo", 1}, {"bar", 2}}; json patch = {{{"op", "move"}, {"path", "/baz"}, {"from", "/baz"}}}; - CHECK_THROWS_AS(j.patch(patch), json::out_of_range); + CHECK_THROWS_AS(j.patch(patch), json::out_of_range&); CHECK_THROWS_WITH(j.patch(patch), "[json.exception.out_of_range.403] key 'baz' not found"); } @@ -908,7 +908,7 @@ TEST_CASE("JSON patch") { json j; json patch = {{{"op", "copy"}}}; - CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_AS(j.patch(patch), json::parse_error&); CHECK_THROWS_WITH(j.patch(patch), "[json.exception.parse_error.105] parse error: operation 'copy' must have member 'path'"); } @@ -917,7 +917,7 @@ TEST_CASE("JSON patch") { json j; json patch = {{{"op", "copy"}, {"path", 1}}}; - CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_AS(j.patch(patch), json::parse_error&); CHECK_THROWS_WITH(j.patch(patch), "[json.exception.parse_error.105] parse error: operation 'copy' must have string member 'path'"); } @@ -926,7 +926,7 @@ TEST_CASE("JSON patch") { json j; json patch = {{{"op", "copy"}, {"path", ""}}}; - CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_AS(j.patch(patch), json::parse_error&); CHECK_THROWS_WITH(j.patch(patch), "[json.exception.parse_error.105] parse error: operation 'copy' must have member 'from'"); } @@ -935,7 +935,7 @@ TEST_CASE("JSON patch") { json j; json patch = {{{"op", "copy"}, {"path", ""}, {"from", 1}}}; - CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_AS(j.patch(patch), json::parse_error&); CHECK_THROWS_WITH(j.patch(patch), "[json.exception.parse_error.105] parse error: operation 'copy' must have string member 'from'"); } @@ -944,7 +944,7 @@ TEST_CASE("JSON patch") { json j = {1, 2, 3}; json patch = {{{"op", "copy"}, {"path", "/0"}, {"from", "/5"}}}; - CHECK_THROWS_AS(j.patch(patch), json::out_of_range); + CHECK_THROWS_AS(j.patch(patch), json::out_of_range&); CHECK_THROWS_WITH(j.patch(patch), "[json.exception.out_of_range.401] array index 5 is out of range"); } @@ -953,7 +953,7 @@ TEST_CASE("JSON patch") { json j = {{"foo", 1}, {"bar", 2}}; json patch = {{{"op", "copy"}, {"path", "/fob"}, {"from", "/baz"}}}; - CHECK_THROWS_AS(j.patch(patch), json::out_of_range); + CHECK_THROWS_AS(j.patch(patch), json::out_of_range&); CHECK_THROWS_WITH(j.patch(patch), "[json.exception.out_of_range.403] key 'baz' not found"); } @@ -965,7 +965,7 @@ TEST_CASE("JSON patch") { json j; json patch = {{{"op", "test"}}}; - CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_AS(j.patch(patch), json::parse_error&); CHECK_THROWS_WITH(j.patch(patch), "[json.exception.parse_error.105] parse error: operation 'test' must have member 'path'"); } @@ -974,7 +974,7 @@ TEST_CASE("JSON patch") { json j; json patch = {{{"op", "test"}, {"path", 1}}}; - CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_AS(j.patch(patch), json::parse_error&); CHECK_THROWS_WITH(j.patch(patch), "[json.exception.parse_error.105] parse error: operation 'test' must have string member 'path'"); } @@ -983,7 +983,7 @@ TEST_CASE("JSON patch") { json j; json patch = {{{"op", "test"}, {"path", ""}}}; - CHECK_THROWS_AS(j.patch(patch), json::parse_error); + CHECK_THROWS_AS(j.patch(patch), json::parse_error&); CHECK_THROWS_WITH(j.patch(patch), "[json.exception.parse_error.105] parse error: operation 'test' must have member 'value'"); } @@ -1177,7 +1177,7 @@ TEST_CASE("JSON patch") )"_json; // the test will fail - CHECK_THROWS_AS(doc.patch(patch), json::other_error); + CHECK_THROWS_AS(doc.patch(patch), json::other_error&); CHECK_THROWS_WITH(doc.patch(patch), "[json.exception.other_error.501] unsuccessful: " + patch[0].dump()); } } diff --git a/test/src/unit-json_pointer.cpp b/test/src/unit-json_pointer.cpp index d14261a0..a9fbc613 100644 --- a/test/src/unit-json_pointer.cpp +++ b/test/src/unit-json_pointer.cpp @@ -36,23 +36,23 @@ TEST_CASE("JSON pointers") { SECTION("errors") { - CHECK_THROWS_AS(json::json_pointer("foo"), json::parse_error); + CHECK_THROWS_AS(json::json_pointer("foo"), json::parse_error&); CHECK_THROWS_WITH(json::json_pointer("foo"), "[json.exception.parse_error.107] parse error at 1: JSON pointer must be empty or begin with '/' - was: 'foo'"); - CHECK_THROWS_AS(json::json_pointer("/~~"), json::parse_error); + CHECK_THROWS_AS(json::json_pointer("/~~"), json::parse_error&); CHECK_THROWS_WITH(json::json_pointer("/~~"), "[json.exception.parse_error.108] parse error: escape character '~' must be followed with '0' or '1'"); - CHECK_THROWS_AS(json::json_pointer("/~"), json::parse_error); + CHECK_THROWS_AS(json::json_pointer("/~"), json::parse_error&); CHECK_THROWS_WITH(json::json_pointer("/~"), "[json.exception.parse_error.108] parse error: escape character '~' must be followed with '0' or '1'"); json::json_pointer p; - CHECK_THROWS_AS(p.top(), json::out_of_range); + CHECK_THROWS_AS(p.top(), json::out_of_range&); CHECK_THROWS_WITH(p.top(), "[json.exception.out_of_range.405] JSON pointer has no parent"); - CHECK_THROWS_AS(p.pop_back(), json::out_of_range); + CHECK_THROWS_AS(p.pop_back(), json::out_of_range&); CHECK_THROWS_WITH(p.pop_back(), "[json.exception.out_of_range.405] JSON pointer has no parent"); } @@ -126,10 +126,10 @@ TEST_CASE("JSON pointers") // unresolved access json j_primitive = 1; - CHECK_THROWS_AS(j_primitive["/foo"_json_pointer], json::out_of_range); + CHECK_THROWS_AS(j_primitive["/foo"_json_pointer], json::out_of_range&); CHECK_THROWS_WITH(j_primitive["/foo"_json_pointer], "[json.exception.out_of_range.404] unresolved reference token 'foo'"); - CHECK_THROWS_AS(j_primitive.at("/foo"_json_pointer), json::out_of_range); + CHECK_THROWS_AS(j_primitive.at("/foo"_json_pointer), json::out_of_range&); CHECK_THROWS_WITH(j_primitive.at("/foo"_json_pointer), "[json.exception.out_of_range.404] unresolved reference token 'foo'"); } @@ -189,16 +189,16 @@ TEST_CASE("JSON pointers") CHECK(j[json::json_pointer("/m~0n")] == j["m~n"]); // unescaped access - CHECK_THROWS_AS(j.at(json::json_pointer("/a/b")), json::out_of_range); + CHECK_THROWS_AS(j.at(json::json_pointer("/a/b")), json::out_of_range&); CHECK_THROWS_WITH(j.at(json::json_pointer("/a/b")), "[json.exception.out_of_range.403] key 'a' not found"); // unresolved access const json j_primitive = 1; - CHECK_THROWS_AS(j_primitive["/foo"_json_pointer], json::out_of_range); + CHECK_THROWS_AS(j_primitive["/foo"_json_pointer], json::out_of_range&); CHECK_THROWS_WITH(j_primitive["/foo"_json_pointer], "[json.exception.out_of_range.404] unresolved reference token 'foo'"); - CHECK_THROWS_AS(j_primitive.at("/foo"_json_pointer), json::out_of_range); + CHECK_THROWS_AS(j_primitive.at("/foo"_json_pointer), json::out_of_range&); CHECK_THROWS_WITH(j_primitive.at("/foo"_json_pointer), "[json.exception.out_of_range.404] unresolved reference token 'foo'"); } @@ -255,35 +255,35 @@ TEST_CASE("JSON pointers") CHECK(j == json({1, 13, 3, 33, nullptr, 55})); // error with leading 0 - CHECK_THROWS_AS(j["/01"_json_pointer], json::parse_error); + CHECK_THROWS_AS(j["/01"_json_pointer], json::parse_error&); CHECK_THROWS_WITH(j["/01"_json_pointer], "[json.exception.parse_error.106] parse error: array index '01' must not begin with '0'"); - CHECK_THROWS_AS(j_const["/01"_json_pointer], json::parse_error); + CHECK_THROWS_AS(j_const["/01"_json_pointer], json::parse_error&); CHECK_THROWS_WITH(j_const["/01"_json_pointer], "[json.exception.parse_error.106] parse error: array index '01' must not begin with '0'"); - CHECK_THROWS_AS(j.at("/01"_json_pointer), json::parse_error); + CHECK_THROWS_AS(j.at("/01"_json_pointer), json::parse_error&); CHECK_THROWS_WITH(j.at("/01"_json_pointer), "[json.exception.parse_error.106] parse error: array index '01' must not begin with '0'"); - CHECK_THROWS_AS(j_const.at("/01"_json_pointer), json::parse_error); + CHECK_THROWS_AS(j_const.at("/01"_json_pointer), json::parse_error&); CHECK_THROWS_WITH(j_const.at("/01"_json_pointer), "[json.exception.parse_error.106] parse error: array index '01' must not begin with '0'"); // error with incorrect numbers - CHECK_THROWS_AS(j["/one"_json_pointer] = 1, json::parse_error); + CHECK_THROWS_AS(j["/one"_json_pointer] = 1, json::parse_error&); CHECK_THROWS_WITH(j["/one"_json_pointer] = 1, "[json.exception.parse_error.109] parse error: array index 'one' is not a number"); - CHECK_THROWS_AS(j_const["/one"_json_pointer] == 1, json::parse_error); + CHECK_THROWS_AS(j_const["/one"_json_pointer] == 1, json::parse_error&); CHECK_THROWS_WITH(j_const["/one"_json_pointer] == 1, "[json.exception.parse_error.109] parse error: array index 'one' is not a number"); - CHECK_THROWS_AS(j.at("/one"_json_pointer) = 1, json::parse_error); + CHECK_THROWS_AS(j.at("/one"_json_pointer) = 1, json::parse_error&); CHECK_THROWS_WITH(j.at("/one"_json_pointer) = 1, "[json.exception.parse_error.109] parse error: array index 'one' is not a number"); - CHECK_THROWS_AS(j_const.at("/one"_json_pointer) == 1, json::parse_error); + CHECK_THROWS_AS(j_const.at("/one"_json_pointer) == 1, json::parse_error&); CHECK_THROWS_WITH(j_const.at("/one"_json_pointer) == 1, "[json.exception.parse_error.109] parse error: array index 'one' is not a number"); - CHECK_THROWS_AS(json({{"/list/0", 1}, {"/list/1", 2}, {"/list/three", 3}}).unflatten(), json::parse_error); + CHECK_THROWS_AS(json({{"/list/0", 1}, {"/list/1", 2}, {"/list/three", 3}}).unflatten(), json::parse_error&); CHECK_THROWS_WITH(json({{"/list/0", 1}, {"/list/1", 2}, {"/list/three", 3}}).unflatten(), "[json.exception.parse_error.109] parse error: array index 'three' is not a number"); @@ -292,15 +292,15 @@ TEST_CASE("JSON pointers") CHECK(j == json({1, 13, 3, 33, nullptr, 55, 99})); // error when using "-" in const object - CHECK_THROWS_AS(j_const["/-"_json_pointer], json::out_of_range); + CHECK_THROWS_AS(j_const["/-"_json_pointer], json::out_of_range&); CHECK_THROWS_WITH(j_const["/-"_json_pointer], "[json.exception.out_of_range.402] array index '-' (3) is out of range"); // error when using "-" with at - CHECK_THROWS_AS(j.at("/-"_json_pointer), json::out_of_range); + CHECK_THROWS_AS(j.at("/-"_json_pointer), json::out_of_range&); CHECK_THROWS_WITH(j.at("/-"_json_pointer), "[json.exception.out_of_range.402] array index '-' (7) is out of range"); - CHECK_THROWS_AS(j_const.at("/-"_json_pointer), json::out_of_range); + CHECK_THROWS_AS(j_const.at("/-"_json_pointer), json::out_of_range&); CHECK_THROWS_WITH(j_const.at("/-"_json_pointer), "[json.exception.out_of_range.402] array index '-' (3) is out of range"); } @@ -315,20 +315,20 @@ TEST_CASE("JSON pointers") CHECK(j["/2"_json_pointer] == j[2]); // assign to nonexisting index - CHECK_THROWS_AS(j.at("/3"_json_pointer), json::out_of_range); + CHECK_THROWS_AS(j.at("/3"_json_pointer), json::out_of_range&); CHECK_THROWS_WITH(j.at("/3"_json_pointer), "[json.exception.out_of_range.401] array index 3 is out of range"); // assign to nonexisting index (with gap) - CHECK_THROWS_AS(j.at("/5"_json_pointer), json::out_of_range); + CHECK_THROWS_AS(j.at("/5"_json_pointer), json::out_of_range&); CHECK_THROWS_WITH(j.at("/5"_json_pointer), "[json.exception.out_of_range.401] array index 5 is out of range"); // assign to "-" - CHECK_THROWS_AS(j["/-"_json_pointer], json::out_of_range); + CHECK_THROWS_AS(j["/-"_json_pointer], json::out_of_range&); CHECK_THROWS_WITH(j["/-"_json_pointer], "[json.exception.out_of_range.402] array index '-' (3) is out of range"); - CHECK_THROWS_AS(j.at("/-"_json_pointer), json::out_of_range); + CHECK_THROWS_AS(j.at("/-"_json_pointer), json::out_of_range&); CHECK_THROWS_WITH(j.at("/-"_json_pointer), "[json.exception.out_of_range.402] array index '-' (3) is out of range"); } @@ -386,18 +386,18 @@ TEST_CASE("JSON pointers") CHECK(j_flatten.unflatten() == j); // error for nonobjects - CHECK_THROWS_AS(json(1).unflatten(), json::type_error); + CHECK_THROWS_AS(json(1).unflatten(), json::type_error&); CHECK_THROWS_WITH(json(1).unflatten(), "[json.exception.type_error.314] only objects can be unflattened"); // error for nonprimitve values - CHECK_THROWS_AS(json({{"/1", {1, 2, 3}}}).unflatten(), json::type_error); + CHECK_THROWS_AS(json({{"/1", {1, 2, 3}}}).unflatten(), json::type_error&); CHECK_THROWS_WITH(json({{"/1", {1, 2, 3}}}).unflatten(), "[json.exception.type_error.315] values in object must be primitive"); // error for conflicting values json j_error = {{"", 42}, {"/foo", 17}}; - CHECK_THROWS_AS(j_error.unflatten(), json::type_error); + CHECK_THROWS_AS(j_error.unflatten(), json::type_error&); CHECK_THROWS_WITH(j_error.unflatten(), "[json.exception.type_error.313] invalid value to unflatten"); diff --git a/test/src/unit-modifiers.cpp b/test/src/unit-modifiers.cpp index b860beee..eeb70a0c 100644 --- a/test/src/unit-modifiers.cpp +++ b/test/src/unit-modifiers.cpp @@ -152,7 +152,7 @@ TEST_CASE("modifiers") SECTION("other type") { json j = 1; - CHECK_THROWS_AS(j.push_back("Hello"), json::type_error); + CHECK_THROWS_AS(j.push_back("Hello"), json::type_error&); CHECK_THROWS_WITH(j.push_back("Hello"), "[json.exception.type_error.308] cannot use push_back() with number"); } } @@ -182,7 +182,7 @@ TEST_CASE("modifiers") { json j = 1; json k("Hello"); - CHECK_THROWS_AS(j.push_back(k), json::type_error); + CHECK_THROWS_AS(j.push_back(k), json::type_error&); CHECK_THROWS_WITH(j.push_back(k), "[json.exception.type_error.308] cannot use push_back() with number"); } } @@ -215,7 +215,7 @@ TEST_CASE("modifiers") { json j = 1; json k("Hello"); - CHECK_THROWS_AS(j.push_back(json::object_t::value_type({"one", 1})), json::type_error); + CHECK_THROWS_AS(j.push_back(json::object_t::value_type({"one", 1})), json::type_error&); CHECK_THROWS_WITH(j.push_back(json::object_t::value_type({"one", 1})), "[json.exception.type_error.308] cannot use push_back() with number"); } @@ -252,7 +252,7 @@ TEST_CASE("modifiers") CHECK(j == json({{"key1", 1}, {"key2", "bar"}})); json k = {{"key1", 1}}; - CHECK_THROWS_AS(k.push_back({1, 2, 3, 4}), json::type_error); + CHECK_THROWS_AS(k.push_back({1, 2, 3, 4}), json::type_error&); CHECK_THROWS_WITH(k.push_back({1, 2, 3, 4}), "[json.exception.type_error.308] cannot use push_back() with object"); } } @@ -291,7 +291,7 @@ TEST_CASE("modifiers") SECTION("other type") { json j = 1; - CHECK_THROWS_AS(j.emplace_back("Hello"), json::type_error); + CHECK_THROWS_AS(j.emplace_back("Hello"), json::type_error&); CHECK_THROWS_WITH(j.emplace_back("Hello"), "[json.exception.type_error.311] cannot use emplace_back() with number"); } @@ -351,7 +351,7 @@ TEST_CASE("modifiers") SECTION("other type") { json j = 1; - CHECK_THROWS_AS(j.emplace("foo", "bar"), json::type_error); + CHECK_THROWS_AS(j.emplace("foo", "bar"), json::type_error&); CHECK_THROWS_WITH(j.emplace("foo", "bar"), "[json.exception.type_error.311] cannot use emplace() with number"); } @@ -383,7 +383,7 @@ TEST_CASE("modifiers") SECTION("other type") { json j = 1; - CHECK_THROWS_AS(j += "Hello", json::type_error); + CHECK_THROWS_AS(j += "Hello", json::type_error&); CHECK_THROWS_WITH(j += "Hello", "[json.exception.type_error.308] cannot use push_back() with number"); } } @@ -413,7 +413,7 @@ TEST_CASE("modifiers") { json j = 1; json k("Hello"); - CHECK_THROWS_AS(j += k, json::type_error); + CHECK_THROWS_AS(j += k, json::type_error&); CHECK_THROWS_WITH(j += k, "[json.exception.type_error.308] cannot use push_back() with number"); } } @@ -446,7 +446,7 @@ TEST_CASE("modifiers") { json j = 1; json k("Hello"); - CHECK_THROWS_AS(j += json::object_t::value_type({"one", 1}), json::type_error); + CHECK_THROWS_AS(j += json::object_t::value_type({"one", 1}), json::type_error&); CHECK_THROWS_WITH(j += json::object_t::value_type({"one", 1}), "[json.exception.type_error.308] cannot use push_back() with number"); } @@ -483,7 +483,7 @@ TEST_CASE("modifiers") CHECK(j == json({{"key1", 1}, {"key2", "bar"}})); json k = {{"key1", 1}}; - CHECK_THROWS_AS((k += {1, 2, 3, 4}), json::type_error); + CHECK_THROWS_AS((k += {1, 2, 3, 4}), json::type_error&); CHECK_THROWS_WITH((k += {1, 2, 3, 4}), "[json.exception.type_error.308] cannot use push_back() with object"); } } @@ -620,9 +620,9 @@ TEST_CASE("modifiers") json j_other_array2 = {"first", "second"}; CHECK_THROWS_AS(j_array.insert(j_array.end(), j_array.begin(), j_array.end()), - json::invalid_iterator); + json::invalid_iterator&); CHECK_THROWS_AS(j_array.insert(j_array.end(), j_other_array.begin(), j_other_array2.end()), - json::invalid_iterator); + json::invalid_iterator&); CHECK_THROWS_WITH(j_array.insert(j_array.end(), j_array.begin(), j_array.end()), "[json.exception.invalid_iterator.211] passed iterators may not belong to container"); @@ -652,9 +652,9 @@ TEST_CASE("modifiers") { json j_other_array2 = {"first", "second"}; - CHECK_THROWS_AS(j_array.insert(j_object2.begin(), j_object2.end()), json::type_error); - CHECK_THROWS_AS(j_object1.insert(j_object1.begin(), j_object2.end()), json::invalid_iterator); - CHECK_THROWS_AS(j_object1.insert(j_array.begin(), j_array.end()), json::invalid_iterator); + CHECK_THROWS_AS(j_array.insert(j_object2.begin(), j_object2.end()), json::type_error&); + CHECK_THROWS_AS(j_object1.insert(j_object1.begin(), j_object2.end()), json::invalid_iterator&); + CHECK_THROWS_AS(j_object1.insert(j_array.begin(), j_array.end()), json::invalid_iterator&); CHECK_THROWS_WITH(j_array.insert(j_object2.begin(), j_object2.end()), "[json.exception.type_error.309] cannot use insert() with array"); @@ -700,11 +700,11 @@ TEST_CASE("modifiers") // pass iterator to a different array json j_another_array = {1, 2}; json j_yet_another_array = {"first", "second"}; - CHECK_THROWS_AS(j_array.insert(j_another_array.end(), 10), json::invalid_iterator); - CHECK_THROWS_AS(j_array.insert(j_another_array.end(), j_value), json::invalid_iterator); - CHECK_THROWS_AS(j_array.insert(j_another_array.end(), 10, 11), json::invalid_iterator); - CHECK_THROWS_AS(j_array.insert(j_another_array.end(), j_yet_another_array.begin(), j_yet_another_array.end()), json::invalid_iterator); - CHECK_THROWS_AS(j_array.insert(j_another_array.end(), {1, 2, 3, 4}), json::invalid_iterator); + CHECK_THROWS_AS(j_array.insert(j_another_array.end(), 10), json::invalid_iterator&); + CHECK_THROWS_AS(j_array.insert(j_another_array.end(), j_value), json::invalid_iterator&); + CHECK_THROWS_AS(j_array.insert(j_another_array.end(), 10, 11), json::invalid_iterator&); + CHECK_THROWS_AS(j_array.insert(j_another_array.end(), j_yet_another_array.begin(), j_yet_another_array.end()), json::invalid_iterator&); + CHECK_THROWS_AS(j_array.insert(j_another_array.end(), {1, 2, 3, 4}), json::invalid_iterator&); CHECK_THROWS_WITH(j_array.insert(j_another_array.end(), 10), "[json.exception.invalid_iterator.202] iterator does not fit current value"); @@ -723,12 +723,12 @@ TEST_CASE("modifiers") // call insert on a non-array type json j_nonarray = 3; json j_yet_another_array = {"first", "second"}; - CHECK_THROWS_AS(j_nonarray.insert(j_nonarray.end(), 10), json::type_error); - CHECK_THROWS_AS(j_nonarray.insert(j_nonarray.end(), j_value), json::type_error); - CHECK_THROWS_AS(j_nonarray.insert(j_nonarray.end(), 10, 11), json::type_error); + CHECK_THROWS_AS(j_nonarray.insert(j_nonarray.end(), 10), json::type_error&); + CHECK_THROWS_AS(j_nonarray.insert(j_nonarray.end(), j_value), json::type_error&); + CHECK_THROWS_AS(j_nonarray.insert(j_nonarray.end(), 10, 11), json::type_error&); CHECK_THROWS_AS(j_nonarray.insert(j_nonarray.end(), j_yet_another_array.begin(), - j_yet_another_array.end()), json::type_error); - CHECK_THROWS_AS(j_nonarray.insert(j_nonarray.end(), {1, 2, 3, 4}), json::type_error); + j_yet_another_array.end()), json::type_error&); + CHECK_THROWS_AS(j_nonarray.insert(j_nonarray.end(), {1, 2, 3, 4}), json::type_error&); CHECK_THROWS_WITH(j_nonarray.insert(j_nonarray.end(), 10), "[json.exception.type_error.309] cannot use insert() with number"); CHECK_THROWS_WITH(j_nonarray.insert(j_nonarray.end(), j_value), "[json.exception.type_error.309] cannot use insert() with number"); @@ -788,7 +788,7 @@ TEST_CASE("modifiers") json j = 17; json::array_t a = {"foo", "bar", "baz"}; - CHECK_THROWS_AS(j.swap(a), json::type_error); + CHECK_THROWS_AS(j.swap(a), json::type_error&); CHECK_THROWS_WITH(j.swap(a), "[json.exception.type_error.310] cannot use swap() with number"); } } @@ -814,7 +814,7 @@ TEST_CASE("modifiers") json j = 17; json::object_t o = {{"cow", "Kuh"}, {"chicken", "Huhn"}}; - CHECK_THROWS_AS(j.swap(o), json::type_error); + CHECK_THROWS_AS(j.swap(o), json::type_error&); CHECK_THROWS_WITH(j.swap(o), "[json.exception.type_error.310] cannot use swap() with number"); } } @@ -840,7 +840,7 @@ TEST_CASE("modifiers") json j = 17; json::string_t s = "Hallo Welt"; - CHECK_THROWS_AS(j.swap(s), json::type_error); + CHECK_THROWS_AS(j.swap(s), json::type_error&); CHECK_THROWS_WITH(j.swap(s), "[json.exception.type_error.310] cannot use swap() with number"); } } diff --git a/test/src/unit-msgpack.cpp b/test/src/unit-msgpack.cpp index 5b1c301d..baef3e35 100644 --- a/test/src/unit-msgpack.cpp +++ b/test/src/unit-msgpack.cpp @@ -1016,21 +1016,21 @@ TEST_CASE("MessagePack") { SECTION("too short byte vector") { - CHECK_THROWS_AS(json::from_msgpack(std::vector({0xcc})), json::parse_error); - CHECK_THROWS_AS(json::from_msgpack(std::vector({0xcd})), json::parse_error); - CHECK_THROWS_AS(json::from_msgpack(std::vector({0xcd, 0x00})), json::parse_error); - CHECK_THROWS_AS(json::from_msgpack(std::vector({0xce})), json::parse_error); - CHECK_THROWS_AS(json::from_msgpack(std::vector({0xce, 0x00})), json::parse_error); - CHECK_THROWS_AS(json::from_msgpack(std::vector({0xce, 0x00, 0x00})), json::parse_error); - CHECK_THROWS_AS(json::from_msgpack(std::vector({0xce, 0x00, 0x00, 0x00})), json::parse_error); - CHECK_THROWS_AS(json::from_msgpack(std::vector({0xcf})), json::parse_error); - CHECK_THROWS_AS(json::from_msgpack(std::vector({0xcf, 0x00})), json::parse_error); - CHECK_THROWS_AS(json::from_msgpack(std::vector({0xcf, 0x00, 0x00})), json::parse_error); - CHECK_THROWS_AS(json::from_msgpack(std::vector({0xcf, 0x00, 0x00, 0x00})), json::parse_error); - CHECK_THROWS_AS(json::from_msgpack(std::vector({0xcf, 0x00, 0x00, 0x00, 0x00})), json::parse_error); - CHECK_THROWS_AS(json::from_msgpack(std::vector({0xcf, 0x00, 0x00, 0x00, 0x00, 0x00})), json::parse_error); - CHECK_THROWS_AS(json::from_msgpack(std::vector({0xcf, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00})), json::parse_error); - CHECK_THROWS_AS(json::from_msgpack(std::vector({0xcf, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00})), json::parse_error); + CHECK_THROWS_AS(json::from_msgpack(std::vector({0xcc})), json::parse_error&); + CHECK_THROWS_AS(json::from_msgpack(std::vector({0xcd})), json::parse_error&); + CHECK_THROWS_AS(json::from_msgpack(std::vector({0xcd, 0x00})), json::parse_error&); + CHECK_THROWS_AS(json::from_msgpack(std::vector({0xce})), json::parse_error&); + CHECK_THROWS_AS(json::from_msgpack(std::vector({0xce, 0x00})), json::parse_error&); + CHECK_THROWS_AS(json::from_msgpack(std::vector({0xce, 0x00, 0x00})), json::parse_error&); + CHECK_THROWS_AS(json::from_msgpack(std::vector({0xce, 0x00, 0x00, 0x00})), json::parse_error&); + CHECK_THROWS_AS(json::from_msgpack(std::vector({0xcf})), json::parse_error&); + CHECK_THROWS_AS(json::from_msgpack(std::vector({0xcf, 0x00})), json::parse_error&); + CHECK_THROWS_AS(json::from_msgpack(std::vector({0xcf, 0x00, 0x00})), json::parse_error&); + CHECK_THROWS_AS(json::from_msgpack(std::vector({0xcf, 0x00, 0x00, 0x00})), json::parse_error&); + CHECK_THROWS_AS(json::from_msgpack(std::vector({0xcf, 0x00, 0x00, 0x00, 0x00})), json::parse_error&); + CHECK_THROWS_AS(json::from_msgpack(std::vector({0xcf, 0x00, 0x00, 0x00, 0x00, 0x00})), json::parse_error&); + CHECK_THROWS_AS(json::from_msgpack(std::vector({0xcf, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00})), json::parse_error&); + CHECK_THROWS_AS(json::from_msgpack(std::vector({0xcf, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00})), json::parse_error&); CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xcc})), "[json.exception.parse_error.110] parse error at 2: unexpected end of input"); @@ -1068,10 +1068,10 @@ TEST_CASE("MessagePack") { SECTION("concrete examples") { - CHECK_THROWS_AS(json::from_msgpack(std::vector({0xc1})), json::parse_error); + CHECK_THROWS_AS(json::from_msgpack(std::vector({0xc1})), json::parse_error&); CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xc1})), "[json.exception.parse_error.112] parse error at 1: error reading MessagePack; last byte: 0xc1"); - CHECK_THROWS_AS(json::from_msgpack(std::vector({0xc6})), json::parse_error); + CHECK_THROWS_AS(json::from_msgpack(std::vector({0xc6})), json::parse_error&); CHECK_THROWS_WITH(json::from_msgpack(std::vector({0xc6})), "[json.exception.parse_error.112] parse error at 1: error reading MessagePack; last byte: 0xc6"); } @@ -1090,14 +1090,14 @@ TEST_CASE("MessagePack") 0xd4, 0xd5, 0xd6, 0xd7, 0xd8 }) { - CHECK_THROWS_AS(json::from_msgpack(std::vector({static_cast(byte)})), json::parse_error); + CHECK_THROWS_AS(json::from_msgpack(std::vector({static_cast(byte)})), json::parse_error&); } } } SECTION("invalid string in map") { - CHECK_THROWS_AS(json::from_msgpack(std::vector({0x81, 0xff, 0x01})), json::parse_error); + CHECK_THROWS_AS(json::from_msgpack(std::vector({0x81, 0xff, 0x01})), json::parse_error&); CHECK_THROWS_WITH(json::from_msgpack(std::vector({0x81, 0xff, 0x01})), "[json.exception.parse_error.113] parse error at 2: expected a MessagePack string; last byte: 0xff"); } diff --git a/test/src/unit-regression.cpp b/test/src/unit-regression.cpp index d4017c7a..e91163c7 100644 --- a/test/src/unit-regression.cpp +++ b/test/src/unit-regression.cpp @@ -572,7 +572,7 @@ TEST_CASE("regression tests") SECTION("issue #329 - serialized value not always can be parsed") { - CHECK_THROWS_AS(json::parse("22e2222"), json::out_of_range); + CHECK_THROWS_AS(json::parse("22e2222"), json::out_of_range&); CHECK_THROWS_WITH(json::parse("22e2222"), "[json.exception.out_of_range.406] number overflow parsing '22e2222'"); } @@ -580,7 +580,7 @@ TEST_CASE("regression tests") SECTION("issue #366 - json::parse on failed stream gets stuck") { std::ifstream f("file_not_found.json"); - CHECK_THROWS_AS(json::parse(f), json::parse_error); + CHECK_THROWS_AS(json::parse(f), json::parse_error&); CHECK_THROWS_WITH(json::parse(f), "[json.exception.parse_error.111] parse error: bad input stream"); } @@ -595,7 +595,7 @@ TEST_CASE("regression tests") // ss is not at EOF; this yielded an error before the fix // (threw basic_string::append). No, it should just throw // a parse error because of the EOF. - CHECK_THROWS_AS(ss >> j, json::parse_error); + CHECK_THROWS_AS(ss >> j, json::parse_error&); CHECK_THROWS_WITH(ss >> j, "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected end of input; expected '[', '{', or a literal"); } @@ -606,7 +606,7 @@ TEST_CASE("regression tests") { std::stringstream ss; json j; - CHECK_THROWS_AS(ss >> j, json::parse_error); + CHECK_THROWS_AS(ss >> j, json::parse_error&); CHECK_THROWS_WITH(ss >> j, "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected end of input; expected '[', '{', or a literal"); } @@ -616,7 +616,7 @@ TEST_CASE("regression tests") std::stringstream ss; ss << " "; json j; - CHECK_THROWS_AS(ss >> j, json::parse_error); + CHECK_THROWS_AS(ss >> j, json::parse_error&); CHECK_THROWS_WITH(ss >> j, "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected end of input; expected '[', '{', or a literal"); } @@ -629,7 +629,7 @@ TEST_CASE("regression tests") CHECK_NOTHROW(ss >> j); CHECK(j == 111); - CHECK_THROWS_AS(ss >> j, json::parse_error); + CHECK_THROWS_AS(ss >> j, json::parse_error&); CHECK_THROWS_WITH(ss >> j, "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected end of input; expected '[', '{', or a literal"); } @@ -642,7 +642,7 @@ TEST_CASE("regression tests") CHECK_NOTHROW(ss >> j); CHECK(j == 222); - CHECK_THROWS_AS(ss >> j, json::parse_error); + CHECK_THROWS_AS(ss >> j, json::parse_error&); CHECK_THROWS_WITH(ss >> j, "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected end of input; expected '[', '{', or a literal"); } @@ -655,7 +655,7 @@ TEST_CASE("regression tests") CHECK_NOTHROW(ss >> j); CHECK(j == 333); - CHECK_THROWS_AS(ss >> j, json::parse_error); + CHECK_THROWS_AS(ss >> j, json::parse_error&); CHECK_THROWS_WITH(ss >> j, "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected end of input; expected '[', '{', or a literal"); } @@ -672,7 +672,7 @@ TEST_CASE("regression tests") CHECK_NOTHROW(ss >> j); CHECK(j == 333); - CHECK_THROWS_AS(ss >> j, json::parse_error); + CHECK_THROWS_AS(ss >> j, json::parse_error&); CHECK_THROWS_WITH(ss >> j, "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected end of input; expected '[', '{', or a literal"); } @@ -691,7 +691,7 @@ TEST_CASE("regression tests") CHECK_NOTHROW(ss >> j); CHECK(j == ""); - CHECK_THROWS_AS(ss >> j, json::parse_error); + CHECK_THROWS_AS(ss >> j, json::parse_error&); CHECK_THROWS_WITH(ss >> j, "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected end of input; expected '[', '{', or a literal"); } @@ -706,7 +706,7 @@ TEST_CASE("regression tests") CHECK_NOTHROW(ss >> j); CHECK(j == json({{"three", 3}})); - CHECK_THROWS_AS(ss >> j, json::parse_error); + CHECK_THROWS_AS(ss >> j, json::parse_error&); CHECK_THROWS_WITH(ss >> j, "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected end of input; expected '[', '{', or a literal"); } @@ -776,7 +776,7 @@ TEST_CASE("regression tests") { // original test case std::vector vec {0x65, 0xf5, 0x0a, 0x48, 0x21}; - CHECK_THROWS_AS(json::from_cbor(vec), json::parse_error); + CHECK_THROWS_AS(json::from_cbor(vec), json::parse_error&); CHECK_THROWS_WITH(json::from_cbor(vec), "[json.exception.parse_error.110] parse error at 6: unexpected end of input"); } @@ -785,31 +785,31 @@ TEST_CASE("regression tests") { // original test case: incomplete float64 std::vector vec1 {0xcb, 0x8f, 0x0a}; - CHECK_THROWS_AS(json::from_msgpack(vec1), json::parse_error); + CHECK_THROWS_AS(json::from_msgpack(vec1), json::parse_error&); CHECK_THROWS_WITH(json::from_msgpack(vec1), "[json.exception.parse_error.110] parse error at 4: unexpected end of input"); // related test case: incomplete float32 std::vector vec2 {0xca, 0x8f, 0x0a}; - CHECK_THROWS_AS(json::from_msgpack(vec2), json::parse_error); + CHECK_THROWS_AS(json::from_msgpack(vec2), json::parse_error&); CHECK_THROWS_WITH(json::from_msgpack(vec2), "[json.exception.parse_error.110] parse error at 4: unexpected end of input"); // related test case: incomplete Half-Precision Float (CBOR) std::vector vec3 {0xf9, 0x8f}; - CHECK_THROWS_AS(json::from_cbor(vec3), json::parse_error); + CHECK_THROWS_AS(json::from_cbor(vec3), json::parse_error&); CHECK_THROWS_WITH(json::from_cbor(vec3), "[json.exception.parse_error.110] parse error at 3: unexpected end of input"); // related test case: incomplete Single-Precision Float (CBOR) std::vector vec4 {0xfa, 0x8f, 0x0a}; - CHECK_THROWS_AS(json::from_cbor(vec4), json::parse_error); + CHECK_THROWS_AS(json::from_cbor(vec4), json::parse_error&); CHECK_THROWS_WITH(json::from_cbor(vec4), "[json.exception.parse_error.110] parse error at 4: unexpected end of input"); // related test case: incomplete Double-Precision Float (CBOR) std::vector vec5 {0xfb, 0x8f, 0x0a}; - CHECK_THROWS_AS(json::from_cbor(vec5), json::parse_error); + CHECK_THROWS_AS(json::from_cbor(vec5), json::parse_error&); CHECK_THROWS_WITH(json::from_cbor(vec5), "[json.exception.parse_error.110] parse error at 4: unexpected end of input"); } @@ -818,7 +818,7 @@ TEST_CASE("regression tests") { // original test case std::vector vec1 {0x87}; - CHECK_THROWS_AS(json::from_msgpack(vec1), json::parse_error); + CHECK_THROWS_AS(json::from_msgpack(vec1), json::parse_error&); CHECK_THROWS_WITH(json::from_msgpack(vec1), "[json.exception.parse_error.110] parse error at 2: unexpected end of input"); @@ -832,7 +832,7 @@ TEST_CASE("regression tests") }) { std::vector vec(1, static_cast(b)); - CHECK_THROWS_AS(json::from_msgpack(vec), json::parse_error); + CHECK_THROWS_AS(json::from_msgpack(vec), json::parse_error&); } // more test cases for CBOR @@ -847,15 +847,15 @@ TEST_CASE("regression tests") }) { std::vector vec(1, static_cast(b)); - CHECK_THROWS_AS(json::from_cbor(vec), json::parse_error); + CHECK_THROWS_AS(json::from_cbor(vec), json::parse_error&); } // special case: empty input std::vector vec2; - CHECK_THROWS_AS(json::from_cbor(vec2), json::parse_error); + CHECK_THROWS_AS(json::from_cbor(vec2), json::parse_error&); CHECK_THROWS_WITH(json::from_cbor(vec2), "[json.exception.parse_error.110] parse error at 1: unexpected end of input"); - CHECK_THROWS_AS(json::from_msgpack(vec2), json::parse_error); + CHECK_THROWS_AS(json::from_msgpack(vec2), json::parse_error&); CHECK_THROWS_WITH(json::from_msgpack(vec2), "[json.exception.parse_error.110] parse error at 1: unexpected end of input"); } @@ -864,19 +864,19 @@ TEST_CASE("regression tests") { // original test case: empty UTF-8 string (indefinite length) std::vector vec1 {0x7f}; - CHECK_THROWS_AS(json::from_cbor(vec1), json::parse_error); + CHECK_THROWS_AS(json::from_cbor(vec1), json::parse_error&); CHECK_THROWS_WITH(json::from_cbor(vec1), "[json.exception.parse_error.110] parse error at 2: unexpected end of input"); // related test case: empty array (indefinite length) std::vector vec2 {0x9f}; - CHECK_THROWS_AS(json::from_cbor(vec2), json::parse_error); + CHECK_THROWS_AS(json::from_cbor(vec2), json::parse_error&); CHECK_THROWS_WITH(json::from_cbor(vec2), "[json.exception.parse_error.110] parse error at 2: unexpected end of input"); // related test case: empty map (indefinite length) std::vector vec3 {0xbf}; - CHECK_THROWS_AS(json::from_cbor(vec3), json::parse_error); + CHECK_THROWS_AS(json::from_cbor(vec3), json::parse_error&); CHECK_THROWS_WITH(json::from_cbor(vec3), "[json.exception.parse_error.110] parse error at 2: unexpected end of input"); } @@ -904,25 +904,25 @@ TEST_CASE("regression tests") 0x60, 0x60, 0x60, 0x60, 0x60, 0x60, 0x60, 0x60, 0x60, 0x60, 0x60, 0x60, 0x60, 0x60, 0x60, 0x60 }; - CHECK_THROWS_AS(json::from_cbor(vec), json::parse_error); + CHECK_THROWS_AS(json::from_cbor(vec), json::parse_error&); CHECK_THROWS_WITH(json::from_cbor(vec), "[json.exception.parse_error.113] parse error at 2: expected a CBOR string; last byte: 0x98"); // related test case: nonempty UTF-8 string (indefinite length) std::vector vec1 {0x7f, 0x61, 0x61}; - CHECK_THROWS_AS(json::from_cbor(vec1), json::parse_error); + CHECK_THROWS_AS(json::from_cbor(vec1), json::parse_error&); CHECK_THROWS_WITH(json::from_cbor(vec1), "[json.exception.parse_error.110] parse error at 4: unexpected end of input"); // related test case: nonempty array (indefinite length) std::vector vec2 {0x9f, 0x01}; - CHECK_THROWS_AS(json::from_cbor(vec2), json::parse_error); + CHECK_THROWS_AS(json::from_cbor(vec2), json::parse_error&); CHECK_THROWS_WITH(json::from_cbor(vec2), "[json.exception.parse_error.110] parse error at 3: unexpected end of input"); // related test case: nonempty map (indefinite length) std::vector vec3 {0xbf, 0x61, 0x61, 0x01}; - CHECK_THROWS_AS(json::from_cbor(vec3), json::parse_error); + CHECK_THROWS_AS(json::from_cbor(vec3), json::parse_error&); CHECK_THROWS_WITH(json::from_cbor(vec3), "[json.exception.parse_error.110] parse error at 5: unexpected end of input"); } @@ -957,7 +957,7 @@ TEST_CASE("regression tests") 0x96, 0x96, 0xb4, 0xb4, 0xfa, 0x94, 0x94, 0x61, 0x61, 0x61, 0x61, 0x61, 0x61, 0x61, 0x61, 0xfa }; - CHECK_THROWS_AS(json::from_cbor(vec1), json::parse_error); + CHECK_THROWS_AS(json::from_cbor(vec1), json::parse_error&); CHECK_THROWS_WITH(json::from_cbor(vec1), "[json.exception.parse_error.113] parse error at 13: expected a CBOR string; last byte: 0xb4"); @@ -971,7 +971,7 @@ TEST_CASE("regression tests") 0x96, 0x96, 0xb4, 0xb4, 0xfa, 0x94, 0x94, 0x61, 0x61, 0x61, 0x61, 0x61, 0x61, 0x61, 0x61, 0xfb }; - CHECK_THROWS_AS(json::from_cbor(vec2), json::parse_error); + CHECK_THROWS_AS(json::from_cbor(vec2), json::parse_error&); CHECK_THROWS_WITH(json::from_cbor(vec2), "[json.exception.parse_error.113] parse error at 13: expected a CBOR string; last byte: 0xb4"); } @@ -979,7 +979,7 @@ TEST_CASE("regression tests") SECTION("issue #452 - Heap-buffer-overflow (OSS-Fuzz issue 585)") { std::vector vec = {'-', '0', '1', '2', '2', '7', '4'}; - CHECK_THROWS_AS(json::parse(vec), json::parse_error); + CHECK_THROWS_AS(json::parse(vec), json::parse_error&); } SECTION("issue #454 - doubles are printed as integers") @@ -1019,9 +1019,9 @@ TEST_CASE("regression tests") }; CHECK_NOTHROW(create(j_array)); - CHECK_THROWS_AS(create(j_number), json::type_error); + CHECK_THROWS_AS(create(j_number), json::type_error&); CHECK_THROWS_WITH(create(j_number), "[json.exception.type_error.302] type must be array, but is number"); - CHECK_THROWS_AS(create(j_null), json::type_error); + CHECK_THROWS_AS(create(j_null), json::type_error&); CHECK_THROWS_WITH(create(j_null), "[json.exception.type_error.302] type must be array, but is null"); } @@ -1033,9 +1033,9 @@ TEST_CASE("regression tests") }; CHECK_NOTHROW(create(j_array)); - CHECK_THROWS_AS(create(j_number), json::type_error); + CHECK_THROWS_AS(create(j_number), json::type_error&); CHECK_THROWS_WITH(create(j_number), "[json.exception.type_error.302] type must be array, but is number"); - CHECK_THROWS_AS(create(j_null), json::type_error); + CHECK_THROWS_AS(create(j_null), json::type_error&); CHECK_THROWS_WITH(create(j_null), "[json.exception.type_error.302] type must be array, but is null"); } @@ -1047,9 +1047,9 @@ TEST_CASE("regression tests") }; CHECK_NOTHROW(create(j_array)); - CHECK_THROWS_AS(create(j_number), json::type_error); + CHECK_THROWS_AS(create(j_number), json::type_error&); CHECK_THROWS_WITH(create(j_number), "[json.exception.type_error.302] type must be array, but is number"); - CHECK_THROWS_AS(create(j_null), json::type_error); + CHECK_THROWS_AS(create(j_null), json::type_error&); CHECK_THROWS_WITH(create(j_null), "[json.exception.type_error.302] type must be array, but is null"); } } @@ -1084,7 +1084,7 @@ TEST_CASE("regression tests") l.m_stream->setstate(std::ios_base::failbit); - CHECK_THROWS_AS(l.fill_line_buffer(), json::parse_error); + CHECK_THROWS_AS(l.fill_line_buffer(), json::parse_error&); CHECK_THROWS_WITH(l.fill_line_buffer(), "[json.exception.parse_error.111] parse error: bad input stream"); } @@ -1099,7 +1099,7 @@ TEST_CASE("regression tests") l.m_stream->setstate(std::ios_base::badbit); - CHECK_THROWS_AS(l.fill_line_buffer(), json::parse_error); + CHECK_THROWS_AS(l.fill_line_buffer(), json::parse_error&); CHECK_THROWS_WITH(l.fill_line_buffer(), "[json.exception.parse_error.111] parse error: bad input stream"); } } @@ -1163,7 +1163,7 @@ TEST_CASE("regression tests") SECTION("issue #575 - heap-buffer-overflow (OSS-Fuzz 1400)") { std::vector vec = {'"', '\\', '"', 'X', '"', '"'}; - CHECK_THROWS_AS(json::parse(vec), json::parse_error); + CHECK_THROWS_AS(json::parse(vec), json::parse_error&); } SECTION("issue #600 - how does one convert a map in Json back to std::map?") diff --git a/test/src/unit-testsuites.cpp b/test/src/unit-testsuites.cpp index d3dbaf4b..d281c679 100644 --- a/test/src/unit-testsuites.cpp +++ b/test/src/unit-testsuites.cpp @@ -78,7 +78,7 @@ TEST_CASE("compliance tests from json.org") { CAPTURE(filename); std::ifstream f(filename); - CHECK_THROWS_AS(json::parse(f), json::parse_error); + CHECK_THROWS_AS(json::parse(f), json::parse_error&); } } @@ -772,7 +772,7 @@ TEST_CASE("nst's JSONTestSuite") { CAPTURE(filename); std::ifstream f(filename); - CHECK_THROWS_AS(json::parse(f), json::parse_error); + CHECK_THROWS_AS(json::parse(f), json::parse_error&); } } @@ -848,7 +848,7 @@ TEST_CASE("nst's JSONTestSuite") CAPTURE(filename); std::ifstream f(filename); json j; - CHECK_THROWS_AS(f >> j, json::out_of_range); + CHECK_THROWS_AS(f >> j, json::out_of_range&); } } @@ -875,7 +875,7 @@ TEST_CASE("nst's JSONTestSuite") CAPTURE(filename); std::ifstream f(filename); json j; - CHECK_THROWS_AS(f >> j, json::parse_error); + CHECK_THROWS_AS(f >> j, json::parse_error&); } } } diff --git a/test/src/unit-unicode.cpp b/test/src/unit-unicode.cpp index 70be088e..f2172a9d 100644 --- a/test/src/unit-unicode.cpp +++ b/test/src/unit-unicode.cpp @@ -79,7 +79,7 @@ void check_utf8string(bool success_expected, int byte1, int byte2 = -1, int byte } else { - CHECK_THROWS_AS(json::parse(json_string), json::parse_error); + CHECK_THROWS_AS(json::parse(json_string), json::parse_error&); } } @@ -933,7 +933,7 @@ TEST_CASE("Unicode", "[hide]") { std::string json_text = "\"" + codepoint_to_unicode(cp) + "\""; CAPTURE(json_text); - CHECK_THROWS_AS(json::parse(json_text), json::parse_error); + CHECK_THROWS_AS(json::parse(json_text), json::parse_error&); } } @@ -952,7 +952,7 @@ TEST_CASE("Unicode", "[hide]") std::string json_text = "\"" + codepoint_to_unicode(cp1) + codepoint_to_unicode(cp2) + "\""; CAPTURE(json_text); - CHECK_THROWS_AS(json::parse(json_text), json::parse_error); + CHECK_THROWS_AS(json::parse(json_text), json::parse_error&); } } } @@ -965,7 +965,7 @@ TEST_CASE("Unicode", "[hide]") { std::string json_text = "\"" + codepoint_to_unicode(cp) + "\""; CAPTURE(json_text); - CHECK_THROWS_AS(json::parse(json_text), json::parse_error); + CHECK_THROWS_AS(json::parse(json_text), json::parse_error&); } } @@ -1036,7 +1036,7 @@ TEST_CASE("Unicode", "[hide]") SECTION("error for incomplete/wrong BOM") { - CHECK_THROWS_AS(json::parse("\xef\xbb"), json::parse_error); - CHECK_THROWS_AS(json::parse("\xef\xbb\xbb"), json::parse_error); + CHECK_THROWS_AS(json::parse("\xef\xbb"), json::parse_error&); + CHECK_THROWS_AS(json::parse("\xef\xbb\xbb"), json::parse_error&); } } From 07b78c993c1b5a8b360ce6db11f6f79a048e8a9a Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Fri, 7 Jul 2017 23:37:16 +0200 Subject: [PATCH 265/530] :bug: fixed undefined behavior bug When an empty vector was passed to the parse function, an empty iterator range was used to construct an input iterator. Unfortunately, we then cannot use the start iterator to derive a pointer from. Found with Xcode's undefined behavior sanitizer. --- src/json.hpp | 13 +++++++++++-- 1 file changed, 11 insertions(+), 2 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 918a646b..0a588631 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -8886,8 +8886,17 @@ class basic_json static_assert(sizeof(typename std::iterator_traits::value_type) == 1, "each element in the iterator range must have the size of 1 byte"); - return create(reinterpret_cast(&(*first)), - static_cast(std::distance(first, last))); + const auto len = static_cast(std::distance(first, last)); + if (JSON_LIKELY(len > 0)) + { + // there is at least one element: use the address of first + return create(reinterpret_cast(&(*first)), len); + } + else + { + // the address of first cannot be used - use nullptr + return create(nullptr, len); + } } /// input adapter for array From 63ecbfd36be3087d24bca385370413869900ff35 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Fri, 7 Jul 2017 23:38:04 +0200 Subject: [PATCH 266/530] :white_check_mark: re-added test I have no idea why this test has been removed in the first place. --- test/src/unit-pointer_access.cpp | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/test/src/unit-pointer_access.cpp b/test/src/unit-pointer_access.cpp index 5f70780d..f3830e5d 100644 --- a/test/src/unit-pointer_access.cpp +++ b/test/src/unit-pointer_access.cpp @@ -257,11 +257,11 @@ TEST_CASE("pointer access") const test_type* p2 = value.get_ptr(); CHECK(p1 == value.get_ptr()); - //CHECK(*p2 == value.get()); + CHECK(*p2 == value.get()); const test_type* const p3 = value.get_ptr(); CHECK(p1 == value.get_ptr()); - //CHECK(*p3 == value.get()); + CHECK(*p3 == value.get()); // check if null pointers are returned correctly CHECK(value.get_ptr() == nullptr); From df2feabb72a433edbadf0227378cd8fcebc818d9 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sat, 8 Jul 2017 11:57:09 +0200 Subject: [PATCH 267/530] :construction_worker: added Xcode 9 beta See https://twitter.com/travisci/status/880550780368855040 --- .travis.yml | 3 +++ 1 file changed, 3 insertions(+) diff --git a/.travis.yml b/.travis.yml index 8f19baa2..4ad1a555 100644 --- a/.travis.yml +++ b/.travis.yml @@ -152,6 +152,9 @@ matrix: - os: osx osx_image: xcode8.3 + - os: osx + osx_image: xcode9 + # Linux / GCC - os: linux From 015c88783e9ca0479a15d246104c22abf918bc27 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sat, 8 Jul 2017 12:03:12 +0200 Subject: [PATCH 268/530] :hammer: fixed a pedantic warning --- .gitignore | 1 + test/src/unit-unicode.cpp | 1 + 2 files changed, 2 insertions(+) diff --git a/.gitignore b/.gitignore index 58ced04e..ef8d050e 100644 --- a/.gitignore +++ b/.gitignore @@ -7,6 +7,7 @@ fuzz-testing *.gcno *.gcda +ndk build working diff --git a/test/src/unit-unicode.cpp b/test/src/unit-unicode.cpp index f2172a9d..9b379d96 100644 --- a/test/src/unit-unicode.cpp +++ b/test/src/unit-unicode.cpp @@ -34,6 +34,7 @@ using nlohmann::json; #include +extern size_t calls; size_t calls = 0; void check_utf8string(bool success_expected, int byte1, int byte2, int byte3, int byte4); From ac423337494cae79291e32cc32d0b56df2346c3c Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sat, 8 Jul 2017 12:11:45 +0200 Subject: [PATCH 269/530] :arrow_up: Catch v1.9.6 --- test/thirdparty/catch/catch.hpp | 110 ++++++++++++++++++++------------ 1 file changed, 70 insertions(+), 40 deletions(-) diff --git a/test/thirdparty/catch/catch.hpp b/test/thirdparty/catch/catch.hpp index 2c93e370..f7681f49 100644 --- a/test/thirdparty/catch/catch.hpp +++ b/test/thirdparty/catch/catch.hpp @@ -1,6 +1,6 @@ /* - * Catch v1.9.5 - * Generated: 2017-06-15 12:03:23.301505 + * Catch v1.9.6 + * Generated: 2017-06-27 12:19:54.557875 * ---------------------------------------------------------- * This file has been merged from multiple headers. Please don't edit it directly * Copyright (c) 2012 Two Blue Cubes Ltd. All rights reserved. @@ -934,15 +934,17 @@ namespace Catch { struct AssertionInfo { AssertionInfo() {} - AssertionInfo( std::string const& _macroName, + AssertionInfo( char const * _macroName, SourceLineInfo const& _lineInfo, - std::string const& _capturedExpression, - ResultDisposition::Flags _resultDisposition ); + char const * _capturedExpression, + ResultDisposition::Flags _resultDisposition, + char const * _secondArg = ""); - std::string macroName; + char const * macroName; SourceLineInfo lineInfo; - std::string capturedExpression; + char const * capturedExpression; ResultDisposition::Flags resultDisposition; + char const * secondArg; }; struct AssertionResultData @@ -1217,7 +1219,7 @@ namespace Catch { template ResultBuilder& operator << ( T const& value ) { - m_stream.oss << value; + m_stream().oss << value; return *this; } @@ -1250,7 +1252,12 @@ namespace Catch { private: AssertionInfo m_assertionInfo; AssertionResultData m_data; - CopyableStream m_stream; + + static CopyableStream &m_stream() + { + static CopyableStream s; + return s; + } bool m_shouldDebugBreak; bool m_shouldThrow; @@ -3938,6 +3945,7 @@ namespace Catch { listTags( false ), listReporters( false ), listTestNamesOnly( false ), + listExtraInfo( false ), showSuccessfulTests( false ), shouldDebugBreak( false ), noThrow( false ), @@ -3957,6 +3965,7 @@ namespace Catch { bool listTags; bool listReporters; bool listTestNamesOnly; + bool listExtraInfo; bool showSuccessfulTests; bool shouldDebugBreak; @@ -4015,6 +4024,7 @@ namespace Catch { bool listTestNamesOnly() const { return m_data.listTestNamesOnly; } bool listTags() const { return m_data.listTags; } bool listReporters() const { return m_data.listReporters; } + bool listExtraInfo() const { return m_data.listExtraInfo; } std::string getProcessName() const { return m_data.processName; } @@ -5276,6 +5286,10 @@ namespace Catch { .describe( "list all/matching test cases names only" ) .bind( &ConfigData::listTestNamesOnly ); + cli["--list-extra-info"] + .describe( "list all/matching test cases with more info" ) + .bind( &ConfigData::listExtraInfo ); + cli["--list-reporters"] .describe( "list all reporters" ) .bind( &ConfigData::listReporters ); @@ -5804,8 +5818,9 @@ namespace Catch { } std::size_t matchedTests = 0; - TextAttributes nameAttr, tagsAttr; + TextAttributes nameAttr, descAttr, tagsAttr; nameAttr.setInitialIndent( 2 ).setIndent( 4 ); + descAttr.setIndent( 4 ); tagsAttr.setIndent( 6 ); std::vector matchedTestCases = filterTests( getAllTestCasesSorted( config ), testSpec, config ); @@ -5820,6 +5835,13 @@ namespace Catch { Colour colourGuard( colour ); Catch::cout() << Text( testCaseInfo.name, nameAttr ) << std::endl; + if( config.listExtraInfo() ) { + Catch::cout() << " " << testCaseInfo.lineInfo << std::endl; + std::string description = testCaseInfo.description; + if( description.empty() ) + description = "(NO DESCRIPTION)"; + Catch::cout() << Text( description, descAttr ) << std::endl; + } if( !testCaseInfo.tags.empty() ) Catch::cout() << Text( testCaseInfo.tagsAsString, tagsAttr ) << std::endl; } @@ -5843,9 +5865,12 @@ namespace Catch { matchedTests++; TestCaseInfo const& testCaseInfo = it->getTestCaseInfo(); if( startsWith( testCaseInfo.name, '#' ) ) - Catch::cout() << '"' << testCaseInfo.name << '"' << std::endl; + Catch::cout() << '"' << testCaseInfo.name << '"'; else - Catch::cout() << testCaseInfo.name << std::endl; + Catch::cout() << testCaseInfo.name; + if ( config.listExtraInfo() ) + Catch::cout() << "\t@" << testCaseInfo.lineInfo; + Catch::cout() << std::endl; } return matchedTests; } @@ -5937,7 +5962,7 @@ namespace Catch { inline Option list( Config const& config ) { Option listedCount; - if( config.listTests() ) + if( config.listTests() || ( config.listExtraInfo() && !config.listTestNamesOnly() ) ) listedCount = listedCount.valueOr(0) + listTests( config ); if( config.listTestNamesOnly() ) listedCount = listedCount.valueOr(0) + listTestsNamesOnly( config ); @@ -6647,7 +6672,7 @@ namespace Catch { static_cast(m_reporter->assertionEnded(AssertionStats(result, m_messages, m_totals))); // Reset working state - m_lastAssertionInfo = AssertionInfo( std::string(), m_lastAssertionInfo.lineInfo, "{Unknown expression after the reported line}" , m_lastAssertionInfo.resultDisposition ); + m_lastAssertionInfo = AssertionInfo( "", m_lastAssertionInfo.lineInfo, "{Unknown expression after the reported line}" , m_lastAssertionInfo.resultDisposition ); m_lastResult = result; } @@ -6777,7 +6802,7 @@ namespace Catch { double duration = 0; m_shouldReportUnexpected = true; try { - m_lastAssertionInfo = AssertionInfo( "TEST_CASE", testCaseInfo.lineInfo, std::string(), ResultDisposition::Normal ); + m_lastAssertionInfo = AssertionInfo( "TEST_CASE", testCaseInfo.lineInfo, "", ResultDisposition::Normal ); seedRng( *m_config ); @@ -6829,9 +6854,9 @@ namespace Catch { private: ResultBuilder makeUnexpectedResultBuilder() const { - return ResultBuilder( m_lastAssertionInfo.macroName.c_str(), + return ResultBuilder( m_lastAssertionInfo.macroName, m_lastAssertionInfo.lineInfo, - m_lastAssertionInfo.capturedExpression.c_str(), + m_lastAssertionInfo.capturedExpression, m_lastAssertionInfo.resultDisposition ); } @@ -8008,14 +8033,16 @@ namespace Catch { namespace Catch { - AssertionInfo::AssertionInfo( std::string const& _macroName, + AssertionInfo::AssertionInfo( char const * _macroName, SourceLineInfo const& _lineInfo, - std::string const& _capturedExpression, - ResultDisposition::Flags _resultDisposition ) + char const * _capturedExpression, + ResultDisposition::Flags _resultDisposition, + char const * _secondArg) : macroName( _macroName ), lineInfo( _lineInfo ), capturedExpression( _capturedExpression ), - resultDisposition( _resultDisposition ) + resultDisposition( _resultDisposition ), + secondArg( _secondArg ) {} AssertionResult::AssertionResult() {} @@ -8042,24 +8069,30 @@ namespace Catch { } bool AssertionResult::hasExpression() const { - return !m_info.capturedExpression.empty(); + return m_info.capturedExpression[0] != 0; } bool AssertionResult::hasMessage() const { return !m_resultData.message.empty(); } + std::string capturedExpressionWithSecondArgument( char const * capturedExpression, char const * secondArg ) { + return (secondArg[0] == 0 || secondArg[0] == '"' && secondArg[1] == '"') + ? capturedExpression + : std::string(capturedExpression) + ", " + secondArg; + } + std::string AssertionResult::getExpression() const { if( isFalseTest( m_info.resultDisposition ) ) - return '!' + m_info.capturedExpression; + return '!' + capturedExpressionWithSecondArgument(m_info.capturedExpression, m_info.secondArg); else - return m_info.capturedExpression; + return capturedExpressionWithSecondArgument(m_info.capturedExpression, m_info.secondArg); } std::string AssertionResult::getExpressionInMacro() const { - if( m_info.macroName.empty() ) - return m_info.capturedExpression; + if( m_info.macroName[0] == 0 ) + return capturedExpressionWithSecondArgument(m_info.capturedExpression, m_info.secondArg); else - return m_info.macroName + "( " + m_info.capturedExpression + " )"; + return std::string(m_info.macroName) + "( " + capturedExpressionWithSecondArgument(m_info.capturedExpression, m_info.secondArg) + " )"; } bool AssertionResult::hasExpandedExpression() const { @@ -8309,7 +8342,7 @@ namespace Catch { } inline Version libraryVersion() { - static Version version( 1, 9, 5, "", 0 ); + static Version version( 1, 9, 6, "", 0 ); return version; } @@ -9009,26 +9042,23 @@ std::string toString( std::nullptr_t ) { namespace Catch { - std::string capturedExpressionWithSecondArgument( std::string const& capturedExpression, std::string const& secondArg ) { - return secondArg.empty() || secondArg == "\"\"" - ? capturedExpression - : capturedExpression + ", " + secondArg; - } ResultBuilder::ResultBuilder( char const* macroName, SourceLineInfo const& lineInfo, char const* capturedExpression, ResultDisposition::Flags resultDisposition, char const* secondArg ) - : m_assertionInfo( macroName, lineInfo, capturedExpressionWithSecondArgument( capturedExpression, secondArg ), resultDisposition ), + : m_assertionInfo( macroName, lineInfo, capturedExpression, resultDisposition, secondArg ), m_shouldDebugBreak( false ), m_shouldThrow( false ), m_guardException( false ) - {} + { + m_stream().oss.str(""); + } ResultBuilder::~ResultBuilder() { #if defined(CATCH_CONFIG_FAST_COMPILE) if ( m_guardException ) { - m_stream.oss << "Exception translation was disabled by CATCH_CONFIG_FAST_COMPILE"; + m_stream().oss << "Exception translation was disabled by CATCH_CONFIG_FAST_COMPILE"; captureResult( ResultWas::ThrewException ); getCurrentContext().getResultCapture()->exceptionEarlyReported(); } @@ -9051,7 +9081,7 @@ namespace Catch { void ResultBuilder::useActiveException( ResultDisposition::Flags resultDisposition ) { m_assertionInfo.resultDisposition = resultDisposition; - m_stream.oss << Catch::translateActiveException(); + m_stream().oss << Catch::translateActiveException(); captureResult( ResultWas::ThrewException ); } @@ -9072,7 +9102,7 @@ namespace Catch { assert( !isFalseTest( m_assertionInfo.resultDisposition ) ); AssertionResultData data = m_data; data.resultType = ResultWas::Ok; - data.reconstructedExpression = m_assertionInfo.capturedExpression; + data.reconstructedExpression = capturedExpressionWithSecondArgument(m_assertionInfo.capturedExpression, m_assertionInfo.secondArg); std::string actualMessage = Catch::translateActiveException(); if( !matcher.match( actualMessage ) ) { @@ -9138,13 +9168,13 @@ namespace Catch { data.negate( expr.isBinaryExpression() ); } - data.message = m_stream.oss.str(); + data.message = m_stream().oss.str(); data.decomposedExpression = &expr; // for lazy reconstruction return AssertionResult( m_assertionInfo, data ); } void ResultBuilder::reconstructExpression( std::string& dest ) const { - dest = m_assertionInfo.capturedExpression; + dest = capturedExpressionWithSecondArgument(m_assertionInfo.capturedExpression, m_assertionInfo.secondArg); } void ResultBuilder::setExceptionGuard() { From 6f12749ef60309c7cc576512296a4561ab75441b Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sat, 8 Jul 2017 12:55:05 +0200 Subject: [PATCH 270/530] :memo: added version string for Xcode 9 --- README.md | 1 + 1 file changed, 1 insertion(+) diff --git a/README.md b/README.md index b3a51bdf..9aabc0fd 100644 --- a/README.md +++ b/README.md @@ -751,6 +751,7 @@ The following compilers are currently used in continuous integration at [Travis] | Clang Xcode 8.1 | Darwin Kernel Version 16.1.0 (macOS 10.12.1) | Apple LLVM version 8.0.0 (clang-800.0.42.1) | | Clang Xcode 8.2 | Darwin Kernel Version 16.1.0 (macOS 10.12.1) | Apple LLVM version 8.0.0 (clang-800.0.42.1) | | Clang Xcode 8.3 | Darwin Kernel Version 16.5.0 (macOS 10.12.4) | Apple LLVM version 8.1.0 (clang-802.0.38) | +| Clang Xcode 9 beta | Darwin Kernel Version 16.6.0 (macOS 10.12.5) | Apple LLVM version 9.0.0 (clang-900.0.26) | | Visual Studio 14 2015 | Windows Server 2012 R2 (x64) | Microsoft (R) Build Engine version 14.0.25420.1 | | Visual Studio 2017 | Windows Server 2016 | Microsoft (R) Build Engine version 15.1.1012.6693 | From 566f6dcf307225b870c96b8f23a1bdbfe9a62d58 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sat, 8 Jul 2017 13:33:13 +0200 Subject: [PATCH 271/530] :fire: removing usused functions #645 It seems these functions are not required any more. The code was added in commit 7e32457 to fix issue #171. There are still regression tests for #171, so when this commit passes the CI, the functions may be removed for good. --- src/json.hpp | 68 ---------------------------------------------------- 1 file changed, 68 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 0a588631..8f5b09a3 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -4240,74 +4240,6 @@ class basic_json with range checking @sa @ref value() for access by value with a default value - @since version 1.0.0 - */ - template - reference operator[](T * (&key)[n]) - { - return operator[](static_cast(key)); - } - - /*! - @brief read-only access specified object element - - Returns a const reference to the element at with specified key @a key. No - bounds checking is performed. - - @warning If the element with key @a key does not exist, the behavior is - undefined. - - @note This function is required for compatibility reasons with Clang. - - @param[in] key key of the element to access - - @return const reference to the element at key @a key - - @throw type_error.305 if the JSON value is not an object; in that cases, - using the [] operator with a key makes no sense. - - @complexity Logarithmic in the size of the container. - - @liveexample{The example below shows how object elements can be read using - the `[]` operator.,operatorarray__key_type_const} - - @sa @ref at(const typename object_t::key_type&) for access by reference - with range checking - @sa @ref value() for access by value with a default value - - @since version 1.0.0 - */ - template - const_reference operator[](T * (&key)[n]) const - { - return operator[](static_cast(key)); - } - - /*! - @brief access specified object element - - Returns a reference to the element at with specified key @a key. - - @note If @a key is not found in the object, then it is silently added to - the object and filled with a `null` value to make `key` a valid reference. - In case the value was `null` before, it is converted to an object. - - @param[in] key key of the element to access - - @return reference to the element at key @a key - - @throw type_error.305 if the JSON value is not an object or null; in that - cases, using the [] operator with a key makes no sense. - - @complexity Logarithmic in the size of the container. - - @liveexample{The example below shows how object elements can be read and - written using the `[]` operator.,operatorarray__key_type} - - @sa @ref at(const typename object_t::key_type&) for access by reference - with range checking - @sa @ref value() for access by value with a default value - @since version 1.1.0 */ template From 552622343e805127a9c9fb974c4134052da05673 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sat, 8 Jul 2017 16:18:27 +0200 Subject: [PATCH 272/530] :memo: fixed typos --- src/json.hpp | 22 +++++++++++----------- 1 file changed, 11 insertions(+), 11 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 0a588631..c562c827 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -210,7 +210,7 @@ class parse_error : public exception /*! @brief create a parse error exception @param[in] id the id of the exception - @param[in] byte_ the byte index where the error occured (or 0 if + @param[in] byte_ the byte index where the error occurred (or 0 if the position cannot be determined) @param[in] what_arg the explanatory string @return parse_error object @@ -260,7 +260,7 @@ json.exception.invalid_iterator.209 | cannot use offsets with object iterators | json.exception.invalid_iterator.210 | iterators do not fit | The iterator range passed to the insert function are not compatible, meaning they do not belong to the same container. Therefore, the range (@a first, @a last) is invalid. json.exception.invalid_iterator.211 | passed iterators may not belong to container | The iterator range passed to the insert function must not be a subrange of the container to insert to. json.exception.invalid_iterator.212 | cannot compare iterators of different containers | When two iterators are compared, they must belong to the same container. -json.exception.invalid_iterator.213 | cannot compare order of object iterators | The order of object iterators cannot be compated, because JSON objects are unordered. +json.exception.invalid_iterator.213 | cannot compare order of object iterators | The order of object iterators cannot be compared, because JSON objects are unordered. json.exception.invalid_iterator.214 | cannot get value | Cannot get value for iterator: Either the iterator belongs to a null value or it is an iterator to a primitive type (number, boolean, or string), but the iterator is different to @ref begin(). @since version 3.0.0 @@ -2969,8 +2969,8 @@ class basic_json members will be pretty-printed with that indent level. An indent level of `0` will only insert newlines. `-1` (the default) selects the most compact representation. - @param[in] indent_char The character to use for indentation of @a indent is - greate than `0`. The default is ` ` (space). + @param[in] indent_char The character to use for indentation if @a indent is + greater than `0`. The default is ` ` (space). @return string containing the serialization of the JSON value @@ -2981,7 +2981,7 @@ class basic_json @see https://docs.python.org/2/library/json.html#json.dump - @since version 1.0.0; indentaction character added in version 3.0.0 + @since version 1.0.0; indentation character added in version 3.0.0 */ string_t dump(const int indent = -1, const char indent_char = ' ') const { @@ -3460,7 +3460,7 @@ class basic_json /*! @brief helper function to implement get_ref() - This funcion helps to implement get_ref() without code duplication for + This function helps to implement get_ref() without code duplication for const and non-const overloads @tparam ThisType will be deduced as `basic_json` or `const basic_json` @@ -6744,7 +6744,7 @@ class basic_json std::vector& v; }; - /// putput adatpter for output streams + /// putput adapter for output streams template class output_stream_adapter : public output_adapter { @@ -8186,7 +8186,7 @@ class basic_json /*! @note The conventional copy constructor and copy assignment are implicitly defined. - Combined with the following converting constructor and assigment, + Combined with the following converting constructor and assignment, they support: copy from iterator to iterator, copy from const iterator to const iterator, and conversion from iterator to const iterator. @@ -10033,7 +10033,7 @@ class basic_json @return string @throw parse_error.110 if input ended - @throw parse_error.113 if an unexpexted byte is read + @throw parse_error.113 if an unexpected byte is read */ std::string get_cbor_string() { @@ -10124,7 +10124,7 @@ class basic_json @return string @throw parse_error.110 if input ended - @throw parse_error.113 if an unexpexted byte is read + @throw parse_error.113 if an unexpected byte is read */ std::string get_msgpack_string() { @@ -13969,7 +13969,7 @@ scan_number_done: @complexity Linear in the size the JSON value. @throw type_error.314 if value is not an object - @throw type_error.315 if object values are not primitve + @throw type_error.315 if object values are not primitive @liveexample{The following code shows how a flattened JSON object is unflattened into the original nested JSON object.,unflatten} From b3ac36db9309878cbff0d16ed070d5609e8b6f62 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sat, 8 Jul 2017 18:34:08 +0200 Subject: [PATCH 273/530] :hammer: excluded uncovered lines These breaks were just added to silence a GCC warning - the GCC is right about warning as it cannot know that the expect function will not return in these two scenarios. --- src/json.hpp | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 26676954..c312b3e8 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -12822,14 +12822,14 @@ scan_number_done: { // using "uninitialized" to avoid "expected" message expect(lexer::token_type::uninitialized); - break; + break; // LCOV_EXCL_LINE } default: { // the last token was unexpected; we expected a value expect(lexer::token_type::literal_or_value); - break; + break; // LCOV_EXCL_LINE } } From b38ecb5ca908a3151c0c53435fe9069787e62fc1 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sat, 8 Jul 2017 20:31:13 +0200 Subject: [PATCH 274/530] :hammer: simplified binary write Also added some comments and improved the documentation. --- src/json.hpp | 52 ++++++++++++++++++++++++++-------------------------- 1 file changed, 26 insertions(+), 26 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index c312b3e8..bcec446a 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -9897,22 +9897,23 @@ class basic_json /* @brief read a number from the input - @tparam T the type of the number + @tparam NumberType the type of the number - @return number of type @a T + @return number of type @a NumberType @note This function needs to respect the system's endianess, because bytes in CBOR and MessagePack are stored in network order (big endian) and therefore need reordering on little endian systems. - @throw parse_error.110 if input has less than `sizeof(T)` bytes + @throw parse_error.110 if input has less than `sizeof(NumberType)` + bytes */ - template - T get_number() + template + NumberType get_number() { // step 1: read input into array with system's byte order - std::array vec; - for (size_t i = 0; i < sizeof(T); ++i) + std::array vec; + for (size_t i = 0; i < sizeof(NumberType); ++i) { get(); check_eof(); @@ -9920,7 +9921,7 @@ class basic_json // reverse byte order prior to conversion if necessary if (is_little_endian) { - vec[sizeof(T) - i - 1] = static_cast(current); + vec[sizeof(NumberType) - i - 1] = static_cast(current); } else { @@ -9929,8 +9930,8 @@ class basic_json } // step 2: convert array into number of type T and return - T result; - std::memcpy(&result, vec.data(), sizeof(T)); + NumberType result; + std::memcpy(&result, vec.data(), sizeof(NumberType)); return result; } @@ -9939,6 +9940,10 @@ class basic_json @param[in] len number of bytes to read + @note We can not reserve @a len bytes for the result, because @a len + may be too large. Usually, @ref check_eof() detects the end of + the input before we run out of string memory. + @return string created by reading @a len bytes @throw parse_error.110 if input has less than @a len bytes @@ -10658,33 +10663,28 @@ class basic_json /* @brief write a number to output input - @param[in] n number of type @a T - @tparam T the type of the number + @param[in] n number of type @a NumberType + @tparam NumberType the type of the number @note This function needs to respect the system's endianess, because bytes in CBOR and MessagePack are stored in network order (big endian) and therefore need reordering on little endian systems. */ - template - void write_number(T n) + template + void write_number(NumberType n) { - // step 1: write number to array of length T - std::array vec; - std::memcpy(vec.data(), &n, sizeof(T)); + // step 1: write number to array of length NumberType + std::array vec; + std::memcpy(vec.data(), &n, sizeof(NumberType)); // step 2: write array to output (with possible reordering) - for (size_t i = 0; i < sizeof(T); ++i) + if (is_little_endian) { // reverse byte order prior to conversion if necessary - if (is_little_endian) - { - oa->write_character(vec[sizeof(T) - i - 1]); - } - else - { - oa->write_character(vec[i]); // LCOV_EXCL_LINE - } + std::reverse(vec.begin(), vec.end()); } + + oa->write_characters(vec.data(), sizeof(NumberType)); } private: From 87bccbc3351f11a8e07c3440e75f96027538e1b6 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sat, 8 Jul 2017 21:35:13 +0200 Subject: [PATCH 275/530] :hammer: removed exception parse_error.111 It makes no sense to have this special exception. Instead of throwing when an input adapter is created, it is better to detect a parse error in later usage when an EOF is "read" unexpectedly. --- src/json.hpp | 11 +---------- test/src/unit-regression.cpp | 2 +- 2 files changed, 2 insertions(+), 11 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index bcec446a..e07aaa8c 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -198,7 +198,6 @@ json.exception.parse_error.107 | parse error: JSON pointer must be empty or begi json.exception.parse_error.108 | parse error: escape character '~' must be followed with '0' or '1' | In a JSON Pointer, only `~0` and `~1` are valid escape sequences. json.exception.parse_error.109 | parse error: array index 'one' is not a number | A JSON Pointer array index must be a number. json.exception.parse_error.110 | parse error at 1: cannot read 2 bytes from vector | When parsing CBOR or MessagePack, the byte vector ends before the complete value has been read. -json.exception.parse_error.111 | parse error: bad input stream | Parsing CBOR or MessagePack from an input stream where the [`badbit` or `failbit`](http://en.cppreference.com/w/cpp/io/ios_base/iostate) is set. json.exception.parse_error.112 | parse error at 1: error reading CBOR; last byte: 0xf8 | Not all types of CBOR or MessagePack are supported. This exception occurs if an unsupported byte was read. json.exception.parse_error.113 | parse error at 2: expected a CBOR string; last byte: 0x98 | While parsing a map key, a value that is not a string has been read. @@ -7498,7 +7497,6 @@ class basic_json @throw parse_error.101 in case of an unexpected token @throw parse_error.102 if to_unicode fails or surrogate error @throw parse_error.103 if to_unicode fails - @throw parse_error.111 if input stream is in a bad state @complexity Linear in the length of the input. The parser is a predictive LL(1) parser. The complexity can be higher if the parser callback function @@ -7697,7 +7695,6 @@ class basic_json @throw parse_error.101 in case of an unexpected token @throw parse_error.102 if to_unicode fails or surrogate error @throw parse_error.103 if to_unicode fails - @throw parse_error.111 if input stream is in a bad state @complexity Linear in the length of the input. The parser is a predictive LL(1) parser. @@ -8864,12 +8861,6 @@ class basic_json cached_input_stream_adapter(std::istream& i) : is(i), start_position(is.tellg()) { - // immediately abort if stream is erroneous - if (JSON_UNLIKELY(i.fail())) - { - JSON_THROW(parse_error::create(111, 0, "bad input stream")); - } - fill_buffer(); // skip byte order mark @@ -9429,7 +9420,7 @@ class basic_json } else { - val = mant == 0 + val = (mant == 0) ? std::numeric_limits::infinity() : std::numeric_limits::quiet_NaN(); } diff --git a/test/src/unit-regression.cpp b/test/src/unit-regression.cpp index e91163c7..f59fef1d 100644 --- a/test/src/unit-regression.cpp +++ b/test/src/unit-regression.cpp @@ -581,7 +581,7 @@ TEST_CASE("regression tests") { std::ifstream f("file_not_found.json"); CHECK_THROWS_AS(json::parse(f), json::parse_error&); - CHECK_THROWS_WITH(json::parse(f), "[json.exception.parse_error.111] parse error: bad input stream"); + CHECK_THROWS_WITH(json::parse(f), "[json.exception.parse_error.101] parse error at 1: syntax error - unexpected end of input; expected '[', '{', or a literal"); } SECTION("issue #367 - calling stream at EOF") From f697fec5221e5b15ae2592cc5974009cbac1f841 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sat, 8 Jul 2017 22:16:45 +0200 Subject: [PATCH 276/530] :memo: fixed documentation There was an issue in the documentation example before. Unfortunately, I did not recreate the example outputs. --- doc/examples/operator__notequal.link | 2 +- doc/examples/operator__notequal.output | 8 ++++---- 2 files changed, 5 insertions(+), 5 deletions(-) diff --git a/doc/examples/operator__notequal.link b/doc/examples/operator__notequal.link index fa7acee8..bf369092 100644 --- a/doc/examples/operator__notequal.link +++ b/doc/examples/operator__notequal.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/operator__notequal.output b/doc/examples/operator__notequal.output index 9eba626e..6cbc0e8e 100644 --- a/doc/examples/operator__notequal.output +++ b/doc/examples/operator__notequal.output @@ -1,4 +1,4 @@ -[1,2,3] == [1,2,4] true -{"A":"a","B":"b"} == {"A":"a","B":"b"} false -17 == 17.0 false -"foo" == "bar" true +[1,2,3] != [1,2,4] true +{"A":"a","B":"b"} != {"A":"a","B":"b"} false +17 != 17.0 false +"foo" != "bar" true From 7b05e1a68736153d9b83ac3ebd6aa2ccf41c73dc Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sat, 8 Jul 2017 22:59:04 +0200 Subject: [PATCH 277/530] :hammer: improved performance in dump for deeply nested values --- src/json.hpp | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index e07aaa8c..612c7e58 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -6789,9 +6789,9 @@ class basic_json // variable to hold indentation for recursive calls const auto new_indent = current_indent + indent_step; - if (indent_string.size() < new_indent) + if (JSON_UNLIKELY(indent_string.size() < new_indent)) { - indent_string.resize(new_indent, ' '); + indent_string.resize(indent_string.size() * 2, ' '); } // first n-1 elements From f1c543cb7b229d8ff26a4d45ba92f8be55d2a47a Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sat, 8 Jul 2017 22:59:19 +0200 Subject: [PATCH 278/530] :construction_worker: added GCC7 --- .travis.yml | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/.travis.yml b/.travis.yml index 4ad1a555..dffec399 100644 --- a/.travis.yml +++ b/.travis.yml @@ -191,6 +191,14 @@ matrix: sources: ['ubuntu-toolchain-r-test'] packages: g++-6 + - os: linux + compiler: gcc + env: COMPILER=g++-7 + addons: + apt: + sources: ['ubuntu-toolchain-r-test'] + packages: g++-7 + # Linux / Clang - os: linux From f0ca96d462491a406aa975b3f4670a72b3a721e2 Mon Sep 17 00:00:00 2001 From: Oleg Endo Date: Sun, 9 Jul 2017 15:04:49 +0900 Subject: [PATCH 279/530] include instead of avoid bloat caused by and std::cout and friends in apps where iostream are not used. --- src/json.hpp | 2 +- test/src/unit-readme.cpp | 1 + test/src/unit-unicode.cpp | 1 + 3 files changed, 3 insertions(+), 1 deletion(-) diff --git a/src/json.hpp b/src/json.hpp index 612c7e58..b378c77d 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -43,7 +43,7 @@ SOFTWARE. #include // function, hash, less #include // initializer_list #include // hex -#include // istream, ostream +#include // istream, ostream #include // advance, begin, back_inserter, bidirectional_iterator_tag, distance, end, inserter, iterator, iterator_traits, next, random_access_iterator_tag, reverse_iterator #include // numeric_limits #include // locale diff --git a/test/src/unit-readme.cpp b/test/src/unit-readme.cpp index 1b22f1b3..e921c4b6 100644 --- a/test/src/unit-readme.cpp +++ b/test/src/unit-readme.cpp @@ -36,6 +36,7 @@ using nlohmann::json; #include #include #include +#include TEST_CASE("README", "[hide]") { diff --git a/test/src/unit-unicode.cpp b/test/src/unit-unicode.cpp index 9b379d96..e19905d5 100644 --- a/test/src/unit-unicode.cpp +++ b/test/src/unit-unicode.cpp @@ -33,6 +33,7 @@ SOFTWARE. using nlohmann::json; #include +#include extern size_t calls; size_t calls = 0; From b753cb6ee75cc59559b1f9976b62efae25bb76f7 Mon Sep 17 00:00:00 2001 From: Oleg Endo Date: Sun, 9 Jul 2017 15:19:08 +0900 Subject: [PATCH 280/530] use std::make_shared --- src/json.hpp | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index b378c77d..abf6b1f6 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -6634,17 +6634,17 @@ class basic_json static std::shared_ptr> create(std::vector& vec) { - return std::shared_ptr(new output_vector_adapter(vec)); + return std::make_shared>(vec); } static std::shared_ptr> create(std::ostream& s) { - return std::shared_ptr(new output_stream_adapter(s)); + return std::make_shared>(s); } static std::shared_ptr> create(std::string& s) { - return std::shared_ptr(new output_string_adapter(s)); + return std::make_shared>(s); } }; @@ -8767,19 +8767,19 @@ class basic_json /// input adapter for input stream static std::shared_ptr create(std::istream& i) { - return std::shared_ptr(new cached_input_stream_adapter<16384>(i)); + return std::make_shared> (i); } /// input adapter for input stream static std::shared_ptr create(std::istream&& i) { - return std::shared_ptr(new cached_input_stream_adapter<16384>(i)); + return std::make_shared>(i); } /// input adapter for buffer static std::shared_ptr create(const char* b, size_t l) { - return std::shared_ptr(new input_buffer_adapter(b, l)); + return std::make_shared(b, l); } // derived support From e3bb156f826edaa25dd4186cd094440274b9b8b1 Mon Sep 17 00:00:00 2001 From: Oleg Endo Date: Sun, 9 Jul 2017 15:20:28 +0900 Subject: [PATCH 281/530] fix typo --- src/json.hpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/json.hpp b/src/json.hpp index abf6b1f6..1bde08ce 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -6675,7 +6675,7 @@ class basic_json std::vector& v; }; - /// putput adapter for output streams + /// output adapter for output streams template class output_stream_adapter : public output_adapter { From 1b2fabe802a2975d4c9e1b02facdfb59d8062203 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 9 Jul 2017 11:10:18 +0200 Subject: [PATCH 282/530] :construction_worker: adding two recent Clang versions Maybe the simple approach is working now - at least it is in https://github.com/taocpp/PEGTL/blob/master/.travis.yml --- .travis.yml | 16 ++++++++++++++++ README.md | 3 ++- 2 files changed, 18 insertions(+), 1 deletion(-) diff --git a/.travis.yml b/.travis.yml index dffec399..ec1cbeb1 100644 --- a/.travis.yml +++ b/.travis.yml @@ -229,6 +229,22 @@ matrix: env: LLVM_VERSION=3.8.1 compiler: clang + - os: linux + compiler: clang + env: COMPILER=clang++-3.9 + addons: + apt: + sources: ['ubuntu-toolchain-r-test', 'llvm-toolchain-trusty-3.9'] + packages: clang-3.9 + + - os: linux + compiler: clang + env: COMPILER=clang++-4.0 + addons: + apt: + sources: ['ubuntu-toolchain-r-test', 'llvm-toolchain-trusty-4.0'] + packages: clang-4.0 + # - os: linux # addons: # apt: diff --git a/README.md b/README.md index 9aabc0fd..86d8b15b 100644 --- a/README.md +++ b/README.md @@ -709,7 +709,7 @@ json j_from_msgpack = json::from_msgpack(v_msgpack); Though it's 2016 already, the support for C++11 is still a bit sparse. Currently, the following compilers are known to work: -- GCC 4.9 - 6.0 (and possibly later) +- GCC 4.9 - 7.1 (and possibly later) - Clang 3.4 - 3.9 (and possibly later) - Microsoft Visual C++ 2015 / Build Tools 14.0.25123.0 (and possibly later) - Microsoft Visual C++ 2017 / Build Tools 15.1.548.43366 (and possibly later) @@ -738,6 +738,7 @@ The following compilers are currently used in continuous integration at [Travis] | GCC 4.9.4 | Ubuntu 14.04.5 LTS | g++-4.9 (Ubuntu 4.9.4-2ubuntu1~14.04.1) 4.9.4 | | GCC 5.4.1 | Ubuntu 14.04.5 LTS | g++-5 (Ubuntu 5.4.1-2ubuntu1~14.04) 5.4.1 20160904 | | GCC 6.3.0 | Ubuntu 14.04.5 LTS | g++-6 (Ubuntu/Linaro 6.3.0-18ubuntu2~14.04) 6.3.0 20170519 | +| GCC 7.1.0 | Ubuntu 14.04.5 LTS | g++-7 (Ubuntu 7.1.0-5ubuntu2~14.04) 7.1.0 | Clang 3.6.0 | Ubuntu 14.04.5 LTS | clang version 3.6.0 (tags/RELEASE_360/final) | | Clang 3.6.1 | Ubuntu 14.04.5 LTS | clang version 3.6.1 (tags/RELEASE_361/final) | | Clang 3.6.2 | Ubuntu 14.04.5 LTS | clang version 3.6.2 (tags/RELEASE_362/final) | From 504012a3db9f61b2468a75cc3f0fbf344b18b506 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 9 Jul 2017 11:51:38 +0200 Subject: [PATCH 283/530] :memo: cleanup after #650 As is not included in json.hpp any more, all code examples need to include now. --- README.md | 1 + doc/examples/README.cpp | 3 ++- doc/examples/README.link | 2 +- doc/examples/array.cpp | 1 + doc/examples/array.link | 2 +- doc/examples/at__object_t_key_type.cpp | 1 + doc/examples/at__object_t_key_type.link | 2 +- doc/examples/at__object_t_key_type_const.cpp | 1 + doc/examples/at__object_t_key_type_const.link | 2 +- doc/examples/at__size_type.cpp | 1 + doc/examples/at__size_type.link | 2 +- doc/examples/at__size_type_const.cpp | 1 + doc/examples/at__size_type_const.link | 2 +- doc/examples/at_json_pointer.cpp | 1 + doc/examples/at_json_pointer.link | 2 +- doc/examples/at_json_pointer_const.cpp | 1 + doc/examples/at_json_pointer_const.link | 2 +- doc/examples/back.cpp | 1 + doc/examples/back.link | 2 +- doc/examples/basic_json__CompatibleType.cpp | 3 ++- doc/examples/basic_json__CompatibleType.link | 2 +- doc/examples/basic_json__InputIt_InputIt.cpp | 1 + doc/examples/basic_json__InputIt_InputIt.link | 2 +- doc/examples/basic_json__basic_json.cpp | 1 + doc/examples/basic_json__basic_json.link | 2 +- doc/examples/basic_json__copyassignment.cpp | 1 + doc/examples/basic_json__copyassignment.link | 2 +- doc/examples/basic_json__list_init_t.cpp | 1 + doc/examples/basic_json__list_init_t.link | 2 +- doc/examples/basic_json__moveconstructor.cpp | 1 + doc/examples/basic_json__moveconstructor.link | 2 +- doc/examples/basic_json__nullptr_t.cpp | 1 + doc/examples/basic_json__nullptr_t.link | 2 +- doc/examples/basic_json__size_type_basic_json.cpp | 1 + doc/examples/basic_json__size_type_basic_json.link | 2 +- doc/examples/basic_json__value.cpp | 1 + doc/examples/basic_json__value.link | 2 +- doc/examples/basic_json__value_ptr.cpp | 1 + doc/examples/basic_json__value_ptr.link | 2 +- doc/examples/basic_json__value_t.cpp | 1 + doc/examples/basic_json__value_t.link | 2 +- doc/examples/begin.cpp | 1 + doc/examples/begin.link | 2 +- doc/examples/cbegin.cpp | 1 + doc/examples/cbegin.link | 2 +- doc/examples/cend.cpp | 1 + doc/examples/cend.link | 2 +- doc/examples/clear.cpp | 1 + doc/examples/clear.link | 2 +- doc/examples/count.cpp | 1 + doc/examples/count.link | 2 +- doc/examples/crbegin.cpp | 1 + doc/examples/crbegin.link | 2 +- doc/examples/crend.cpp | 1 + doc/examples/crend.link | 2 +- doc/examples/diff.cpp | 3 ++- doc/examples/diff.link | 2 +- doc/examples/dump.cpp | 1 + doc/examples/dump.link | 2 +- doc/examples/emplace.cpp | 1 + doc/examples/emplace.link | 2 +- doc/examples/emplace_back.cpp | 1 + doc/examples/emplace_back.link | 2 +- doc/examples/empty.cpp | 1 + doc/examples/empty.link | 2 +- doc/examples/end.cpp | 1 + doc/examples/end.link | 2 +- doc/examples/erase__IteratorType.cpp | 1 + doc/examples/erase__IteratorType.link | 2 +- doc/examples/erase__IteratorType_IteratorType.cpp | 1 + doc/examples/erase__IteratorType_IteratorType.link | 2 +- doc/examples/erase__key_type.cpp | 1 + doc/examples/erase__key_type.link | 2 +- doc/examples/erase__size_type.cpp | 1 + doc/examples/erase__size_type.link | 2 +- doc/examples/find__key_type.cpp | 1 + doc/examples/find__key_type.link | 2 +- doc/examples/flatten.cpp | 3 ++- doc/examples/flatten.link | 2 +- doc/examples/from_cbor.cpp | 3 ++- doc/examples/from_cbor.link | 2 +- doc/examples/from_msgpack.cpp | 3 ++- doc/examples/from_msgpack.link | 2 +- doc/examples/front.cpp | 1 + doc/examples/front.link | 2 +- doc/examples/get__PointerType.cpp | 1 + doc/examples/get__PointerType.link | 2 +- doc/examples/get__ValueType_const.cpp | 3 ++- doc/examples/get__ValueType_const.link | 2 +- doc/examples/get_ptr.cpp | 1 + doc/examples/get_ptr.link | 2 +- doc/examples/get_ref.cpp | 1 + doc/examples/get_ref.link | 2 +- doc/examples/insert.cpp | 1 + doc/examples/insert.link | 2 +- doc/examples/insert__count.cpp | 1 + doc/examples/insert__count.link | 2 +- doc/examples/insert__ilist.cpp | 1 + doc/examples/insert__ilist.link | 2 +- doc/examples/insert__range.cpp | 1 + doc/examples/insert__range.link | 2 +- doc/examples/insert__range_object.cpp | 1 + doc/examples/insert__range_object.link | 2 +- doc/examples/is_array.cpp | 1 + doc/examples/is_array.link | 2 +- doc/examples/is_boolean.cpp | 1 + doc/examples/is_boolean.link | 2 +- doc/examples/is_discarded.cpp | 1 + doc/examples/is_discarded.link | 2 +- doc/examples/is_null.cpp | 1 + doc/examples/is_null.link | 2 +- doc/examples/is_number.cpp | 1 + doc/examples/is_number.link | 2 +- doc/examples/is_number_float.cpp | 1 + doc/examples/is_number_float.link | 2 +- doc/examples/is_number_integer.cpp | 1 + doc/examples/is_number_integer.link | 2 +- doc/examples/is_number_unsigned.cpp | 1 + doc/examples/is_number_unsigned.link | 2 +- doc/examples/is_object.cpp | 1 + doc/examples/is_object.link | 2 +- doc/examples/is_primitive.cpp | 1 + doc/examples/is_primitive.link | 2 +- doc/examples/is_string.cpp | 1 + doc/examples/is_string.link | 2 +- doc/examples/is_structured.cpp | 1 + doc/examples/is_structured.link | 2 +- doc/examples/iterator_wrapper.cpp | 1 + doc/examples/iterator_wrapper.link | 2 +- doc/examples/json_pointer.cpp | 1 + doc/examples/json_pointer.link | 2 +- doc/examples/json_pointer__to_string.cpp | 1 + doc/examples/json_pointer__to_string.link | 2 +- doc/examples/max_size.cpp | 1 + doc/examples/max_size.link | 2 +- doc/examples/meta.cpp | 3 ++- doc/examples/meta.link | 2 +- doc/examples/object.cpp | 1 + doc/examples/object.link | 2 +- doc/examples/operator__ValueType.cpp | 3 ++- doc/examples/operator__ValueType.link | 2 +- doc/examples/operator__equal.cpp | 1 + doc/examples/operator__equal.link | 2 +- doc/examples/operator__equal__nullptr_t.cpp | 1 + doc/examples/operator__equal__nullptr_t.link | 2 +- doc/examples/operator__greater.cpp | 1 + doc/examples/operator__greater.link | 2 +- doc/examples/operator__greaterequal.cpp | 1 + doc/examples/operator__greaterequal.link | 2 +- doc/examples/operator__less.cpp | 1 + doc/examples/operator__less.link | 2 +- doc/examples/operator__lessequal.cpp | 1 + doc/examples/operator__lessequal.link | 2 +- doc/examples/operator__notequal.cpp | 1 + doc/examples/operator__notequal.link | 2 +- doc/examples/operator__notequal__nullptr_t.cpp | 1 + doc/examples/operator__notequal__nullptr_t.link | 2 +- doc/examples/operator__value_t.cpp | 1 + doc/examples/operator__value_t.link | 2 +- doc/examples/operator_deserialize.cpp | 3 ++- doc/examples/operator_deserialize.link | 2 +- doc/examples/operator_serialize.cpp | 3 ++- doc/examples/operator_serialize.link | 2 +- doc/examples/operatorarray__key_type.cpp | 3 ++- doc/examples/operatorarray__key_type.link | 2 +- doc/examples/operatorarray__key_type_const.cpp | 1 + doc/examples/operatorarray__key_type_const.link | 2 +- doc/examples/operatorarray__size_type.cpp | 1 + doc/examples/operatorarray__size_type.link | 2 +- doc/examples/operatorarray__size_type_const.cpp | 1 + doc/examples/operatorarray__size_type_const.link | 2 +- doc/examples/operatorjson_pointer.cpp | 1 + doc/examples/operatorjson_pointer.link | 2 +- doc/examples/operatorjson_pointer_const.cpp | 1 + doc/examples/operatorjson_pointer_const.link | 2 +- doc/examples/parse__array__parser_callback_t.cpp | 3 ++- doc/examples/parse__array__parser_callback_t.link | 2 +- doc/examples/parse__contiguouscontainer__parser_callback_t.cpp | 3 ++- .../parse__contiguouscontainer__parser_callback_t.link | 2 +- doc/examples/parse__istream__parser_callback_t.cpp | 3 ++- doc/examples/parse__istream__parser_callback_t.link | 2 +- doc/examples/parse__iteratortype__parser_callback_t.cpp | 3 ++- doc/examples/parse__iteratortype__parser_callback_t.link | 2 +- doc/examples/parse__string__parser_callback_t.cpp | 3 ++- doc/examples/parse__string__parser_callback_t.link | 2 +- doc/examples/patch.cpp | 1 + doc/examples/patch.link | 2 +- doc/examples/push_back.cpp | 1 + doc/examples/push_back.link | 2 +- doc/examples/push_back__initializer_list.cpp | 1 + doc/examples/push_back__initializer_list.link | 2 +- doc/examples/push_back__object_t__value.cpp | 1 + doc/examples/push_back__object_t__value.link | 2 +- doc/examples/rbegin.cpp | 1 + doc/examples/rbegin.link | 2 +- doc/examples/rend.cpp | 1 + doc/examples/rend.link | 2 +- doc/examples/size.cpp | 1 + doc/examples/size.link | 2 +- doc/examples/swap__array_t.cpp | 1 + doc/examples/swap__array_t.link | 2 +- doc/examples/swap__object_t.cpp | 1 + doc/examples/swap__object_t.link | 2 +- doc/examples/swap__reference.cpp | 1 + doc/examples/swap__reference.link | 2 +- doc/examples/swap__string_t.cpp | 1 + doc/examples/swap__string_t.link | 2 +- doc/examples/to_cbor.cpp | 3 ++- doc/examples/to_cbor.link | 2 +- doc/examples/to_msgpack.cpp | 3 ++- doc/examples/to_msgpack.link | 2 +- doc/examples/type.cpp | 1 + doc/examples/type.link | 2 +- doc/examples/type_name.cpp | 1 + doc/examples/type_name.link | 2 +- doc/examples/unflatten.cpp | 3 ++- doc/examples/unflatten.link | 2 +- 217 files changed, 237 insertions(+), 128 deletions(-) diff --git a/README.md b/README.md index 86d8b15b..a81e1018 100644 --- a/README.md +++ b/README.md @@ -849,6 +849,7 @@ I deeply appreciate the help of the following people. - [Steffen](https://github.com/koemeet) fixed a potential issue with MSVC and `std::min`. - [Mike Tzou](https://github.com/Chocobo1) fixed some typos. - [amrcode](https://github.com/amrcode) noted a missleading documentation about comparison of floats. +- [Oleg Endo](https://github.com/olegendo) reduced the memory consumption by replacing `` with `` Thanks a lot for helping out! Please [let me know](mailto:mail@nlohmann.me) if I forgot someone. diff --git a/doc/examples/README.cpp b/doc/examples/README.cpp index a93ee60a..00c93b5d 100644 --- a/doc/examples/README.cpp +++ b/doc/examples/README.cpp @@ -1,5 +1,6 @@ -#include "json.hpp" +#include #include // for std::setw +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/README.link b/doc/examples/README.link index 8758bf05..19e3976a 100644 --- a/doc/examples/README.link +++ b/doc/examples/README.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/array.cpp b/doc/examples/array.cpp index d3cc4777..b43be5a6 100644 --- a/doc/examples/array.cpp +++ b/doc/examples/array.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/array.link b/doc/examples/array.link index a7e40831..cd77ed14 100644 --- a/doc/examples/array.link +++ b/doc/examples/array.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/at__object_t_key_type.cpp b/doc/examples/at__object_t_key_type.cpp index 611ca226..0430617e 100644 --- a/doc/examples/at__object_t_key_type.cpp +++ b/doc/examples/at__object_t_key_type.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/at__object_t_key_type.link b/doc/examples/at__object_t_key_type.link index cff20bea..b4d6efc8 100644 --- a/doc/examples/at__object_t_key_type.link +++ b/doc/examples/at__object_t_key_type.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/at__object_t_key_type_const.cpp b/doc/examples/at__object_t_key_type_const.cpp index eae7da96..9aa2ed33 100644 --- a/doc/examples/at__object_t_key_type_const.cpp +++ b/doc/examples/at__object_t_key_type_const.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/at__object_t_key_type_const.link b/doc/examples/at__object_t_key_type_const.link index 43bb6c87..6e6ca335 100644 --- a/doc/examples/at__object_t_key_type_const.link +++ b/doc/examples/at__object_t_key_type_const.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/at__size_type.cpp b/doc/examples/at__size_type.cpp index 17381738..202e44b1 100644 --- a/doc/examples/at__size_type.cpp +++ b/doc/examples/at__size_type.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/at__size_type.link b/doc/examples/at__size_type.link index fea3bcd5..5dec3752 100644 --- a/doc/examples/at__size_type.link +++ b/doc/examples/at__size_type.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/at__size_type_const.cpp b/doc/examples/at__size_type_const.cpp index 3f67d378..33108ce8 100644 --- a/doc/examples/at__size_type_const.cpp +++ b/doc/examples/at__size_type_const.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/at__size_type_const.link b/doc/examples/at__size_type_const.link index 70dd9151..7412f09f 100644 --- a/doc/examples/at__size_type_const.link +++ b/doc/examples/at__size_type_const.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/at_json_pointer.cpp b/doc/examples/at_json_pointer.cpp index 2cb62e2d..80ed6c15 100644 --- a/doc/examples/at_json_pointer.cpp +++ b/doc/examples/at_json_pointer.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/at_json_pointer.link b/doc/examples/at_json_pointer.link index 2e6a3e70..61a4410a 100644 --- a/doc/examples/at_json_pointer.link +++ b/doc/examples/at_json_pointer.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/at_json_pointer_const.cpp b/doc/examples/at_json_pointer_const.cpp index 3b09f65d..1496aa5a 100644 --- a/doc/examples/at_json_pointer_const.cpp +++ b/doc/examples/at_json_pointer_const.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/at_json_pointer_const.link b/doc/examples/at_json_pointer_const.link index 1ea3b078..31e0bb08 100644 --- a/doc/examples/at_json_pointer_const.link +++ b/doc/examples/at_json_pointer_const.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/back.cpp b/doc/examples/back.cpp index f4dcced3..cb227c2b 100644 --- a/doc/examples/back.cpp +++ b/doc/examples/back.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/back.link b/doc/examples/back.link index 8d757cde..fa1ce912 100644 --- a/doc/examples/back.link +++ b/doc/examples/back.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/basic_json__CompatibleType.cpp b/doc/examples/basic_json__CompatibleType.cpp index 85dbccd0..1066795b 100644 --- a/doc/examples/basic_json__CompatibleType.cpp +++ b/doc/examples/basic_json__CompatibleType.cpp @@ -1,10 +1,11 @@ -#include "json.hpp" +#include #include #include #include #include #include #include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/basic_json__CompatibleType.link b/doc/examples/basic_json__CompatibleType.link index 3ced745b..af1157b7 100644 --- a/doc/examples/basic_json__CompatibleType.link +++ b/doc/examples/basic_json__CompatibleType.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/basic_json__InputIt_InputIt.cpp b/doc/examples/basic_json__InputIt_InputIt.cpp index 60043ff3..36c6f571 100644 --- a/doc/examples/basic_json__InputIt_InputIt.cpp +++ b/doc/examples/basic_json__InputIt_InputIt.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/basic_json__InputIt_InputIt.link b/doc/examples/basic_json__InputIt_InputIt.link index c7c09d38..aad58f73 100644 --- a/doc/examples/basic_json__InputIt_InputIt.link +++ b/doc/examples/basic_json__InputIt_InputIt.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/basic_json__basic_json.cpp b/doc/examples/basic_json__basic_json.cpp index c218a8e5..98df6101 100644 --- a/doc/examples/basic_json__basic_json.cpp +++ b/doc/examples/basic_json__basic_json.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/basic_json__basic_json.link b/doc/examples/basic_json__basic_json.link index e8ac682d..e894b663 100644 --- a/doc/examples/basic_json__basic_json.link +++ b/doc/examples/basic_json__basic_json.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/basic_json__copyassignment.cpp b/doc/examples/basic_json__copyassignment.cpp index b44e5993..4f1bc439 100644 --- a/doc/examples/basic_json__copyassignment.cpp +++ b/doc/examples/basic_json__copyassignment.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/basic_json__copyassignment.link b/doc/examples/basic_json__copyassignment.link index 1bff32a7..eec9258f 100644 --- a/doc/examples/basic_json__copyassignment.link +++ b/doc/examples/basic_json__copyassignment.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/basic_json__list_init_t.cpp b/doc/examples/basic_json__list_init_t.cpp index 5d77dbbc..ba1bce4e 100644 --- a/doc/examples/basic_json__list_init_t.cpp +++ b/doc/examples/basic_json__list_init_t.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/basic_json__list_init_t.link b/doc/examples/basic_json__list_init_t.link index 4419ca9a..75234b62 100644 --- a/doc/examples/basic_json__list_init_t.link +++ b/doc/examples/basic_json__list_init_t.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/basic_json__moveconstructor.cpp b/doc/examples/basic_json__moveconstructor.cpp index ecfdb603..34827259 100644 --- a/doc/examples/basic_json__moveconstructor.cpp +++ b/doc/examples/basic_json__moveconstructor.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/basic_json__moveconstructor.link b/doc/examples/basic_json__moveconstructor.link index 7bad9910..f9047302 100644 --- a/doc/examples/basic_json__moveconstructor.link +++ b/doc/examples/basic_json__moveconstructor.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/basic_json__nullptr_t.cpp b/doc/examples/basic_json__nullptr_t.cpp index 041a3269..886d5ea3 100644 --- a/doc/examples/basic_json__nullptr_t.cpp +++ b/doc/examples/basic_json__nullptr_t.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/basic_json__nullptr_t.link b/doc/examples/basic_json__nullptr_t.link index 143d20a2..003110f1 100644 --- a/doc/examples/basic_json__nullptr_t.link +++ b/doc/examples/basic_json__nullptr_t.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/basic_json__size_type_basic_json.cpp b/doc/examples/basic_json__size_type_basic_json.cpp index 1add4925..a021edda 100644 --- a/doc/examples/basic_json__size_type_basic_json.cpp +++ b/doc/examples/basic_json__size_type_basic_json.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/basic_json__size_type_basic_json.link b/doc/examples/basic_json__size_type_basic_json.link index e699f9ad..e48b9fd9 100644 --- a/doc/examples/basic_json__size_type_basic_json.link +++ b/doc/examples/basic_json__size_type_basic_json.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/basic_json__value.cpp b/doc/examples/basic_json__value.cpp index fced71c3..42d4aa31 100644 --- a/doc/examples/basic_json__value.cpp +++ b/doc/examples/basic_json__value.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/basic_json__value.link b/doc/examples/basic_json__value.link index 3dc8a407..d0182781 100644 --- a/doc/examples/basic_json__value.link +++ b/doc/examples/basic_json__value.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/basic_json__value_ptr.cpp b/doc/examples/basic_json__value_ptr.cpp index 262fbbf6..4f6f999e 100644 --- a/doc/examples/basic_json__value_ptr.cpp +++ b/doc/examples/basic_json__value_ptr.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/basic_json__value_ptr.link b/doc/examples/basic_json__value_ptr.link index 482c75f0..799f96f7 100644 --- a/doc/examples/basic_json__value_ptr.link +++ b/doc/examples/basic_json__value_ptr.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/basic_json__value_t.cpp b/doc/examples/basic_json__value_t.cpp index 20cd3b4d..d52e628b 100644 --- a/doc/examples/basic_json__value_t.cpp +++ b/doc/examples/basic_json__value_t.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/basic_json__value_t.link b/doc/examples/basic_json__value_t.link index 8e999a8d..a66b1dd4 100644 --- a/doc/examples/basic_json__value_t.link +++ b/doc/examples/basic_json__value_t.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/begin.cpp b/doc/examples/begin.cpp index c2bb43ae..ebf97e5f 100644 --- a/doc/examples/begin.cpp +++ b/doc/examples/begin.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/begin.link b/doc/examples/begin.link index da79b631..64ac5fba 100644 --- a/doc/examples/begin.link +++ b/doc/examples/begin.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/cbegin.cpp b/doc/examples/cbegin.cpp index f4353ef0..d12c8608 100644 --- a/doc/examples/cbegin.cpp +++ b/doc/examples/cbegin.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/cbegin.link b/doc/examples/cbegin.link index dee6deb2..fe670f15 100644 --- a/doc/examples/cbegin.link +++ b/doc/examples/cbegin.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/cend.cpp b/doc/examples/cend.cpp index 08914124..8d141acf 100644 --- a/doc/examples/cend.cpp +++ b/doc/examples/cend.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/cend.link b/doc/examples/cend.link index 6dfc2277..3c726426 100644 --- a/doc/examples/cend.link +++ b/doc/examples/cend.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/clear.cpp b/doc/examples/clear.cpp index 055211cf..e892b13e 100644 --- a/doc/examples/clear.cpp +++ b/doc/examples/clear.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/clear.link b/doc/examples/clear.link index 91fa1f25..1123e2f3 100644 --- a/doc/examples/clear.link +++ b/doc/examples/clear.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/count.cpp b/doc/examples/count.cpp index b3f10d06..cd030fff 100644 --- a/doc/examples/count.cpp +++ b/doc/examples/count.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/count.link b/doc/examples/count.link index 6167a673..84666742 100644 --- a/doc/examples/count.link +++ b/doc/examples/count.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/crbegin.cpp b/doc/examples/crbegin.cpp index df0868ab..6dc4a6f7 100644 --- a/doc/examples/crbegin.cpp +++ b/doc/examples/crbegin.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/crbegin.link b/doc/examples/crbegin.link index 49072710..f97dac66 100644 --- a/doc/examples/crbegin.link +++ b/doc/examples/crbegin.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/crend.cpp b/doc/examples/crend.cpp index 20c2f82a..9a00950c 100644 --- a/doc/examples/crend.cpp +++ b/doc/examples/crend.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/crend.link b/doc/examples/crend.link index 7c244144..208c7c96 100644 --- a/doc/examples/crend.link +++ b/doc/examples/crend.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/diff.cpp b/doc/examples/diff.cpp index 982b8f9c..ee0aefc0 100644 --- a/doc/examples/diff.cpp +++ b/doc/examples/diff.cpp @@ -1,5 +1,6 @@ -#include "json.hpp" +#include #include // for std::setw +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/diff.link b/doc/examples/diff.link index d0bed27a..e1cfdb8d 100644 --- a/doc/examples/diff.link +++ b/doc/examples/diff.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/dump.cpp b/doc/examples/dump.cpp index d1359684..4ffd2aa4 100644 --- a/doc/examples/dump.cpp +++ b/doc/examples/dump.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/dump.link b/doc/examples/dump.link index ae5a4aba..2e3fee96 100644 --- a/doc/examples/dump.link +++ b/doc/examples/dump.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/emplace.cpp b/doc/examples/emplace.cpp index 2af633eb..7e80b68c 100644 --- a/doc/examples/emplace.cpp +++ b/doc/examples/emplace.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/emplace.link b/doc/examples/emplace.link index b689c476..809cc21e 100644 --- a/doc/examples/emplace.link +++ b/doc/examples/emplace.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/emplace_back.cpp b/doc/examples/emplace_back.cpp index f76e599f..6a50fd20 100644 --- a/doc/examples/emplace_back.cpp +++ b/doc/examples/emplace_back.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/emplace_back.link b/doc/examples/emplace_back.link index 824c95df..9397a892 100644 --- a/doc/examples/emplace_back.link +++ b/doc/examples/emplace_back.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/empty.cpp b/doc/examples/empty.cpp index a4c71b52..94bc46a3 100644 --- a/doc/examples/empty.cpp +++ b/doc/examples/empty.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/empty.link b/doc/examples/empty.link index c16e0296..90a6590e 100644 --- a/doc/examples/empty.link +++ b/doc/examples/empty.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/end.cpp b/doc/examples/end.cpp index 5eeec81d..2984f9ea 100644 --- a/doc/examples/end.cpp +++ b/doc/examples/end.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/end.link b/doc/examples/end.link index 4ced3b90..2695e1d5 100644 --- a/doc/examples/end.link +++ b/doc/examples/end.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/erase__IteratorType.cpp b/doc/examples/erase__IteratorType.cpp index 2ec6204f..0ba7e6fc 100644 --- a/doc/examples/erase__IteratorType.cpp +++ b/doc/examples/erase__IteratorType.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/erase__IteratorType.link b/doc/examples/erase__IteratorType.link index 50571c91..71d41146 100644 --- a/doc/examples/erase__IteratorType.link +++ b/doc/examples/erase__IteratorType.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/erase__IteratorType_IteratorType.cpp b/doc/examples/erase__IteratorType_IteratorType.cpp index ef3f070c..efbf789e 100644 --- a/doc/examples/erase__IteratorType_IteratorType.cpp +++ b/doc/examples/erase__IteratorType_IteratorType.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/erase__IteratorType_IteratorType.link b/doc/examples/erase__IteratorType_IteratorType.link index fb7fca42..9209f64c 100644 --- a/doc/examples/erase__IteratorType_IteratorType.link +++ b/doc/examples/erase__IteratorType_IteratorType.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/erase__key_type.cpp b/doc/examples/erase__key_type.cpp index e035c23b..592aa020 100644 --- a/doc/examples/erase__key_type.cpp +++ b/doc/examples/erase__key_type.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/erase__key_type.link b/doc/examples/erase__key_type.link index 8a1579c9..b3ab45ef 100644 --- a/doc/examples/erase__key_type.link +++ b/doc/examples/erase__key_type.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/erase__size_type.cpp b/doc/examples/erase__size_type.cpp index 68bbab64..72021bad 100644 --- a/doc/examples/erase__size_type.cpp +++ b/doc/examples/erase__size_type.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/erase__size_type.link b/doc/examples/erase__size_type.link index cd20c392..33a124ca 100644 --- a/doc/examples/erase__size_type.link +++ b/doc/examples/erase__size_type.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/find__key_type.cpp b/doc/examples/find__key_type.cpp index cb4a51c7..c411ba4f 100644 --- a/doc/examples/find__key_type.cpp +++ b/doc/examples/find__key_type.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/find__key_type.link b/doc/examples/find__key_type.link index bf003fb1..0be7a24a 100644 --- a/doc/examples/find__key_type.link +++ b/doc/examples/find__key_type.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/flatten.cpp b/doc/examples/flatten.cpp index b0cf9b6c..a69c31d4 100644 --- a/doc/examples/flatten.cpp +++ b/doc/examples/flatten.cpp @@ -1,5 +1,6 @@ -#include "json.hpp" +#include #include // for std::setw +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/flatten.link b/doc/examples/flatten.link index b5285f3c..4d03de6b 100644 --- a/doc/examples/flatten.link +++ b/doc/examples/flatten.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/from_cbor.cpp b/doc/examples/from_cbor.cpp index 5d78be8f..b0340f17 100644 --- a/doc/examples/from_cbor.cpp +++ b/doc/examples/from_cbor.cpp @@ -1,5 +1,6 @@ -#include "json.hpp" +#include #include // for std::setw +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/from_cbor.link b/doc/examples/from_cbor.link index 15a2bf40..129e0332 100644 --- a/doc/examples/from_cbor.link +++ b/doc/examples/from_cbor.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/from_msgpack.cpp b/doc/examples/from_msgpack.cpp index a16c1bdd..5df9ef3e 100644 --- a/doc/examples/from_msgpack.cpp +++ b/doc/examples/from_msgpack.cpp @@ -1,5 +1,6 @@ -#include "json.hpp" +#include #include // for std::setw +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/from_msgpack.link b/doc/examples/from_msgpack.link index a4818bd7..7d3a1438 100644 --- a/doc/examples/from_msgpack.link +++ b/doc/examples/from_msgpack.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/front.cpp b/doc/examples/front.cpp index e9f09d0c..bbc372e0 100644 --- a/doc/examples/front.cpp +++ b/doc/examples/front.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/front.link b/doc/examples/front.link index e3f57b3d..4f8bcd77 100644 --- a/doc/examples/front.link +++ b/doc/examples/front.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/get__PointerType.cpp b/doc/examples/get__PointerType.cpp index 613305f9..29f9ad34 100644 --- a/doc/examples/get__PointerType.cpp +++ b/doc/examples/get__PointerType.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/get__PointerType.link b/doc/examples/get__PointerType.link index 5c0713d5..73a3fc51 100644 --- a/doc/examples/get__PointerType.link +++ b/doc/examples/get__PointerType.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/get__ValueType_const.cpp b/doc/examples/get__ValueType_const.cpp index 3fdd2834..d78a6b26 100644 --- a/doc/examples/get__ValueType_const.cpp +++ b/doc/examples/get__ValueType_const.cpp @@ -1,5 +1,6 @@ -#include "json.hpp" +#include #include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/get__ValueType_const.link b/doc/examples/get__ValueType_const.link index d58c1789..7bf5e8a3 100644 --- a/doc/examples/get__ValueType_const.link +++ b/doc/examples/get__ValueType_const.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/get_ptr.cpp b/doc/examples/get_ptr.cpp index 54288450..27deb149 100644 --- a/doc/examples/get_ptr.cpp +++ b/doc/examples/get_ptr.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/get_ptr.link b/doc/examples/get_ptr.link index 2b158c8f..6c4a0684 100644 --- a/doc/examples/get_ptr.link +++ b/doc/examples/get_ptr.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/get_ref.cpp b/doc/examples/get_ref.cpp index c07d8488..951fea3e 100644 --- a/doc/examples/get_ref.cpp +++ b/doc/examples/get_ref.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/get_ref.link b/doc/examples/get_ref.link index 1181b7c6..d5bb4cfd 100644 --- a/doc/examples/get_ref.link +++ b/doc/examples/get_ref.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/insert.cpp b/doc/examples/insert.cpp index 99879566..913ad09d 100644 --- a/doc/examples/insert.cpp +++ b/doc/examples/insert.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/insert.link b/doc/examples/insert.link index f3f4e4bc..803d5c62 100644 --- a/doc/examples/insert.link +++ b/doc/examples/insert.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/insert__count.cpp b/doc/examples/insert__count.cpp index 11460e51..3c2ac24a 100644 --- a/doc/examples/insert__count.cpp +++ b/doc/examples/insert__count.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/insert__count.link b/doc/examples/insert__count.link index 103795f5..a76cd8e2 100644 --- a/doc/examples/insert__count.link +++ b/doc/examples/insert__count.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/insert__ilist.cpp b/doc/examples/insert__ilist.cpp index fa9e8612..90965b37 100644 --- a/doc/examples/insert__ilist.cpp +++ b/doc/examples/insert__ilist.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/insert__ilist.link b/doc/examples/insert__ilist.link index 13805fbf..387dce69 100644 --- a/doc/examples/insert__ilist.link +++ b/doc/examples/insert__ilist.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/insert__range.cpp b/doc/examples/insert__range.cpp index 8e8a5977..5b0aa9ea 100644 --- a/doc/examples/insert__range.cpp +++ b/doc/examples/insert__range.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/insert__range.link b/doc/examples/insert__range.link index 30440793..cd8107c0 100644 --- a/doc/examples/insert__range.link +++ b/doc/examples/insert__range.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/insert__range_object.cpp b/doc/examples/insert__range_object.cpp index 3f266b22..b811f8aa 100644 --- a/doc/examples/insert__range_object.cpp +++ b/doc/examples/insert__range_object.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/insert__range_object.link b/doc/examples/insert__range_object.link index f6eb481a..7c67093b 100644 --- a/doc/examples/insert__range_object.link +++ b/doc/examples/insert__range_object.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/is_array.cpp b/doc/examples/is_array.cpp index 7a5d20bf..1dbbf5d2 100644 --- a/doc/examples/is_array.cpp +++ b/doc/examples/is_array.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/is_array.link b/doc/examples/is_array.link index ad18000c..7820d336 100644 --- a/doc/examples/is_array.link +++ b/doc/examples/is_array.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/is_boolean.cpp b/doc/examples/is_boolean.cpp index d3a0e178..485f9704 100644 --- a/doc/examples/is_boolean.cpp +++ b/doc/examples/is_boolean.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/is_boolean.link b/doc/examples/is_boolean.link index bb89c4a0..9b56f6ed 100644 --- a/doc/examples/is_boolean.link +++ b/doc/examples/is_boolean.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/is_discarded.cpp b/doc/examples/is_discarded.cpp index f4953bf9..0b0435c3 100644 --- a/doc/examples/is_discarded.cpp +++ b/doc/examples/is_discarded.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/is_discarded.link b/doc/examples/is_discarded.link index 407667aa..f20d9ce9 100644 --- a/doc/examples/is_discarded.link +++ b/doc/examples/is_discarded.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/is_null.cpp b/doc/examples/is_null.cpp index 9dd0528c..3043e184 100644 --- a/doc/examples/is_null.cpp +++ b/doc/examples/is_null.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/is_null.link b/doc/examples/is_null.link index 6ead545c..05b27018 100644 --- a/doc/examples/is_null.link +++ b/doc/examples/is_null.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/is_number.cpp b/doc/examples/is_number.cpp index a0baff32..08fa138d 100644 --- a/doc/examples/is_number.cpp +++ b/doc/examples/is_number.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/is_number.link b/doc/examples/is_number.link index 7e291e79..b1e9ad2a 100644 --- a/doc/examples/is_number.link +++ b/doc/examples/is_number.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/is_number_float.cpp b/doc/examples/is_number_float.cpp index a67d9271..aeef78e8 100644 --- a/doc/examples/is_number_float.cpp +++ b/doc/examples/is_number_float.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/is_number_float.link b/doc/examples/is_number_float.link index 3f56570a..17208079 100644 --- a/doc/examples/is_number_float.link +++ b/doc/examples/is_number_float.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/is_number_integer.cpp b/doc/examples/is_number_integer.cpp index b8f772ef..90bd2146 100644 --- a/doc/examples/is_number_integer.cpp +++ b/doc/examples/is_number_integer.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/is_number_integer.link b/doc/examples/is_number_integer.link index 692d0208..a35be75d 100644 --- a/doc/examples/is_number_integer.link +++ b/doc/examples/is_number_integer.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/is_number_unsigned.cpp b/doc/examples/is_number_unsigned.cpp index 7cd04a31..499530c5 100644 --- a/doc/examples/is_number_unsigned.cpp +++ b/doc/examples/is_number_unsigned.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/is_number_unsigned.link b/doc/examples/is_number_unsigned.link index 930ec498..0a2ba020 100644 --- a/doc/examples/is_number_unsigned.link +++ b/doc/examples/is_number_unsigned.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/is_object.cpp b/doc/examples/is_object.cpp index 1fe66ec2..a0c03a76 100644 --- a/doc/examples/is_object.cpp +++ b/doc/examples/is_object.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/is_object.link b/doc/examples/is_object.link index 5bfaa41a..0cf3d4ad 100644 --- a/doc/examples/is_object.link +++ b/doc/examples/is_object.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/is_primitive.cpp b/doc/examples/is_primitive.cpp index fb863b61..ab329bdb 100644 --- a/doc/examples/is_primitive.cpp +++ b/doc/examples/is_primitive.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/is_primitive.link b/doc/examples/is_primitive.link index aa46ead7..2a0e91b9 100644 --- a/doc/examples/is_primitive.link +++ b/doc/examples/is_primitive.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/is_string.cpp b/doc/examples/is_string.cpp index df887007..44cb483c 100644 --- a/doc/examples/is_string.cpp +++ b/doc/examples/is_string.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/is_string.link b/doc/examples/is_string.link index b4d8cc39..f3672db3 100644 --- a/doc/examples/is_string.link +++ b/doc/examples/is_string.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/is_structured.cpp b/doc/examples/is_structured.cpp index 66016083..b7dfc858 100644 --- a/doc/examples/is_structured.cpp +++ b/doc/examples/is_structured.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/is_structured.link b/doc/examples/is_structured.link index ef6804b7..8ea3a75f 100644 --- a/doc/examples/is_structured.link +++ b/doc/examples/is_structured.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/iterator_wrapper.cpp b/doc/examples/iterator_wrapper.cpp index 208bb705..61687a92 100644 --- a/doc/examples/iterator_wrapper.cpp +++ b/doc/examples/iterator_wrapper.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/iterator_wrapper.link b/doc/examples/iterator_wrapper.link index c8f696c3..0087d9a1 100644 --- a/doc/examples/iterator_wrapper.link +++ b/doc/examples/iterator_wrapper.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/json_pointer.cpp b/doc/examples/json_pointer.cpp index 333d3963..23e8e84f 100644 --- a/doc/examples/json_pointer.cpp +++ b/doc/examples/json_pointer.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/json_pointer.link b/doc/examples/json_pointer.link index 541a97d8..3ba4355f 100644 --- a/doc/examples/json_pointer.link +++ b/doc/examples/json_pointer.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/json_pointer__to_string.cpp b/doc/examples/json_pointer__to_string.cpp index fc7f2557..70b72778 100644 --- a/doc/examples/json_pointer__to_string.cpp +++ b/doc/examples/json_pointer__to_string.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/json_pointer__to_string.link b/doc/examples/json_pointer__to_string.link index 5cae597e..b1148125 100644 --- a/doc/examples/json_pointer__to_string.link +++ b/doc/examples/json_pointer__to_string.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/max_size.cpp b/doc/examples/max_size.cpp index 27989271..8273f86f 100644 --- a/doc/examples/max_size.cpp +++ b/doc/examples/max_size.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/max_size.link b/doc/examples/max_size.link index 2bca668d..d49f5ea8 100644 --- a/doc/examples/max_size.link +++ b/doc/examples/max_size.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/meta.cpp b/doc/examples/meta.cpp index 8e4ad3de..5a762322 100644 --- a/doc/examples/meta.cpp +++ b/doc/examples/meta.cpp @@ -1,5 +1,6 @@ -#include "json.hpp" +#include #include // for std::setw +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/meta.link b/doc/examples/meta.link index 05b7ef06..ca563582 100644 --- a/doc/examples/meta.link +++ b/doc/examples/meta.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/object.cpp b/doc/examples/object.cpp index 3c089b0f..6c117758 100644 --- a/doc/examples/object.cpp +++ b/doc/examples/object.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/object.link b/doc/examples/object.link index 0a435eb1..f8638334 100644 --- a/doc/examples/object.link +++ b/doc/examples/object.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/operator__ValueType.cpp b/doc/examples/operator__ValueType.cpp index 4895fe1e..2def5271 100644 --- a/doc/examples/operator__ValueType.cpp +++ b/doc/examples/operator__ValueType.cpp @@ -1,5 +1,6 @@ -#include "json.hpp" +#include #include +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/operator__ValueType.link b/doc/examples/operator__ValueType.link index 84497ae4..0c32d631 100644 --- a/doc/examples/operator__ValueType.link +++ b/doc/examples/operator__ValueType.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/operator__equal.cpp b/doc/examples/operator__equal.cpp index e9910f32..d541a212 100644 --- a/doc/examples/operator__equal.cpp +++ b/doc/examples/operator__equal.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/operator__equal.link b/doc/examples/operator__equal.link index 910bf449..ae7c25ae 100644 --- a/doc/examples/operator__equal.link +++ b/doc/examples/operator__equal.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/operator__equal__nullptr_t.cpp b/doc/examples/operator__equal__nullptr_t.cpp index d7c2fbdc..b5d0c640 100644 --- a/doc/examples/operator__equal__nullptr_t.cpp +++ b/doc/examples/operator__equal__nullptr_t.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/operator__equal__nullptr_t.link b/doc/examples/operator__equal__nullptr_t.link index 30030502..90cc1744 100644 --- a/doc/examples/operator__equal__nullptr_t.link +++ b/doc/examples/operator__equal__nullptr_t.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/operator__greater.cpp b/doc/examples/operator__greater.cpp index 1860625c..a5c4f78a 100644 --- a/doc/examples/operator__greater.cpp +++ b/doc/examples/operator__greater.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/operator__greater.link b/doc/examples/operator__greater.link index f28c8fe0..7f082791 100644 --- a/doc/examples/operator__greater.link +++ b/doc/examples/operator__greater.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/operator__greaterequal.cpp b/doc/examples/operator__greaterequal.cpp index e1d5d8b7..da014bde 100644 --- a/doc/examples/operator__greaterequal.cpp +++ b/doc/examples/operator__greaterequal.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/operator__greaterequal.link b/doc/examples/operator__greaterequal.link index a03af050..599f5d7b 100644 --- a/doc/examples/operator__greaterequal.link +++ b/doc/examples/operator__greaterequal.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/operator__less.cpp b/doc/examples/operator__less.cpp index 53a595a9..3865847f 100644 --- a/doc/examples/operator__less.cpp +++ b/doc/examples/operator__less.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/operator__less.link b/doc/examples/operator__less.link index 9d8a489a..0689f995 100644 --- a/doc/examples/operator__less.link +++ b/doc/examples/operator__less.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/operator__lessequal.cpp b/doc/examples/operator__lessequal.cpp index 6d82e894..8e9a21f1 100644 --- a/doc/examples/operator__lessequal.cpp +++ b/doc/examples/operator__lessequal.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/operator__lessequal.link b/doc/examples/operator__lessequal.link index 637846fb..57681851 100644 --- a/doc/examples/operator__lessequal.link +++ b/doc/examples/operator__lessequal.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/operator__notequal.cpp b/doc/examples/operator__notequal.cpp index 802447b0..debcdf8a 100644 --- a/doc/examples/operator__notequal.cpp +++ b/doc/examples/operator__notequal.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/operator__notequal.link b/doc/examples/operator__notequal.link index bf369092..c23e8263 100644 --- a/doc/examples/operator__notequal.link +++ b/doc/examples/operator__notequal.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/operator__notequal__nullptr_t.cpp b/doc/examples/operator__notequal__nullptr_t.cpp index e65dcd7c..8d74e713 100644 --- a/doc/examples/operator__notequal__nullptr_t.cpp +++ b/doc/examples/operator__notequal__nullptr_t.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/operator__notequal__nullptr_t.link b/doc/examples/operator__notequal__nullptr_t.link index f8be3cbe..49864d34 100644 --- a/doc/examples/operator__notequal__nullptr_t.link +++ b/doc/examples/operator__notequal__nullptr_t.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/operator__value_t.cpp b/doc/examples/operator__value_t.cpp index dfcf0c3b..246243c1 100644 --- a/doc/examples/operator__value_t.cpp +++ b/doc/examples/operator__value_t.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/operator__value_t.link b/doc/examples/operator__value_t.link index 2acdc711..94d67cfd 100644 --- a/doc/examples/operator__value_t.link +++ b/doc/examples/operator__value_t.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/operator_deserialize.cpp b/doc/examples/operator_deserialize.cpp index bf9d7795..06656b65 100644 --- a/doc/examples/operator_deserialize.cpp +++ b/doc/examples/operator_deserialize.cpp @@ -1,5 +1,6 @@ -#include "json.hpp" +#include #include // for std::setw +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/operator_deserialize.link b/doc/examples/operator_deserialize.link index caf6025a..4544a65b 100644 --- a/doc/examples/operator_deserialize.link +++ b/doc/examples/operator_deserialize.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/operator_serialize.cpp b/doc/examples/operator_serialize.cpp index 1d817816..d808e869 100644 --- a/doc/examples/operator_serialize.cpp +++ b/doc/examples/operator_serialize.cpp @@ -1,5 +1,6 @@ -#include "json.hpp" +#include #include // for std::setw +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/operator_serialize.link b/doc/examples/operator_serialize.link index 104ce0fd..9d5d0102 100644 --- a/doc/examples/operator_serialize.link +++ b/doc/examples/operator_serialize.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/operatorarray__key_type.cpp b/doc/examples/operatorarray__key_type.cpp index 2438f991..a42335a3 100644 --- a/doc/examples/operatorarray__key_type.cpp +++ b/doc/examples/operatorarray__key_type.cpp @@ -1,5 +1,6 @@ -#include "json.hpp" +#include #include // for std::setw +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/operatorarray__key_type.link b/doc/examples/operatorarray__key_type.link index 7990937b..82961705 100644 --- a/doc/examples/operatorarray__key_type.link +++ b/doc/examples/operatorarray__key_type.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/operatorarray__key_type_const.cpp b/doc/examples/operatorarray__key_type_const.cpp index 36aa1db2..2320e439 100644 --- a/doc/examples/operatorarray__key_type_const.cpp +++ b/doc/examples/operatorarray__key_type_const.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/operatorarray__key_type_const.link b/doc/examples/operatorarray__key_type_const.link index 3b6f2f4a..53a08924 100644 --- a/doc/examples/operatorarray__key_type_const.link +++ b/doc/examples/operatorarray__key_type_const.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/operatorarray__size_type.cpp b/doc/examples/operatorarray__size_type.cpp index 27d7d9ef..bc8ecb96 100644 --- a/doc/examples/operatorarray__size_type.cpp +++ b/doc/examples/operatorarray__size_type.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/operatorarray__size_type.link b/doc/examples/operatorarray__size_type.link index e978214e..3e05678b 100644 --- a/doc/examples/operatorarray__size_type.link +++ b/doc/examples/operatorarray__size_type.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/operatorarray__size_type_const.cpp b/doc/examples/operatorarray__size_type_const.cpp index 44264172..ec9fc30f 100644 --- a/doc/examples/operatorarray__size_type_const.cpp +++ b/doc/examples/operatorarray__size_type_const.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/operatorarray__size_type_const.link b/doc/examples/operatorarray__size_type_const.link index 3dc0ce3e..2eb1a380 100644 --- a/doc/examples/operatorarray__size_type_const.link +++ b/doc/examples/operatorarray__size_type_const.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/operatorjson_pointer.cpp b/doc/examples/operatorjson_pointer.cpp index 63b233d2..d4036673 100644 --- a/doc/examples/operatorjson_pointer.cpp +++ b/doc/examples/operatorjson_pointer.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/operatorjson_pointer.link b/doc/examples/operatorjson_pointer.link index 67b3be46..92bf9d77 100644 --- a/doc/examples/operatorjson_pointer.link +++ b/doc/examples/operatorjson_pointer.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/operatorjson_pointer_const.cpp b/doc/examples/operatorjson_pointer_const.cpp index 8f3bc613..52b2a9e8 100644 --- a/doc/examples/operatorjson_pointer_const.cpp +++ b/doc/examples/operatorjson_pointer_const.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/operatorjson_pointer_const.link b/doc/examples/operatorjson_pointer_const.link index 38d54c6d..22a7c5cb 100644 --- a/doc/examples/operatorjson_pointer_const.link +++ b/doc/examples/operatorjson_pointer_const.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/parse__array__parser_callback_t.cpp b/doc/examples/parse__array__parser_callback_t.cpp index 790cc21d..5681178d 100644 --- a/doc/examples/parse__array__parser_callback_t.cpp +++ b/doc/examples/parse__array__parser_callback_t.cpp @@ -1,5 +1,6 @@ -#include "json.hpp" +#include #include // for std::setw +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/parse__array__parser_callback_t.link b/doc/examples/parse__array__parser_callback_t.link index 76e50355..0830fdac 100644 --- a/doc/examples/parse__array__parser_callback_t.link +++ b/doc/examples/parse__array__parser_callback_t.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/parse__contiguouscontainer__parser_callback_t.cpp b/doc/examples/parse__contiguouscontainer__parser_callback_t.cpp index c953fe67..ab551b0c 100644 --- a/doc/examples/parse__contiguouscontainer__parser_callback_t.cpp +++ b/doc/examples/parse__contiguouscontainer__parser_callback_t.cpp @@ -1,5 +1,6 @@ -#include "json.hpp" +#include #include // for std::setw +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/parse__contiguouscontainer__parser_callback_t.link b/doc/examples/parse__contiguouscontainer__parser_callback_t.link index 27f6f961..1e65c1f0 100644 --- a/doc/examples/parse__contiguouscontainer__parser_callback_t.link +++ b/doc/examples/parse__contiguouscontainer__parser_callback_t.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/parse__istream__parser_callback_t.cpp b/doc/examples/parse__istream__parser_callback_t.cpp index 1d25f705..562ab968 100644 --- a/doc/examples/parse__istream__parser_callback_t.cpp +++ b/doc/examples/parse__istream__parser_callback_t.cpp @@ -1,5 +1,6 @@ -#include "json.hpp" +#include #include // for std::setw +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/parse__istream__parser_callback_t.link b/doc/examples/parse__istream__parser_callback_t.link index f1552bf6..737c8dda 100644 --- a/doc/examples/parse__istream__parser_callback_t.link +++ b/doc/examples/parse__istream__parser_callback_t.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/parse__iteratortype__parser_callback_t.cpp b/doc/examples/parse__iteratortype__parser_callback_t.cpp index 1d752052..9a3e12e4 100644 --- a/doc/examples/parse__iteratortype__parser_callback_t.cpp +++ b/doc/examples/parse__iteratortype__parser_callback_t.cpp @@ -1,5 +1,6 @@ -#include "json.hpp" +#include #include // for std::setw +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/parse__iteratortype__parser_callback_t.link b/doc/examples/parse__iteratortype__parser_callback_t.link index f48f2fd1..e49ebc8d 100644 --- a/doc/examples/parse__iteratortype__parser_callback_t.link +++ b/doc/examples/parse__iteratortype__parser_callback_t.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/parse__string__parser_callback_t.cpp b/doc/examples/parse__string__parser_callback_t.cpp index b044eb32..ea02c330 100644 --- a/doc/examples/parse__string__parser_callback_t.cpp +++ b/doc/examples/parse__string__parser_callback_t.cpp @@ -1,5 +1,6 @@ -#include "json.hpp" +#include #include // for std::setw +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/parse__string__parser_callback_t.link b/doc/examples/parse__string__parser_callback_t.link index 3b44d9dc..535ecd2b 100644 --- a/doc/examples/parse__string__parser_callback_t.link +++ b/doc/examples/parse__string__parser_callback_t.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/patch.cpp b/doc/examples/patch.cpp index ad30b78b..c844b970 100644 --- a/doc/examples/patch.cpp +++ b/doc/examples/patch.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" #include // for std::setw diff --git a/doc/examples/patch.link b/doc/examples/patch.link index 93997149..7844cd94 100644 --- a/doc/examples/patch.link +++ b/doc/examples/patch.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/push_back.cpp b/doc/examples/push_back.cpp index 8d1b1c5e..7b0ec488 100644 --- a/doc/examples/push_back.cpp +++ b/doc/examples/push_back.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/push_back.link b/doc/examples/push_back.link index efb1b6bd..dc2b525e 100644 --- a/doc/examples/push_back.link +++ b/doc/examples/push_back.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/push_back__initializer_list.cpp b/doc/examples/push_back__initializer_list.cpp index 443c3639..d231e710 100644 --- a/doc/examples/push_back__initializer_list.cpp +++ b/doc/examples/push_back__initializer_list.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/push_back__initializer_list.link b/doc/examples/push_back__initializer_list.link index bb56fb16..f6796c25 100644 --- a/doc/examples/push_back__initializer_list.link +++ b/doc/examples/push_back__initializer_list.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/push_back__object_t__value.cpp b/doc/examples/push_back__object_t__value.cpp index fe3318c0..27215fb8 100644 --- a/doc/examples/push_back__object_t__value.cpp +++ b/doc/examples/push_back__object_t__value.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/push_back__object_t__value.link b/doc/examples/push_back__object_t__value.link index 0fad4a34..375315e4 100644 --- a/doc/examples/push_back__object_t__value.link +++ b/doc/examples/push_back__object_t__value.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/rbegin.cpp b/doc/examples/rbegin.cpp index c4f632eb..2e9824b4 100644 --- a/doc/examples/rbegin.cpp +++ b/doc/examples/rbegin.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/rbegin.link b/doc/examples/rbegin.link index a780148a..34174218 100644 --- a/doc/examples/rbegin.link +++ b/doc/examples/rbegin.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/rend.cpp b/doc/examples/rend.cpp index d746ab4f..0c81da12 100644 --- a/doc/examples/rend.cpp +++ b/doc/examples/rend.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/rend.link b/doc/examples/rend.link index 9d33126d..7c9360bd 100644 --- a/doc/examples/rend.link +++ b/doc/examples/rend.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/size.cpp b/doc/examples/size.cpp index 34186499..07d1b6cd 100644 --- a/doc/examples/size.cpp +++ b/doc/examples/size.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/size.link b/doc/examples/size.link index 6b27e7b9..9042e08e 100644 --- a/doc/examples/size.link +++ b/doc/examples/size.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/swap__array_t.cpp b/doc/examples/swap__array_t.cpp index b58c02c4..c83556ee 100644 --- a/doc/examples/swap__array_t.cpp +++ b/doc/examples/swap__array_t.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/swap__array_t.link b/doc/examples/swap__array_t.link index ed7862c8..7562e784 100644 --- a/doc/examples/swap__array_t.link +++ b/doc/examples/swap__array_t.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/swap__object_t.cpp b/doc/examples/swap__object_t.cpp index 15f061de..d67801c4 100644 --- a/doc/examples/swap__object_t.cpp +++ b/doc/examples/swap__object_t.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/swap__object_t.link b/doc/examples/swap__object_t.link index 6dd2dc18..cc6a18bb 100644 --- a/doc/examples/swap__object_t.link +++ b/doc/examples/swap__object_t.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/swap__reference.cpp b/doc/examples/swap__reference.cpp index ee033651..5223950c 100644 --- a/doc/examples/swap__reference.cpp +++ b/doc/examples/swap__reference.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/swap__reference.link b/doc/examples/swap__reference.link index 6bb80015..8c906769 100644 --- a/doc/examples/swap__reference.link +++ b/doc/examples/swap__reference.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/swap__string_t.cpp b/doc/examples/swap__string_t.cpp index 87ee1ed8..aeea028b 100644 --- a/doc/examples/swap__string_t.cpp +++ b/doc/examples/swap__string_t.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/swap__string_t.link b/doc/examples/swap__string_t.link index 5d204f86..29c2f9d0 100644 --- a/doc/examples/swap__string_t.link +++ b/doc/examples/swap__string_t.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/to_cbor.cpp b/doc/examples/to_cbor.cpp index a4fbfbf1..bdffe5eb 100644 --- a/doc/examples/to_cbor.cpp +++ b/doc/examples/to_cbor.cpp @@ -1,5 +1,6 @@ -#include "json.hpp" +#include #include // for std::setw +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/to_cbor.link b/doc/examples/to_cbor.link index 06695fec..ab658253 100644 --- a/doc/examples/to_cbor.link +++ b/doc/examples/to_cbor.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/to_msgpack.cpp b/doc/examples/to_msgpack.cpp index 5e0d1884..7756a9e6 100644 --- a/doc/examples/to_msgpack.cpp +++ b/doc/examples/to_msgpack.cpp @@ -1,5 +1,6 @@ -#include "json.hpp" +#include #include // for std::setw +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/to_msgpack.link b/doc/examples/to_msgpack.link index 904a6540..12014fb0 100644 --- a/doc/examples/to_msgpack.link +++ b/doc/examples/to_msgpack.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/type.cpp b/doc/examples/type.cpp index eb764bf2..b712a852 100644 --- a/doc/examples/type.cpp +++ b/doc/examples/type.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/type.link b/doc/examples/type.link index c4d92739..3fedc680 100644 --- a/doc/examples/type.link +++ b/doc/examples/type.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/type_name.cpp b/doc/examples/type_name.cpp index b849c10a..470d0a65 100644 --- a/doc/examples/type_name.cpp +++ b/doc/examples/type_name.cpp @@ -1,3 +1,4 @@ +#include #include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/type_name.link b/doc/examples/type_name.link index 8f36e649..7f50a398 100644 --- a/doc/examples/type_name.link +++ b/doc/examples/type_name.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file diff --git a/doc/examples/unflatten.cpp b/doc/examples/unflatten.cpp index f17e10ce..6318f859 100644 --- a/doc/examples/unflatten.cpp +++ b/doc/examples/unflatten.cpp @@ -1,5 +1,6 @@ -#include "json.hpp" +#include #include // for std::setw +#include "json.hpp" using json = nlohmann::json; diff --git a/doc/examples/unflatten.link b/doc/examples/unflatten.link index 71d3d55d..6caa6a57 100644 --- a/doc/examples/unflatten.link +++ b/doc/examples/unflatten.link @@ -1 +1 @@ -online \ No newline at end of file +online \ No newline at end of file From d3496347fcd1382896fca3aaf78a0d803c2f52ec Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 9 Jul 2017 12:19:14 +0200 Subject: [PATCH 284/530] :construction_worker: removing Clang 3.9 and 4.0 These versions are still not working properly: https://travis-ci.org/nlohmann/json/jobs/251675281 --- .travis.yml | 26 +++++--------------------- 1 file changed, 5 insertions(+), 21 deletions(-) diff --git a/.travis.yml b/.travis.yml index ec1cbeb1..530ef434 100644 --- a/.travis.yml +++ b/.travis.yml @@ -183,17 +183,17 @@ matrix: - os: linux compiler: gcc - env: - - COMPILER=g++-6 - - CXXFLAGS=-std=c++1z + env: COMPILER=g++-7 addons: apt: sources: ['ubuntu-toolchain-r-test'] - packages: g++-6 + packages: g++-7 - os: linux compiler: gcc - env: COMPILER=g++-7 + env: + - COMPILER=g++-7 + - CXXFLAGS=-std=c++17 addons: apt: sources: ['ubuntu-toolchain-r-test'] @@ -229,22 +229,6 @@ matrix: env: LLVM_VERSION=3.8.1 compiler: clang - - os: linux - compiler: clang - env: COMPILER=clang++-3.9 - addons: - apt: - sources: ['ubuntu-toolchain-r-test', 'llvm-toolchain-trusty-3.9'] - packages: clang-3.9 - - - os: linux - compiler: clang - env: COMPILER=clang++-4.0 - addons: - apt: - sources: ['ubuntu-toolchain-r-test', 'llvm-toolchain-trusty-4.0'] - packages: clang-4.0 - # - os: linux # addons: # apt: From fe086d7432191fbc9f9d3d8adb4766e291f19d50 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Th=C3=A9o=20DELRIEU?= Date: Sat, 1 Jul 2017 19:44:05 +0200 Subject: [PATCH 285/530] forward declare basic_json, add two macros Those macros are used to reduce template argument boilerplate --- src/json.hpp | 53 +++++++++++++++++++++++++++++++++++++--------------- 1 file changed, 38 insertions(+), 15 deletions(-) diff --git a/src/json.hpp b/src/json.hpp index 1bde08ce..3c615451 100644 --- a/src/json.hpp +++ b/src/json.hpp @@ -115,6 +115,39 @@ SOFTWARE. */ namespace nlohmann { +template +struct adl_serializer; + +// forward declaration of basic_json (required to split the class) +template