Custom Serialization/Deserialization¶
See Custom Read/Write in the main README.md for using glz::custom
.
Advanced custom serialization/deserialization is achievable by implementing your own from
and to
specializations within the glz
namespace.
[!NOTE]
Glaze provides
parse
andserialize
helpers that decode the type of the value intended forfrom
orto
specializations. The developer only needs to specializefrom/to
structs, but you can useparse/serialize
for cleaner syntax (see examples or Glaze's codebase).
Example:
struct date
{
uint64_t data;
std::string human_readable;
};
template <>
struct glz::meta<date>
{
using T = date;
static constexpr auto value = object("date", &T::human_readable);
};
namespace glz
{
template <>
struct from<JSON, date>
{
template <auto Opts>
static void op(date& value, is_context auto&& ctx, auto&& it, auto&& end)
{
parse<JSON>::op<Opts>(value.human_readable, ctx, it, end);
value.data = std::stoi(value.human_readable);
}
};
template <>
struct to<JSON, date>
{
template <auto Opts>
static void op(date& value, is_context auto&& ctx, auto&& b, auto&& ix) noexcept
{
value.human_readable = std::to_string(value.data);
serialize<JSON>::op<Opts>(value.human_readable, ctx, b, ix);
}
};
}
void example() {
date d{};
d.data = 55;
std::string s{};
expect(not glz::write_json(d, s));
expect(s == R"("55")");
d.data = 0;
expect(not glz::read_json(d, s));
expect(d.data == 55);
}
Notes:
The templated Opts
parameter contains the compile time options.
For reading (from
specializations), the parameters are:
- value
: The object being parsed into
- ctx
: The context containing runtime options (use is_context auto&&
)
- it
: Iterator to the current position in the input buffer
- end
: Iterator to the end of the input buffer
For writing (to
specializations), the parameters are:
- value
: The object being serialized
- ctx
: The context containing runtime options (use is_context auto&&
)
- b
: The output buffer to write to
- ix
: The current index in the output buffer
UUID Example¶
Say we have a UUID library for converting a uuid_t
from a std::string_view
and to a std::string
.
namespace glz
{
template <>
struct from<JSON, uuid_t>
{
template <auto Opts>
static void op(uuid_t& uuid, is_context auto&& ctx, auto&& it, auto&& end)
{
// Initialize a string_view with the appropriately lengthed buffer
// Alternatively, use a std::string for any size (but this will allocate)
std::string_view str = "XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX";
parse<JSON>::op<Opts>(str, ctx, it, end);
uuid = uuid_lib::uuid_from_string_view(str);
}
};
template <>
struct to<JSON, uuid_t>
{
template <auto Opts>
static void op(const uuid_t& uuid, is_context auto&& ctx, auto&& b, auto&& ix) noexcept
{
std::string str = uuid_lib::uuid_to_string(uuid);
serialize<JSON>::op<Opts>(str, ctx, b, ix);
}
};
}
Handling Ambiguous Partial Specialization¶
You may want to custom parse a class that matches an underlying glaze partial specialization for a template like below:
If your own parsing function desires partial template specialization, then ambiguity may occur:
To solve this problem, glaze will check for custom_read
or custom_write
values within glz::meta
and remove the ambiguity and use the custom parser.