-
Notifications
You must be signed in to change notification settings - Fork 6
6.0. RareJson
RareJson is a reflection-based JSON library geared toward reading and writing complex objects with minimal effort.
Any object to which the REFLECT macro has already been applied can be written to and read from streams as such:
std::cin >> Json::in(myObj);
std::cout << Json::out(myObj) << std::endl;
std::cout << Json::pretty(myObj) << std::endl;
This JSON library will automatically...
- Read or put each field and accompanying value
- Infer the appropriate JSON representation for various types
- Read or put any array, STL iterable, or nested STL iterable as a JSON array
- Read or put any map or STL iterable containg pairs, or nested maps/STL iterables as a JSON object with keys being the field names
- Read or put pairs or tuples as JSON arrays
- Read or put any nested reflected objects as nested JSON objects
- Ignore any unknown fields in an object or automatically read them into a Json::FieldCluster if present
There are a few annotations you can use to alter the JSON representation...
- Json::Ignore will cause a field to be ignored by json I/O
- Json::Stringify will result in the value given by the ostream operator being quoted and escaped as a JSON string
- Json::Unstring will demote an std::string type (which by default behaves like Json::String) so that it's put/read without quotes
- Json::EnumInt can be used to force use of the integer value for an enum field that may otherwise have iostream overloads
To use, simply include the reflect.h and json.h files in your project where needed, then REFLECT the objects you are going to be using and enjoy the awesome power of JSON!
struct MyObject
{
int myInt;
std::string myString;
std::vector<int> myIntCollection;
REFLECT(MyObject, myInt, myString, myIntCollection)
};
int main()
{
MyObject myObject = {};
std::cout << "Enter MyObject:" << std::endl;
std::cin >> Json::in(myObject);
std::cout << std::endl << std::endl << "You entered:" << std::endl;
std::cout << Json::pretty(myObject);
}
You entered:
{
"myInt": 1337,
"myString": "stringy",
"myIntCollection": [ 2, 4, 6 ]
}
While the primary focus of this library is dealing with objects and JSON structures known at compile time, it's not unheard of to have JSON structures that you'll only come to know at runtime, it's perhaps even more common to have a structure that you know at compile time but which includes many fields that you don't need to use except to receive them, perhaps audit them, and pass them along. This is where this libraries JSON generics can come in handy.
A Json::FieldCluster is for the latter case, where you know and use some fields in your program, but others you won't need to reference explicitly. To maximize performance and avoid misuse of field clusters, they may not be the first, nor may they be the only field in an object.
struct SomeStructure
{
int usefulField;
Json::FieldCluster fieldCluster;
REFLECT(RegularFields, usefulField, fieldCluster)
};
Now if you read in something like...
{
"usefulField": 222,
"seedValue": "0x12A39B",
"sequence": 5
}
The "seedValue" and "sequence" values would be automatically stored in the fieldCluster and will be streamed as if you had explicitly declared and reflected those fields, and can also, if for some reason necessary, be traversed programatically.
You can also take in an entirely unknown object using Json::Object, e.g.
Json::Object obj;
std::cin >> Json::in(obj);
All Generic types extend from Json::Value, FieldCluster extends Json::Object but has special I/O behavior and its own type. The following are the generic types...
- Json::Value (pure virtual class, check the real type with the type() method)
- Json::Bool (type: Json::Value::Type::Boolean)
- Json::Number (type: Json::Value::Type::Number)
- Json::String (type: Json::Value::Type::String)
- Json::Object (type: Json::Value::Type::Object)
- Json::NullArray (type: Json::Value::Type::NullArray)
- Json::BoolArray (type: Json::Value::Type::BoolArray)
- Json::NumberArray (type: Json::Value::Type::NumberArray
- Json::StringArray (type: Json::Value::Type::StringArray)
- Json::ObjectArray (type: Json::Value::Type::ObjectArray)
- Json::MixedArray (type: Json::Value::Type::MixedArray)
- Json::FieldCluster (type: Json::Value::Type::FieldCluster)
The null, bool, string, and object arrays are optimized for the type of contents they contain, whereas the elements in MixedArray are all shared pointers to Json::Values. Any type that contains nested arrays are automatically a MixedArray.
Json::Value has virtual methods for getting the contents of any given type, e.g. for a Json::Bool call the boolean() method to get the stored value, if it's not the correct type for that method a TypeMismatch exception is thrown, check the type of a Json::Value in advance to avoid that.
When you have a type which, for whatever reason, is not well-suited for reflection/directt serialization, providing a default mapping to another struct/class which is serialization friendly is an easy option. See RareMapper for how to create mappings and Default Mappings for details on setting a default mapping.
In addition to the annotations, you can further customize behavior for specific types, or specific fields within a type. This is accomplished using partial template specialization on one of the following structs...
- Json::Output::Customize<Object, Value, ... [optional parameters]>
- Json::Output::CustomizeType<Value, ... [optional parameters]>
- Json::Input::Customize<Object, Value, ... [optional parameters]>
- Json::Input::CustomizeType<Value, ... [optional parameters]>
You can use the optional parmeters to further specialize if say, you only wanted the customizer to only apply to a specific field index. The optional parameters can also help should you need to know additional details - such as any annotations applied to the field, whether pretty print is active, the current indentation level, etc. Customizers should usually be written in header (".h", or ".hpp") files, as you will be using template arguments which will be very limited if in a ".cpp" file.
If you had a class Point with fields latitude and longitude you might write a customizer like...
template <>
struct Json::Input::Customize<Point, RareTs::MemberType<Point>::latitude::type, RareTs::IndexOf<Point>::latitude>
{
static bool as(std::istream & is, Context &, const Point &, double & value)
{
is >> Json::in(value);
if ( value < -90 || value > 90 ) // Validating some input
throw std::logic_error("Latitudes must be between -90 and 90");
return true;
}
};
This particular customizer will only run for the latitude field (specialized for <Point, double, latitude_field_index>), but you can supply any number of parameters, and have this run for say, every field on Point (specialize for ), or every double on point (specialize for <Point, double>).
Returning true indicates you're done with this field. Returning false indicates you wish to go back and use default json input behavior for this field.
To help with customization you can add an arbitrary "Context" object to the operation, a context object is any struct or class that has Json::Context in its inheritance heirarchy, via this object you can supply additional information that you might need, such as an object that knows what string value certain integers should take on, or whether or not to convert everything to lowercase or uppercase, and so on.
struct EnhancedContext : public Json::Context
{
virtual ~EnhancedContext() {}
EnhancedContext(int enhanced) : enhanced(enhanced) {}
static std::shared_ptr<EnhancedContext> Make(int enhanced) {
return std::shared_ptr<EnhancedContext>(new EnhancedContext(enhanced));
}
int enhanced;
};
template <>
struct Json::Output::Customize<A, A::TestEnum>
{
static bool As(Json::OutStreamType & os, Context & context, const A & object, const A::TestEnum & value)
{
try {
EnhancedContext & enhanced = dynamic_cast<EnhancedContext &>(context);
switch ( value )
{
case A::TestEnum::first:
Json::Put::String(os, "firstCustom" + std::to_string(enhanced.enhanced));
return true;
case A::TestEnum::second:
Json::Put::String(os, "secondCustom" + std::to_string(enhanced.enhanced));
return true;
}
return true;
} catch ( std::bad_cast & ) {
return false;
}
}
};
std::cout << Json::pretty(a, EnhancedContext::Make(1337)) << std::endl;
There are three main ways you can handle enumerations...
- Have an enumeration with the Json::EnumInt annotation used, this will force use of the integer value regardless of ostream/istream overloads
- Have an enumeration with ostream/istream operator overloads and the Json::String annotation used, this will create a one-to-one relationship with a string value for output, and a one-to-one or one-to-many relationship with a string value for input
- Add a customizer for your enumeration, this allows you to customize the string representation based on the state of the object and which specific field the enum belongs to - this is especially useful when you have multiple enum strings that share the same value (perhaps because the meaning of the enum value changes based on some type field, or because one string makes more grammatical sense than another in different contexts.
(1.)
struct A {
enum class TestEnum : size_t {
first,
second
};
NOTE(testEnum, Json::EnumInt)
TestEnum testEnum;
REFLECT(A, testEnum)
};
(2.)
struct A {
enum class TestEnum : size_t {
first,
second
};
NOTE(testEnum, Json::Stringify)
TestEnum testEnum;
REFLECT(A, testEnum)
static const std::unordered_map<std::string, TestEnum> TestEnumCache;
};
const std::unordered_map<std::string, A::TestEnum> A::TestEnumCache = {
{ "first", A::TestEnum::first },
{ "second", A::TestEnum::second }
};
std::ostream & operator<<(std::ostream & os, const A::TestEnum & testEnum)
{
switch ( testEnum )
{
case A::TestEnum::first: os << "first"; break;
case A::TestEnum::second: os << "second"; break;
}
return os;
}
std::istream & operator>>(std::istream & is, A::TestEnum & testEnum)
{
std::string input;
is >> input;
if ( is.good() )
{
auto found = A::TestEnumCache.find(input);
if ( found != A::TestEnumCache.end() )
testEnum = found->second;
}
return is;
}
(3.)
struct A {
enum class TestEnum : size_t {
first,
second
};
bool alternate = false;
TestEnum testEnum;
REFLECT(A, alternate, testEnum)
static const std::unordered_map<std::string, TestEnum> TestEnumCache;
static const std::unordered_map<std::string, TestEnum> AltTestEnumCache;
};
const std::unordered_map<std::string, A::TestEnum> A::TestEnumCache = {
{ "first", A::TestEnum::first },
{ "second", A::TestEnum::second }
};
const std::unordered_map<std::string, A::TestEnum> A::AltTestEnumCache = {
{ "one", A::TestEnum::first },
{ "two", A::TestEnum::second }
};
template <>
struct Json::Input::Customize<A, RareTs::MemberType<A>::testEnum::type, RareTs::IndexOf<A>::testEnum>
{
static bool as(std::istream & is, Context &, const A & object, A::TestEnum & value)
{
std::string input = Json::Read::string(is);
if ( object.alternate )
{
auto found = A::AltTestEnumCache.find(input);
if ( found != A::AltTestEnumCache.end() )
{
value = found->second;
return true;
}
}
else
{
auto found = A::TestEnumCache.find(input);
if ( found != A::TestEnumCache.end() )
{
value = found->second;
return true;
}
}
return true;
}
};
template <>
struct Json::Output::Customize<A, RareTs::MemberType<A>::testEnum::type, RareTs::IndexOf<A>::testEnum>
{
static bool as(Json::OutStreamType & os, Context &, const A & object, const A::TestEnum & value)
{
if ( object.alternate )
{
switch ( value )
{
case A::TestEnum::first: os << "one"; break;
case A::TestEnum::second: os << "two"; break;
}
}
else
{
switch ( value )
{
case A::TestEnum::first: os << "first"; break;
case A::TestEnum::second: os << "second"; break;
}
}
return true;
}
};
Depending on your system and compiler, using StringBuffers instead of std::stringstream can result in a massive performance boost - with MSVC runtime is consistently halved or more. To use, string_buffer.h needs to be included in your project along with json.h, then you need to have USE_BUFFERED_STREAMS within your pre-defined macros; any output customizers will need to ensure they're using Json::OutStreamType rather than std::ostream, no other changes should be required.