Glaze provides a number of wrappers that indicate at compile time how a value should be read and/or written. These wrappers allow you to modify the read/write behavior of a type without affecting your C++ class.
glz::append_arrays<&T::x> // When reading into an array that is appendable, the new data will be appended rather than overwrite
glz::bools_as_numbers<&T::x> // Read and write booleans as numbers
glz::quoted_num<&T::x> // Read and write numbers as strings
glz::quoted<&T::x> // Read a value as a string and unescape, to avoid the user having to parse twice
glz::number<&T::x> // Read a string as a number and writes the string as a number
glz::raw<&T::x> // Write out string like types without quotes
glz::raw_string<&T::string> // Do not decode/encode escaped characters for strings (improves read/write performance)
glz::escaped<&T::string> // Opposite of glz::raw_string, it turns off this behavior
glz::partial_read<&T::x> // Reads into only existing fields and elements and then exits without parsing the rest of the input
glz::invoke<&T::func> // Invoke a std::function, lambda, or member function with n-arguments as an array input
glz::write_float32<&T::x> // Writes out numbers with a maximum precision of float32_t
glz::write_float64<&T::x> // Writes out numbers with a maximum precision of float64_t
glz::write_float_full<&T::x> // Writes out numbers with full precision (turns off higher level float precision wrappers)
glz::custom<&T::read, &T::write> // Calls custom read and write std::functions, lambdas, or member functions
glz::manage<&T::x, &T::read_x, &T::write_x> // Calls read_x() after reading x and calls write_x() before writing x
glz::opts
is the compile time options struct passed to most of Glaze functions to configure read/write behavior. Often wrappers are associated with compile time options and can also be set via glz::opts
. For example, the glz::quoted_num
wrapper is associated with the quoted_num
boolean in glz::opts
.
When reading into an array that is appendable, the new data will be appended rather than overwrite
Associated option: glz::opts{.append_arrays = true};
struct append_obj
{
std::vector<std::string> names{};
std::vector<std::array<int, 2>> arrays{};
};
template <>
struct glz::meta<append_obj>
{
using T = append_obj;
static constexpr auto value = object("names", append_arrays<&T::names>, "arrays", append_arrays<&T::arrays>);
};
In use:
append_obj obj{};
expect(not glz::read_json(obj, R"({"names":["Bob"],"arrays":[[0,0]]})"));
expect(obj.names == std::vector<std::string>{"Bob"});
expect(obj.arrays == std::vector<std::array<int, 2>>{{0,0}});
expect(not glz::read_json(obj, R"({"names":["Liz"],"arrays":[[1,1]]})"));
expect(obj.names == std::vector<std::string>{"Bob", "Liz"});
expect(obj.arrays == std::vector<std::array<int, 2>>{{0,0},{1,1}});
Read and write booleans as numbers
Associated option: glz::opts{.bools_as_numbers = true};
struct bools_as_numbers_struct
{
bool a{};
bool b{};
bool c{};
bool d{};
struct glaze {
using T = bools_as_numbers_struct;
static constexpr auto value = glz::object("a", glz::bools_as_numbers<&T::a>, "b", glz::bools_as_numbers<&T::b>, &T::c, &T::d);
};
};
In use:
std::string s = R"({"a":1,"b":0,"c":true,"d":false})";
bools_as_numbers_struct obj{};
expect(!glz::read_json(obj, s));
expect(obj.a == true);
expect(obj.b == false);
expect(glz::write_json(obj) == s);
You don't have to use wrappers if you want the global behavior to handle booleans as numbers.
std::string s = R"([1,0,1,0])";
std::array<bool, 4> obj{};
constexpr glz::opts opts{.bools_as_numbers = true};
expect(!glz::read<opts>(obj, s));
expect(glz::write<opts>(obj) == s);
Read and write numbers as strings.
Associated option: glz::opts{.quoted_num = true};
struct foo {
int x{};
};
template <>
struct glz::meta<foo> {
using T = foo;
static constexpr auto value = object("x", quoted_num<&T::x>);
};
In use:
std::string input = R"({ "x": "5" })";
foo obj{};
expect(!glz::read_json(obj, input));
expect(glz::write_json(obj) == R"({ "x": "5" })");
quoted_num
is more efficient thanquoted
for numbers.
When reading, first reads a value as a string, which unescapes, and then reads the value normally. When writing, will first write the value as a string and then write the string to produce escapes.
glz::quoted
is useful for storing escaped JSON inside of a higher level JSON object.
struct client_state
{
uint64_t id{};
std::map<std::string, std::vector<std::string>> layouts{};
};
template <>
struct glz::meta<client_state>
{
using T = client_state;
static constexpr auto value = object("id", &T::id, "layouts", quoted<&T::layouts>);
};
In use:
client_state obj{};
std::string input = R"({
"id": 4848,
"layouts": "{\"first layout\": [ \"inner1\", \"inner2\" ] }"
})";
expect(!glz::read_json(obj, input));
expect(obj.id == 4848);
expect(obj.layouts.at("first layout") == std::vector<std::string>{"inner1", "inner2"});
std::string out{};
glz::write_json(obj, out);
expect(out == R"({"id":4848,"layouts":"{\"first layout\":[\"inner1\",\"inner2\"]}"})");
Read JSON numbers into strings and write strings as JSON numbers.
Associated option: glz::opts{.number = true};
struct numbers_as_strings
{
std::string x{};
std::string y{};
};
template <>
struct glz::meta<numbers_as_strings>
{
using T = numbers_as_strings;
static constexpr auto value = object("x", glz::number<&T::x>, "y", glz::number<&T::y>);
};
In use:
std::string input = R"({"x":555,"y":3.14})";
numbers_as_strings obj{};
expect(!glz::read_json(obj, input));
expect(obj.x == "555");
expect(obj.y == "3.14");
std::string output;
glz::write_json(obj, output);
expect(input == output);
Write out string like types without quotes.
Useful for when a string is already in JSON format and doesn't need to be quoted.
Associated option: glz::opts{.raw = true};
struct raw_struct
{
std::string str{};
};
template <>
struct glz::meta<raw_struct>
{
using T = raw_struct;
static constexpr auto value = object("str", glz::raw<&T::str>);
};
In use:
suite raw_test = [] {
raw_struct obj{.str = R"("Hello")"};
// quotes would have been escaped if str were not wrapped with raw
expect(glz::write_json(obj) == R"({"str":"Hello"})");
};
Do not decode/encode escaped characters for strings (improves read/write performance).
If your code does not care about decoding escaped characters or you know your input will never have escaped characters, this wrapper makes reading/writing that string faster.
Associated option: glz::opts{.raw_string = true};
struct raw_stuff
{
std::string a{};
std::string b{};
std::string c{};
struct glaze
{
using T = raw_stuff;
static constexpr auto value = glz::object(&T::a, &T::b, &T::c);
};
};
struct raw_stuff_wrapper
{
raw_stuff data{};
struct glaze
{
using T = raw_stuff_wrapper;
static constexpr auto value{glz::raw_string<&T::data>};
};
};
In use:
raw_stuff_wrapper obj{};
std::string buffer = R"({"a":"Hello\nWorld","b":"Hello World","c":"\tHello\bWorld"})";
expect(!glz::read_json(obj, buffer));
expect(obj.data.a == R"(Hello\nWorld)");
expect(obj.data.b == R"(Hello World)");
expect(obj.data.c == R"(\tHello\bWorld)");
buffer.clear();
glz::write_json(obj, buffer);
expect(buffer == R"({"a":"Hello\nWorld","b":"Hello World","c":"\tHello\bWorld"})");
The glz::escaped
wrapper turns off the effects of glz::raw_string
.
struct raw_stuff_escaped
{
raw_stuff data{};
struct glaze
{
using T = raw_stuff_escaped;
static constexpr auto value{glz::escaped<&T::data>};
};
};
In use:
raw_stuff_escaped obj{};
std::string buffer = R"({"a":"Hello\nWorld"})";
expect(!glz::read_json(obj, buffer));
expect(obj.data.a ==
R"(Hello
World)");
buffer.clear();
glz::write_json(obj, buffer);
expect(buffer == R"({"a":"Hello\nWorld","b":"","c":""})");
Reads into existing object and array elements and then exits without parsing the rest of the input. More documentation concerning partial_read
can be found in the Partial Read documentation.
partial_read
is useful when parsing header information before deciding how to decode the rest of a document. Or, when you only care about the first few elements of an array.
Invoke a std::function or member function with n-arguments as an array input.
struct invoke_struct
{
int y{};
std::function<void(int x)> square{};
void add_one() { ++y; }
// MSVC requires this constructor for 'this' to be captured
invoke_struct()
{
square = [&](int x) { y = x * x; };
}
};
template <>
struct glz::meta<invoke_struct>
{
using T = invoke_struct;
static constexpr auto value = object("square", invoke<&T::square>, "add_one", invoke<&T::add_one>);
};
In use:
std::string s = R"(
{
"square":[5],
"add_one":[]
})";
invoke_struct obj{};
expect(!glz::read_json(obj, s));
expect(obj.y == 26); // 5 * 5 + 1
};
Writes out numbers with a maximum precision of float32_t
.
struct write_precision_t
{
double pi = std::numbers::pi_v<double>;
struct glaze
{
using T = write_precision_t;
static constexpr auto value = glz::object("pi", glz::write_float32<&T::pi>);
};
};
In use:
write_precision_t obj{};
std::string json_float = glz::write_json(obj);
expect(json_float == R"({"pi":3.1415927})") << json_float;
Writes out numbers with a maximum precision of float64_t
.
Writes out numbers with full precision (turns off higher level float precision wrappers).
enum struct float_precision : uint8_t { full, float32 = 4, float64 = 8, float128 = 16 };
glz::opts
// The maximum precision type used for writing floats, higher precision floats will be cast down to this precision
float_precision float_max_write_precision{};
Calls custom read and write std::functions, lambdas, or member functions.
struct custom_encoding
{
uint64_t x{};
std::string y{};
std::array<uint32_t, 3> z{};
void read_x(const std::string& s) { x = std::stoi(s); }
uint64_t write_x() { return x; }
void read_y(const std::string& s) { y = "hello" + s; }
auto& write_z()
{
z[0] = 5;
return z;
}
};
template <>
struct glz::meta<custom_encoding>
{
using T = custom_encoding;
static constexpr auto value = object("x", custom<&T::read_x, &T::write_x>, //
"y", custom<&T::read_y, &T::y>, //
"z", custom<&T::z, &T::write_z>);
};
In use:
"custom_reading"_test = [] {
custom_encoding obj{};
std::string s = R"({"x":"3","y":"world","z":[1,2,3]})";
expect(!glz::read_json(obj, s));
expect(obj.x == 3);
expect(obj.y == "helloworld");
expect(obj.z == std::array<uint32_t, 3>{1, 2, 3});
};
"custom_writing"_test = [] {
custom_encoding obj{};
std::string s = R"({"x":"3","y":"world","z":[1,2,3]})";
expect(!glz::read_json(obj, s));
std::string out{};
glz::write_json(obj, out);
expect(out == R"({"x":3,"y":"helloworld","z":[5,2,3]})");
};
Showing use of constexpr lambdas for customization.
struct custom_buffer_input
{
std::string str{};
};
template <>
struct glz::meta<custom_buffer_input>
{
static constexpr auto read_x = [](custom_buffer_input& s, const std::string& input) { s.str = input; };
static constexpr auto write_x = [](auto& s) -> auto& { return s.str; };
static constexpr auto value = glz::object("str", glz::custom<read_x, write_x>);
};
In use:
std::string s = R"({"str":"Hello!"})";
custom_buffer_input obj{};
expect(!glz::read_json(obj, s));
expect(obj.str == "Hello!");
s.clear();
glz::write_json(obj, s);
expect(s == R"({"str":"Hello!"})");
expect(obj.str == "Hello!");
Note
With read lambdas like [](custom_buffer_input& s, const std::string& input)
, both types must be concrete (cannot use auto
), otherwise you'll get a compilation error noting this. The reason is that Glaze must be able to determine what type to decode into before passing the decoded value to input
.
Calls a read function after reading and calls a write function before writing.
glz::manage
is useful for transforming state from a user facing format into a more complex or esoteric internal format.
struct manage_x
{
std::vector<int> x{};
std::vector<int> y{};
bool read_x()
{
y = x;
return true;
}
bool write_x()
{
x = y;
return true;
}
};
template <>
struct glz::meta<manage_x>
{
using T = manage_x;
static constexpr auto value = object("x", manage<&T::x, &T::read_x, &T::write_x>);
};
In use:
manage_x obj{};
std::string s = R"({"x":[1,2,3]})";
expect(!glz::read_json(obj, s));
expect(obj.y[0] == 1);
expect(obj.y[1] == 2);
obj.x.clear();
s.clear();
glz::write_json(obj, s);
expect(s == R"({"x":[1,2,3]})");
expect(obj.x[0] == 1);
expect(obj.x[1] == 2);
struct manage_x_lambda
{
std::vector<int> x{};
std::vector<int> y{};
};
template <>
struct glz::meta<manage_x_lambda>
{
using T = manage_x_lambda;
static constexpr auto read_x = [](auto& s) {
s.y = s.x;
return true;
};
static constexpr auto write_x = [](auto& s) {
s.x = s.y;
return true;
};
[[maybe_unused]] static constexpr auto value = object("x", manage<&T::x, read_x, write_x>);
};
In use:
manage_x_lambda obj{};
std::string s = R"({"x":[1,2,3]})";
expect(!glz::read_json(obj, s));
expect(obj.y[0] == 1);
expect(obj.y[1] == 2);
obj.x.clear();
s.clear();
glz::write_json(obj, s);
expect(s == R"({"x":[1,2,3]})");
expect(obj.x[0] == 1);
expect(obj.x[1] == 2);