1 /* 2 __ _____ _____ _____ 3 __| | __| | | | JSON for Modern C++ (test suite) 4 | | |__ | | | | | | version 3.10.0 5 |_____|_____|_____|_|___| https://github.com/nlohmann/json 6 7 Licensed under the MIT License <http://opensource.org/licenses/MIT>. 8 SPDX-License-Identifier: MIT 9 Copyright (c) 2013-2019 Niels Lohmann <http://nlohmann.me>. 10 11 Permission is hereby granted, free of charge, to any person obtaining a copy 12 of this software and associated documentation files (the "Software"), to deal 13 in the Software without restriction, including without limitation the rights 14 to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 15 copies of the Software, and to permit persons to whom the Software is 16 furnished to do so, subject to the following conditions: 17 18 The above copyright notice and this permission notice shall be included in all 19 copies or substantial portions of the Software. 20 21 THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 22 IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 23 FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 24 AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 25 LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 26 OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 27 SOFTWARE. 28 */ 29 30 #include "doctest_compatibility.h" 31 32 #include <nlohmann/json.hpp> 33 using nlohmann::json; 34 35 #include <fstream> 36 #include <sstream> 37 #include <test_data.hpp> 38 39 TEST_CASE("object inspection") 40 { 41 SECTION("convenience type checker") 42 { 43 SECTION("object") 44 { 45 json j {{"foo", 1}, {"bar", false}}; 46 CHECK(!j.is_null()); 47 CHECK(!j.is_boolean()); 48 CHECK(!j.is_number()); 49 CHECK(!j.is_number_integer()); 50 CHECK(!j.is_number_unsigned()); 51 CHECK(!j.is_number_float()); 52 CHECK(!j.is_binary()); 53 CHECK(j.is_object()); 54 CHECK(!j.is_array()); 55 CHECK(!j.is_string()); 56 CHECK(!j.is_discarded()); 57 CHECK(!j.is_primitive()); 58 CHECK(j.is_structured()); 59 } 60 61 SECTION("array") 62 { 63 json j {"foo", 1, 1u, 42.23, false}; 64 CHECK(!j.is_null()); 65 CHECK(!j.is_boolean()); 66 CHECK(!j.is_number()); 67 CHECK(!j.is_number_integer()); 68 CHECK(!j.is_number_unsigned()); 69 CHECK(!j.is_number_float()); 70 CHECK(!j.is_binary()); 71 CHECK(!j.is_object()); 72 CHECK(j.is_array()); 73 CHECK(!j.is_string()); 74 CHECK(!j.is_discarded()); 75 CHECK(!j.is_primitive()); 76 CHECK(j.is_structured()); 77 } 78 79 SECTION("null") 80 { 81 json j(nullptr); 82 CHECK(j.is_null()); 83 CHECK(!j.is_boolean()); 84 CHECK(!j.is_number()); 85 CHECK(!j.is_number_integer()); 86 CHECK(!j.is_number_unsigned()); 87 CHECK(!j.is_number_float()); 88 CHECK(!j.is_binary()); 89 CHECK(!j.is_object()); 90 CHECK(!j.is_array()); 91 CHECK(!j.is_string()); 92 CHECK(!j.is_discarded()); 93 CHECK(j.is_primitive()); 94 CHECK(!j.is_structured()); 95 } 96 97 SECTION("boolean") 98 { 99 json j(true); 100 CHECK(!j.is_null()); 101 CHECK(j.is_boolean()); 102 CHECK(!j.is_number()); 103 CHECK(!j.is_number_integer()); 104 CHECK(!j.is_number_unsigned()); 105 CHECK(!j.is_number_float()); 106 CHECK(!j.is_binary()); 107 CHECK(!j.is_object()); 108 CHECK(!j.is_array()); 109 CHECK(!j.is_string()); 110 CHECK(!j.is_discarded()); 111 CHECK(j.is_primitive()); 112 CHECK(!j.is_structured()); 113 } 114 115 SECTION("string") 116 { 117 json j("Hello world"); 118 CHECK(!j.is_null()); 119 CHECK(!j.is_boolean()); 120 CHECK(!j.is_number()); 121 CHECK(!j.is_number_integer()); 122 CHECK(!j.is_number_unsigned()); 123 CHECK(!j.is_number_float()); 124 CHECK(!j.is_binary()); 125 CHECK(!j.is_object()); 126 CHECK(!j.is_array()); 127 CHECK(j.is_string()); 128 CHECK(!j.is_discarded()); 129 CHECK(j.is_primitive()); 130 CHECK(!j.is_structured()); 131 } 132 133 SECTION("number (integer)") 134 { 135 json j(42); 136 CHECK(!j.is_null()); 137 CHECK(!j.is_boolean()); 138 CHECK(j.is_number()); 139 CHECK(j.is_number_integer()); 140 CHECK(!j.is_number_unsigned()); 141 CHECK(!j.is_number_float()); 142 CHECK(!j.is_binary()); 143 CHECK(!j.is_object()); 144 CHECK(!j.is_array()); 145 CHECK(!j.is_string()); 146 CHECK(!j.is_discarded()); 147 CHECK(j.is_primitive()); 148 CHECK(!j.is_structured()); 149 } 150 151 SECTION("number (unsigned)") 152 { 153 json j(42u); 154 CHECK(!j.is_null()); 155 CHECK(!j.is_boolean()); 156 CHECK(j.is_number()); 157 CHECK(j.is_number_integer()); 158 CHECK(j.is_number_unsigned()); 159 CHECK(!j.is_number_float()); 160 CHECK(!j.is_binary()); 161 CHECK(!j.is_object()); 162 CHECK(!j.is_array()); 163 CHECK(!j.is_string()); 164 CHECK(!j.is_discarded()); 165 CHECK(j.is_primitive()); 166 CHECK(!j.is_structured()); 167 } 168 169 SECTION("number (floating-point)") 170 { 171 json j(42.23); 172 CHECK(!j.is_null()); 173 CHECK(!j.is_boolean()); 174 CHECK(j.is_number()); 175 CHECK(!j.is_number_integer()); 176 CHECK(!j.is_number_unsigned()); 177 CHECK(j.is_number_float()); 178 CHECK(!j.is_binary()); 179 CHECK(!j.is_object()); 180 CHECK(!j.is_array()); 181 CHECK(!j.is_string()); 182 CHECK(!j.is_discarded()); 183 CHECK(j.is_primitive()); 184 CHECK(!j.is_structured()); 185 } 186 187 SECTION("binary") 188 { 189 json j(json::value_t::binary); 190 CHECK(!j.is_null()); 191 CHECK(!j.is_boolean()); 192 CHECK(!j.is_number()); 193 CHECK(!j.is_number_integer()); 194 CHECK(!j.is_number_unsigned()); 195 CHECK(!j.is_number_float()); 196 CHECK(j.is_binary()); 197 CHECK(!j.is_object()); 198 CHECK(!j.is_array()); 199 CHECK(!j.is_string()); 200 CHECK(!j.is_discarded()); 201 CHECK(j.is_primitive()); 202 CHECK(!j.is_structured()); 203 } 204 205 SECTION("discarded") 206 { 207 json j(json::value_t::discarded); 208 CHECK(!j.is_null()); 209 CHECK(!j.is_boolean()); 210 CHECK(!j.is_number()); 211 CHECK(!j.is_number_integer()); 212 CHECK(!j.is_number_unsigned()); 213 CHECK(!j.is_number_float()); 214 CHECK(!j.is_binary()); 215 CHECK(!j.is_object()); 216 CHECK(!j.is_array()); 217 CHECK(!j.is_string()); 218 CHECK(j.is_discarded()); 219 CHECK(!j.is_primitive()); 220 CHECK(!j.is_structured()); 221 } 222 } 223 224 SECTION("serialization") 225 { 226 json j {{"object", json::object()}, {"array", {1, 2, 3, 4}}, {"number", 42}, {"boolean", false}, {"null", nullptr}, {"string", "Hello world"} }; 227 228 SECTION("no indent / indent=-1") 229 { 230 CHECK(j.dump() == 231 "{\"array\":[1,2,3,4],\"boolean\":false,\"null\":null,\"number\":42,\"object\":{},\"string\":\"Hello world\"}"); 232 233 CHECK(j.dump() == j.dump(-1)); 234 } 235 236 SECTION("indent=0") 237 { 238 CHECK(j.dump(0) == 239 "{\n\"array\": [\n1,\n2,\n3,\n4\n],\n\"boolean\": false,\n\"null\": null,\n\"number\": 42,\n\"object\": {},\n\"string\": \"Hello world\"\n}"); 240 } 241 242 SECTION("indent=1, space='\t'") 243 { 244 CHECK(j.dump(1, '\t') == 245 "{\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}"); 246 } 247 248 SECTION("indent=4") 249 { 250 CHECK(j.dump(4) == 251 "{\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}"); 252 } 253 254 SECTION("indent=x") 255 { 256 CHECK(j.dump().size() == 94); 257 CHECK(j.dump(1).size() == 127); 258 CHECK(j.dump(2).size() == 142); 259 CHECK(j.dump(512).size() == 7792); 260 261 // important test, because it yields a resize of the indent_string 262 // inside the dump() function 263 CHECK(j.dump(1024).size() == 15472); 264 265 const auto binary = json::binary({1, 2, 3}, 128); 266 CHECK(binary.dump(1024).size() == 2086); 267 } 268 269 SECTION("dump and floating-point numbers") 270 { 271 auto s = json(42.23).dump(); 272 CHECK(s.find("42.23") != std::string::npos); 273 } 274 275 SECTION("dump and small floating-point numbers") 276 { 277 auto s = json(1.23456e-78).dump(); 278 CHECK(s.find("1.23456e-78") != std::string::npos); 279 } 280 281 SECTION("dump and non-ASCII characters") 282 { 283 CHECK(json("ä").dump() == "\"ä\""); 284 CHECK(json("Ö").dump() == "\"Ö\""); 285 CHECK(json("❤️").dump() == "\"❤️\""); 286 } 287 288 SECTION("dump with ensure_ascii and non-ASCII characters") 289 { 290 CHECK(json("ä").dump(-1, ' ', true) == "\"\\u00e4\""); 291 CHECK(json("Ö").dump(-1, ' ', true) == "\"\\u00d6\""); 292 CHECK(json("❤️").dump(-1, ' ', true) == "\"\\u2764\\ufe0f\""); 293 } 294 295 SECTION("full Unicode escaping to ASCII") 296 { 297 SECTION("parsing yields the same JSON value") 298 { 299 std::ifstream f_escaped(TEST_DATA_DIRECTORY "/json_nlohmann_tests/all_unicode_ascii.json"); 300 std::ifstream f_unescaped(TEST_DATA_DIRECTORY "/json_nlohmann_tests/all_unicode.json"); 301 302 json j1 = json::parse(f_escaped); 303 json j2 = json::parse(f_unescaped); 304 CHECK(j1 == j2); 305 } 306 307 SECTION("dumping yields the same JSON text") 308 { 309 std::ifstream f_escaped(TEST_DATA_DIRECTORY "/json_nlohmann_tests/all_unicode_ascii.json"); 310 std::ifstream f_unescaped(TEST_DATA_DIRECTORY "/json_nlohmann_tests/all_unicode.json"); 311 312 json value = json::parse(f_unescaped); 313 std::string text = value.dump(4, ' ', true); 314 315 std::string expected((std::istreambuf_iterator<char>(f_escaped)), 316 std::istreambuf_iterator<char>()); 317 CHECK(text == expected); 318 } 319 } 320 321 SECTION("serialization of discarded element") 322 { 323 json j_discarded(json::value_t::discarded); 324 CHECK(j_discarded.dump() == "<discarded>"); 325 } 326 327 SECTION("check that precision is reset after serialization") 328 { 329 // create stringstream and set precision 330 std::stringstream ss; 331 ss.precision(3); 332 ss << 3.141592653589793 << std::fixed; 333 CHECK(ss.str() == "3.14"); 334 335 // reset stringstream 336 ss.str(std::string()); 337 338 // use stringstream for JSON serialization 339 json j_number = 3.14159265358979; 340 ss << j_number; 341 342 // check that precision has been overridden during serialization 343 CHECK(ss.str() == "3.14159265358979"); 344 345 // check that precision has been restored 346 CHECK(ss.precision() == 3); 347 } 348 } 349 350 SECTION("round trips") 351 { 352 for (const auto& s : 353 {"3.141592653589793", "1000000000000000010E5" 354 }) 355 { 356 json j1 = json::parse(s); 357 std::string s1 = j1.dump(); 358 json j2 = json::parse(s1); 359 std::string s2 = j2.dump(); 360 CHECK(s1 == s2); 361 } 362 } 363 364 SECTION("return the type of the object (explicit)") 365 { 366 SECTION("null") 367 { 368 json j = nullptr; 369 CHECK(j.type() == json::value_t::null); 370 } 371 372 SECTION("object") 373 { 374 json j = {{"foo", "bar"}}; 375 CHECK(j.type() == json::value_t::object); 376 } 377 378 SECTION("array") 379 { 380 json j = {1, 2, 3, 4}; 381 CHECK(j.type() == json::value_t::array); 382 } 383 384 SECTION("boolean") 385 { 386 json j = true; 387 CHECK(j.type() == json::value_t::boolean); 388 } 389 390 SECTION("string") 391 { 392 json j = "Hello world"; 393 CHECK(j.type() == json::value_t::string); 394 } 395 396 SECTION("number (integer)") 397 { 398 json j = 23; 399 CHECK(j.type() == json::value_t::number_integer); 400 } 401 402 SECTION("number (unsigned)") 403 { 404 json j = 23u; 405 CHECK(j.type() == json::value_t::number_unsigned); 406 } 407 408 SECTION("number (floating-point)") 409 { 410 json j = 42.23; 411 CHECK(j.type() == json::value_t::number_float); 412 } 413 } 414 415 SECTION("return the type of the object (implicit)") 416 { 417 SECTION("null") 418 { 419 json j = nullptr; 420 json::value_t t = j; 421 CHECK(t == j.type()); 422 } 423 424 SECTION("object") 425 { 426 json j = {{"foo", "bar"}}; 427 json::value_t t = j; 428 CHECK(t == j.type()); 429 } 430 431 SECTION("array") 432 { 433 json j = {1, 2, 3, 4}; 434 json::value_t t = j; 435 CHECK(t == j.type()); 436 } 437 438 SECTION("boolean") 439 { 440 json j = true; 441 json::value_t t = j; 442 CHECK(t == j.type()); 443 } 444 445 SECTION("string") 446 { 447 json j = "Hello world"; 448 json::value_t t = j; 449 CHECK(t == j.type()); 450 } 451 452 SECTION("number (integer)") 453 { 454 json j = 23; 455 json::value_t t = j; 456 CHECK(t == j.type()); 457 } 458 459 SECTION("number (unsigned)") 460 { 461 json j = 23u; 462 json::value_t t = j; 463 CHECK(t == j.type()); 464 } 465 466 SECTION("number (floating-point)") 467 { 468 json j = 42.23; 469 json::value_t t = j; 470 CHECK(t == j.type()); 471 } 472 473 SECTION("binary") 474 { 475 json j = json::binary({}); 476 json::value_t t = j; 477 CHECK(t == j.type()); 478 } 479 } 480 } 481