2016-10-16 23:29:57 +08:00
|
|
|
/*
|
|
|
|
__ _____ _____ _____
|
|
|
|
__| | __| | | | JSON for Modern C++ (test suite)
|
2016-11-17 03:49:24 +08:00
|
|
|
| | |__ | | | | | | version 2.0.7
|
2016-10-16 23:29:57 +08:00
|
|
|
|_____|_____|_____|_|___| https://github.com/nlohmann/json
|
|
|
|
|
|
|
|
Licensed under the MIT License <http://opensource.org/licenses/MIT>.
|
|
|
|
Copyright (c) 2013-2016 Niels Lohmann <http://nlohmann.me>.
|
|
|
|
|
|
|
|
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.
|
|
|
|
*/
|
|
|
|
|
2016-11-29 20:09:51 +08:00
|
|
|
#include <array>
|
2016-10-16 23:29:57 +08:00
|
|
|
#include <string>
|
|
|
|
#include <memory>
|
|
|
|
#include "catch.hpp"
|
|
|
|
|
|
|
|
#include "json.hpp"
|
|
|
|
|
|
|
|
namespace udt
|
|
|
|
{
|
2016-12-14 07:03:55 +08:00
|
|
|
enum class country
|
|
|
|
{
|
|
|
|
china,
|
|
|
|
france,
|
|
|
|
russia
|
|
|
|
};
|
|
|
|
|
2016-11-29 20:09:51 +08:00
|
|
|
struct age
|
2016-10-18 05:41:53 +08:00
|
|
|
{
|
2016-12-01 06:16:54 +08:00
|
|
|
int m_val;
|
2016-11-29 20:09:51 +08:00
|
|
|
};
|
2016-10-18 05:41:53 +08:00
|
|
|
|
2016-11-29 20:09:51 +08:00
|
|
|
struct name
|
2016-10-16 23:29:57 +08:00
|
|
|
{
|
2016-12-01 06:16:54 +08:00
|
|
|
std::string m_val;
|
2016-11-29 20:09:51 +08:00
|
|
|
};
|
2016-10-16 23:29:57 +08:00
|
|
|
|
2016-11-29 20:09:51 +08:00
|
|
|
struct address
|
2016-10-16 23:29:57 +08:00
|
|
|
{
|
2016-12-01 06:16:54 +08:00
|
|
|
std::string m_val;
|
2016-11-29 20:09:51 +08:00
|
|
|
};
|
2016-10-16 23:29:57 +08:00
|
|
|
|
2016-11-29 20:09:51 +08:00
|
|
|
struct person
|
2016-10-16 23:29:57 +08:00
|
|
|
{
|
2016-12-01 06:16:54 +08:00
|
|
|
age m_age;
|
|
|
|
name m_name;
|
2016-12-14 07:03:55 +08:00
|
|
|
country m_country;
|
2016-11-29 20:09:51 +08:00
|
|
|
};
|
2016-10-16 23:29:57 +08:00
|
|
|
|
2016-11-29 20:09:51 +08:00
|
|
|
struct contact
|
2016-10-16 23:29:57 +08:00
|
|
|
{
|
2016-12-01 06:16:54 +08:00
|
|
|
person m_person;
|
|
|
|
address m_address;
|
2016-11-29 20:09:51 +08:00
|
|
|
};
|
2016-10-16 23:29:57 +08:00
|
|
|
|
2016-11-29 20:09:51 +08:00
|
|
|
struct contact_book
|
2016-10-16 23:29:57 +08:00
|
|
|
{
|
2016-12-01 06:16:54 +08:00
|
|
|
name m_book_name;
|
|
|
|
std::vector<contact> m_contacts;
|
2016-11-29 20:09:51 +08:00
|
|
|
};
|
2016-10-16 23:29:57 +08:00
|
|
|
}
|
2016-10-18 05:41:53 +08:00
|
|
|
|
2016-12-14 07:03:55 +08:00
|
|
|
// to_json methods
|
2016-11-29 20:09:51 +08:00
|
|
|
namespace udt
|
2016-10-18 05:41:53 +08:00
|
|
|
{
|
2016-11-29 20:09:51 +08:00
|
|
|
void to_json(nlohmann::json& j, age a)
|
2016-10-18 05:41:53 +08:00
|
|
|
{
|
2016-12-01 06:16:54 +08:00
|
|
|
j = a.m_val;
|
2016-10-18 05:41:53 +08:00
|
|
|
}
|
|
|
|
|
2016-11-29 20:09:51 +08:00
|
|
|
void to_json(nlohmann::json& j, name const& n)
|
2016-10-18 05:41:53 +08:00
|
|
|
{
|
2016-12-01 06:16:54 +08:00
|
|
|
j = n.m_val;
|
2016-10-18 05:41:53 +08:00
|
|
|
}
|
|
|
|
|
2016-12-14 07:03:55 +08:00
|
|
|
void to_json(nlohmann::json& j, country c)
|
|
|
|
{
|
|
|
|
switch (c)
|
|
|
|
{
|
|
|
|
case country::china:
|
|
|
|
j = u8"中华人民共和国";
|
|
|
|
return;
|
|
|
|
case country::france:
|
|
|
|
j = "France";
|
|
|
|
return;
|
|
|
|
case country::russia:
|
|
|
|
j = u8"Российская Федерация";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-11-29 20:09:51 +08:00
|
|
|
void to_json(nlohmann::json& j, person const& p)
|
2016-10-18 05:41:53 +08:00
|
|
|
{
|
2016-11-29 20:09:51 +08:00
|
|
|
using nlohmann::json;
|
2016-12-14 07:03:55 +08:00
|
|
|
j = json{{"age", p.m_age}, {"name", p.m_name}, {"country", p.m_country}};
|
2016-10-18 05:41:53 +08:00
|
|
|
}
|
|
|
|
|
2016-11-29 20:09:51 +08:00
|
|
|
void to_json(nlohmann::json& j, address const& a)
|
2016-10-18 05:41:53 +08:00
|
|
|
{
|
2016-12-01 06:16:54 +08:00
|
|
|
j = a.m_val;
|
2016-10-18 05:41:53 +08:00
|
|
|
}
|
|
|
|
|
2016-11-29 20:09:51 +08:00
|
|
|
void to_json(nlohmann::json& j, contact const& c)
|
2016-10-18 05:41:53 +08:00
|
|
|
{
|
2016-11-29 20:09:51 +08:00
|
|
|
using nlohmann::json;
|
2016-12-05 00:52:55 +08:00
|
|
|
j = json{{"person", c.m_person}, {"address", c.m_address}};
|
2016-10-18 05:41:53 +08:00
|
|
|
}
|
|
|
|
|
2016-11-29 20:09:51 +08:00
|
|
|
void to_json(nlohmann::json& j, contact_book const& cb)
|
2016-10-18 05:41:53 +08:00
|
|
|
{
|
2016-11-29 20:09:51 +08:00
|
|
|
using nlohmann::json;
|
2016-12-05 00:52:55 +08:00
|
|
|
j = json{{"name", cb.m_book_name}, {"contacts", cb.m_contacts}};
|
2016-12-01 06:16:54 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// operators
|
|
|
|
bool operator==(age lhs, age rhs)
|
|
|
|
{
|
|
|
|
return lhs.m_val == rhs.m_val;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool operator==(address const &lhs, address const &rhs)
|
|
|
|
{
|
|
|
|
return lhs.m_val == rhs.m_val;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool operator==(name const &lhs, name const &rhs)
|
|
|
|
{
|
|
|
|
return lhs.m_val == rhs.m_val;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool operator==(person const &lhs, person const &rhs)
|
|
|
|
{
|
|
|
|
return std::tie(lhs.m_name, lhs.m_age) == std::tie(rhs.m_name, rhs.m_age);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool operator==(contact const &lhs, contact const &rhs)
|
|
|
|
{
|
|
|
|
return std::tie(lhs.m_person, lhs.m_address) ==
|
|
|
|
std::tie(rhs.m_person, rhs.m_address);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool operator==(contact_book const &lhs, contact_book const &rhs)
|
|
|
|
{
|
|
|
|
return std::tie(lhs.m_book_name, lhs.m_contacts) ==
|
|
|
|
std::tie(rhs.m_book_name, rhs.m_contacts);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-12-14 07:03:55 +08:00
|
|
|
// from_json methods
|
2016-12-01 06:16:54 +08:00
|
|
|
namespace udt
|
|
|
|
{
|
|
|
|
void from_json(nlohmann::json const& j, age &a)
|
|
|
|
{
|
|
|
|
a.m_val = j.get<int>();
|
|
|
|
}
|
|
|
|
|
|
|
|
void from_json(nlohmann::json const& j, name &n)
|
|
|
|
{
|
|
|
|
n.m_val = j.get<std::string>();
|
|
|
|
}
|
|
|
|
|
2016-12-14 07:03:55 +08:00
|
|
|
void from_json(nlohmann::json const &j, country &c)
|
|
|
|
{
|
|
|
|
const auto str = j.get<std::string>();
|
|
|
|
static const std::map<std::string, country> m = {
|
|
|
|
{u8"中华人民共和国", country::china},
|
|
|
|
{"France", country::france},
|
|
|
|
{"Российская Федерация", country::russia}};
|
|
|
|
|
|
|
|
const auto it = m.find(str);
|
|
|
|
// TODO test exceptions
|
|
|
|
c = it->second;
|
|
|
|
}
|
|
|
|
|
2016-12-01 06:16:54 +08:00
|
|
|
void from_json(nlohmann::json const& j, person &p)
|
|
|
|
{
|
|
|
|
p.m_age = j["age"].get<age>();
|
|
|
|
p.m_name = j["name"].get<name>();
|
2016-12-14 07:03:55 +08:00
|
|
|
p.m_country = j["country"].get<country>();
|
2016-12-01 06:16:54 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void from_json(nlohmann::json const &j, address &a)
|
|
|
|
{
|
|
|
|
a.m_val = j.get<std::string>();
|
|
|
|
}
|
|
|
|
|
|
|
|
void from_json(nlohmann::json const& j, contact &c)
|
|
|
|
{
|
|
|
|
c.m_person = j["person"].get<person>();
|
|
|
|
c.m_address = j["address"].get<address>();
|
|
|
|
}
|
|
|
|
|
|
|
|
void from_json(nlohmann::json const&j, contact_book &cb)
|
|
|
|
{
|
|
|
|
cb.m_book_name = j["name"].get<name>();
|
|
|
|
cb.m_contacts = j["contacts"].get<std::vector<contact>>();
|
2016-10-18 05:41:53 +08:00
|
|
|
}
|
|
|
|
}
|
2016-11-15 21:22:12 +08:00
|
|
|
|
2016-11-29 20:09:51 +08:00
|
|
|
TEST_CASE("basic usage", "[udt]")
|
2016-11-15 21:22:12 +08:00
|
|
|
{
|
2016-11-29 20:09:51 +08:00
|
|
|
using nlohmann::json;
|
2016-11-15 21:22:12 +08:00
|
|
|
|
2016-12-01 06:16:54 +08:00
|
|
|
// a bit narcissic maybe :) ?
|
|
|
|
const udt::age a{23};
|
|
|
|
const udt::name n{"theo"};
|
2016-12-14 07:03:55 +08:00
|
|
|
const udt::country c{udt::country::france};
|
|
|
|
const udt::person sfinae_addict{a, n, c};
|
|
|
|
const udt::person senior_programmer{{42}, {u8"王芳"}, udt::country::china};
|
2016-12-01 06:16:54 +08:00
|
|
|
const udt::address addr{"Paris"};
|
|
|
|
const udt::contact cpp_programmer{sfinae_addict, addr};
|
2016-12-14 07:03:55 +08:00
|
|
|
const udt::contact_book book{{"C++"}, {cpp_programmer, {senior_programmer, addr}}};
|
2016-12-01 06:16:54 +08:00
|
|
|
|
2016-11-29 20:09:51 +08:00
|
|
|
SECTION("conversion to json via free-functions")
|
2016-11-15 21:22:12 +08:00
|
|
|
{
|
2016-12-05 00:52:55 +08:00
|
|
|
CHECK(json(a) == json(23));
|
|
|
|
CHECK(json(n) == json("theo"));
|
2016-12-14 07:03:55 +08:00
|
|
|
CHECK(json(c) == json("France"));
|
|
|
|
CHECK(json(sfinae_addict) == R"({"name":"theo", "age":23, "country":"France"})"_json);
|
2016-12-05 00:52:55 +08:00
|
|
|
CHECK(json("Paris") == json(addr));
|
|
|
|
CHECK(json(cpp_programmer) ==
|
2016-12-14 07:03:55 +08:00
|
|
|
R"({"person" : {"age":23, "name":"theo", "country":"France"}, "address":"Paris"})"_json);
|
2016-11-15 21:22:12 +08:00
|
|
|
|
2016-12-01 06:16:54 +08:00
|
|
|
CHECK(
|
2016-12-05 00:52:55 +08:00
|
|
|
json(book) ==
|
2016-12-14 07:03:55 +08:00
|
|
|
R"({"name":"C++", "contacts" : [{"person" : {"age":23, "name":"theo", "country":"France"}, "address":"Paris"}, {"person" : {"age":42, "country":"中华人民共和国", "name":"王芳"}, "address":"Paris"}]})"_json);
|
|
|
|
|
2016-12-01 06:16:54 +08:00
|
|
|
}
|
2016-11-15 21:22:12 +08:00
|
|
|
|
2016-12-01 06:16:54 +08:00
|
|
|
SECTION("conversion from json via free-functions")
|
|
|
|
{
|
|
|
|
const auto big_json =
|
2016-12-14 07:03:55 +08:00
|
|
|
R"({"name":"C++", "contacts" : [{"person" : {"age":23, "name":"theo", "country":"France"}, "address":"Paris"}, {"person" : {"age":42, "country":"中华人民共和国", "name":"王芳"}, "address":"Paris"}]})"_json;
|
2016-12-01 06:16:54 +08:00
|
|
|
const auto parsed_book = big_json.get<udt::contact_book>();
|
|
|
|
const auto book_name = big_json["name"].get<udt::name>();
|
|
|
|
const auto contacts = big_json["contacts"].get<std::vector<udt::contact>>();
|
|
|
|
const auto contact_json = big_json["contacts"].at(0);
|
|
|
|
const auto contact = contact_json.get<udt::contact>();
|
|
|
|
const auto person = contact_json["person"].get<udt::person>();
|
|
|
|
const auto address = contact_json["address"].get<udt::address>();
|
|
|
|
const auto age = contact_json["person"]["age"].get<udt::age>();
|
2016-12-14 07:03:55 +08:00
|
|
|
const auto country = contact_json["person"]["country"].get<udt::country>();
|
2016-12-01 06:16:54 +08:00
|
|
|
const auto name = contact_json["person"]["name"].get<udt::name>();
|
|
|
|
|
|
|
|
CHECK(age == a);
|
|
|
|
CHECK(name == n);
|
2016-12-14 07:03:55 +08:00
|
|
|
CHECK(country == c);
|
2016-12-01 06:16:54 +08:00
|
|
|
CHECK(address == addr);
|
|
|
|
CHECK(person == sfinae_addict);
|
|
|
|
CHECK(contact == cpp_programmer);
|
|
|
|
CHECK(contacts == book.m_contacts);
|
|
|
|
CHECK(book_name == udt::name{"C++"});
|
|
|
|
CHECK(book == parsed_book);
|
2016-11-15 21:22:12 +08:00
|
|
|
}
|
2016-12-01 06:16:54 +08:00
|
|
|
}
|
2016-12-14 05:01:38 +08:00
|
|
|
|
|
|
|
namespace udt
|
|
|
|
{
|
|
|
|
template <typename T>
|
|
|
|
class optional_type
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
optional_type() = default;
|
|
|
|
optional_type(T t) { _impl = std::make_shared<T>(std::move(t)); }
|
|
|
|
optional_type(std::nullptr_t) { _impl = nullptr; }
|
|
|
|
|
|
|
|
optional_type &operator=(std::nullptr_t)
|
|
|
|
{
|
|
|
|
_impl = nullptr;
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
optional_type& operator=(T t)
|
|
|
|
{
|
|
|
|
_impl = std::make_shared<T>(std::move(t));
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
explicit operator bool() const noexcept { return _impl != nullptr; }
|
|
|
|
T const &operator*() const noexcept { return *_impl; }
|
|
|
|
T &operator*() noexcept { return *_impl; }
|
|
|
|
|
|
|
|
private:
|
|
|
|
std::shared_ptr<T> _impl;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct legacy_type
|
|
|
|
{
|
|
|
|
std::string number;
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace nlohmann
|
|
|
|
{
|
|
|
|
template <typename T>
|
|
|
|
struct adl_serializer<udt::optional_type<T>>
|
|
|
|
{
|
|
|
|
static void to_json(json& j, udt::optional_type<T> const& opt)
|
|
|
|
{
|
|
|
|
if (opt)
|
|
|
|
j = *opt;
|
|
|
|
else
|
|
|
|
j = nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void from_json(json const &j, udt::optional_type<T> &opt)
|
|
|
|
{
|
|
|
|
if (j.is_null())
|
|
|
|
opt = nullptr;
|
|
|
|
else
|
|
|
|
opt = j.get<T>();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
template <>
|
|
|
|
struct adl_serializer<udt::legacy_type>
|
|
|
|
{
|
|
|
|
static void to_json(json& j, udt::legacy_type const& l)
|
|
|
|
{
|
|
|
|
j = std::stoi(l.number);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void from_json(json const& j, udt::legacy_type& l)
|
|
|
|
{
|
|
|
|
l.number = std::to_string(j.get<int>());
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_CASE("adl_serializer specialization", "[udt]")
|
|
|
|
{
|
|
|
|
using nlohmann::json;
|
|
|
|
|
|
|
|
SECTION("partial specialization")
|
|
|
|
{
|
|
|
|
SECTION("to_json")
|
|
|
|
{
|
|
|
|
udt::optional_type<udt::person> optPerson;
|
|
|
|
|
|
|
|
json j = optPerson;
|
|
|
|
CHECK(j.is_null());
|
|
|
|
|
|
|
|
optPerson = udt::person{{42}, {"John Doe"}};
|
|
|
|
j = optPerson;
|
|
|
|
CHECK_FALSE(j.is_null());
|
|
|
|
|
|
|
|
CHECK(j.get<udt::person>() == *optPerson);
|
|
|
|
}
|
|
|
|
|
|
|
|
SECTION("from_json")
|
|
|
|
{
|
|
|
|
auto person = udt::person{{42}, {"John Doe"}};
|
|
|
|
json j = person;
|
|
|
|
|
|
|
|
auto optPerson = j.get<udt::optional_type<udt::person>>();
|
|
|
|
REQUIRE(optPerson);
|
|
|
|
CHECK(*optPerson == person);
|
|
|
|
|
|
|
|
j = nullptr;
|
|
|
|
optPerson = j.get<udt::optional_type<udt::person>>();
|
|
|
|
CHECK(!optPerson);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
SECTION("total specialization")
|
|
|
|
{
|
|
|
|
SECTION("to_json")
|
|
|
|
{
|
|
|
|
udt::legacy_type lt{"4242"};
|
|
|
|
|
|
|
|
json j = lt;
|
|
|
|
CHECK(j.get<int>() == 4242);
|
|
|
|
}
|
|
|
|
|
|
|
|
SECTION("from_json")
|
|
|
|
{
|
|
|
|
json j = 4242;
|
|
|
|
auto lt = j.get<udt::legacy_type>();
|
|
|
|
CHECK(lt.number == "4242");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|