diff options
Diffstat (limited to 'json4cpp/tests/src/unit-convenience.cpp')
| -rw-r--r-- | json4cpp/tests/src/unit-convenience.cpp | 205 |
1 files changed, 205 insertions, 0 deletions
diff --git a/json4cpp/tests/src/unit-convenience.cpp b/json4cpp/tests/src/unit-convenience.cpp new file mode 100644 index 0000000000..f5104855f0 --- /dev/null +++ b/json4cpp/tests/src/unit-convenience.cpp @@ -0,0 +1,205 @@ +// __ _____ _____ _____ +// __| | __| | | | JSON for Modern C++ (supporting code) +// | | |__ | | | | | | version 3.12.0 +// |_____|_____|_____|_|___| https://github.com/nlohmann/json +// +// SPDX-FileCopyrightText: 2013-2026 Niels Lohmann <https://nlohmann.me> +// SPDX-License-Identifier: MIT + +#include "doctest_compatibility.h" + +#define JSON_TESTS_PRIVATE +#include <nlohmann/json.hpp> +using nlohmann::json; + +#include <sstream> + +namespace +{ +struct alt_string_iter +{ + alt_string_iter() = default; + alt_string_iter(const char* cstr) + : impl(cstr) + {} + + void reserve(std::size_t s) + { + impl.reserve(s); + } + + template<typename Iter> + void append(Iter first, Iter last) + { + impl.append(first, last); + } + + std::string::const_iterator begin() const noexcept + { + return impl.begin(); + } + + std::string::const_iterator end() const noexcept + { + return impl.end(); + } + + std::size_t size() const noexcept + { + return impl.size(); + } + + alt_string_iter& operator+=(const char c) + { + impl += c; + return *this; + } + + std::string impl{}; // NOLINT(readability-redundant-member-init) +}; + +struct alt_string_data +{ + alt_string_data() = default; + alt_string_data(const char* cstr) + : impl(cstr) + {} + + void reserve(std::size_t s) + { + impl.reserve(s); + } + + void append(const char* p, std::size_t s) + { + impl.append(p, s); + } + + const char* data() const + { + return impl.data(); + } + + std::size_t size() const + { + return impl.size(); + } + + alt_string_data& operator+=(const char c) + { + impl += c; + return *this; + } + + std::string impl{}; // NOLINT(readability-redundant-member-init) +}; + +void check_escaped(const char* original, const char* escaped = "", bool ensure_ascii = false); +void check_escaped(const char* original, const char* escaped, const bool ensure_ascii) +{ + std::stringstream ss; + json::serializer s(nlohmann::detail::output_adapter<char>(ss), ' '); + s.dump_escaped(original, ensure_ascii); + CHECK(ss.str() == escaped); +} +} // namespace + +TEST_CASE("convenience functions") +{ + SECTION("type name as string") + { + CHECK(std::string(json(json::value_t::null).type_name()) == "null"); + CHECK(std::string(json(json::value_t::object).type_name()) == "object"); + CHECK(std::string(json(json::value_t::array).type_name()) == "array"); + CHECK(std::string(json(json::value_t::number_integer).type_name()) == "number"); + CHECK(std::string(json(json::value_t::number_unsigned).type_name()) == "number"); + CHECK(std::string(json(json::value_t::number_float).type_name()) == "number"); + CHECK(std::string(json(json::value_t::binary).type_name()) == "binary"); + CHECK(std::string(json(json::value_t::boolean).type_name()) == "boolean"); + CHECK(std::string(json(json::value_t::string).type_name()) == "string"); + CHECK(std::string(json(json::value_t::discarded).type_name()) == "discarded"); + } + + SECTION("string escape") + { + 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"); + + // invalid UTF-8 characters + CHECK_THROWS_WITH_AS(check_escaped("ä\xA9ü"), "[json.exception.type_error.316] invalid UTF-8 byte at index 2: 0xA9", json::type_error&); + + CHECK_THROWS_WITH_AS(check_escaped("\xC2"), "[json.exception.type_error.316] incomplete UTF-8 string; last byte: 0xC2", json::type_error&); + } + + SECTION("string concat") + { + using nlohmann::detail::concat; + + const char* expected = "Hello, world!"; + alt_string_iter const hello_iter{"Hello, "}; + alt_string_data const hello_data{"Hello, "}; + std::string const world = "world"; + + SECTION("std::string") + { + const std::string str1 = concat(hello_iter, world, '!'); + const std::string str2 = concat(hello_data, world, '!'); + const std::string str3 = concat("Hello, ", world, '!'); + + CHECK(str1 == expected); + CHECK(str2 == expected); + CHECK(str3 == expected); + } + + SECTION("alt_string_iter") + { + const alt_string_iter str = concat<alt_string_iter>(hello_iter, world, '!'); + + CHECK(str.impl == expected); + } + + SECTION("alt_string_data") + { + const alt_string_data str = concat<alt_string_data>(hello_data, world, '!'); + + CHECK(str.impl == expected); + } + } +} |
