Processors can be used to apply transformations to struct serialization and deserialization.
For instance, C++ usually uses snake_case, but JSON uses camelCase. One way to handle this is rfl::Rename, but a more automated way would be to use a processor:
The problem here is that button_pressed_t and button_released_t virtually look
indistinguishable when they are serialized. The will both be serialized to {}.
But you can add this processor to automatically add tags and avoid the problem:
Note that there are other ways to address problems like this, for instance rfl::TaggedUnion.
Please refer to the relevant sections of the documentation.
By default, reflect-cpp does not allow reading into raw pointers. (Writing from raw pointers is never a problem.) This is because reading into raw pointers means that the library will allocate memory that the user then has to manually delete. This can lead to misunderstandings and memory leaks.
You might want to consider using some alternatives, such as std::unique_ptr, rfl::Box,
std::shared_ptr, rfl::Ref or std::optional. But if you absolutely have to use raw pointers, you can pass rfl::AllowRawPtrs to read:
The rfl::DefaultIfMissing processor is only relevant for reading data. For writing data, it will make no difference.
Usually, when fields are missing in the input data, this will lead to an error
(unless they are optional fields).
But if you pass the rfl::DefaultIfMissing processor, then missing fields will be
replaced by their default value.
last_name and town have been replaced by the default values.
Because you have not passed a default value to town, the default value
of the type is used instead.
This is particularly relevant for binary formats, which do not emphasize readability,
like msgpack or flexbuffers. Removing the field names can reduce the size of the
resulting bytestrings and significantly speed up read and write time,
depending on the dataset.
However, it makes it more difficult to maintain backwards compatability.
Note that rfl::NoFieldNames is not supported for BSON, TOML, XML, or YAML, due
to limitations of these formats.
As we have seen in the section on optional fields, when a std::optional is
std::nullopt, it is usually not written at all. But if you want them to be explicitly
written as null, you can use this processor. The same thing applies to std::shared_ptr and
std::unique_ptr.
By default, rfl::json::read will accept both "town":null and just
leaving out the field town. However, if you want to require the field
town to be included, you can add rfl::NoOptionals to read:
In principle, writing your own processors is not very difficult. You need to define a struct, which takes has a static method called process taking a named tuple as an input and then returning a modified named tuple. The process method should accept the type of the original struct as a template parameter.