Skip to content

JSON Pointer Syntax

Link to simple JSON pointer syntax explanation

Glaze supports JSON pointer syntax access in a C++ context. This is extremely helpful for building generic APIs, which allows values of complex objects to be accessed without needed know the encapsulating class.

my_struct s{};
auto d = glz::get<double>(s, "/d");
// d.value() is a std::reference_wrapper to d in the structure s
my_struct s{};
glz::set(s, "/d", 42.0);
// d is now 42.0

JSON pointer syntax works with deeply nested objects and anything serializable.

// Tuple Example
auto tuple = std::make_tuple(3, 2.7, std::string("curry"));
glz::set(tuple, "/0", 5);
expect(std::get<0>(tuple) == 5.0);

read_as

read_as allows you to read into an object from a JSON pointer and an input buffer.

Thing thing{};
glz::read_as_json(thing, "/vec3", "[7.6, 1292.1, 0.333]");
expect(thing.vec3.x == 7.6 && thing.vec3.y == 1292.1 &&
thing.vec3.z == 0.333);

glz::read_as_json(thing, "/vec3/2", "999.9");
expect(thing.vec3.z == 999.9);

get_as_json

get_as_json allows you to get a targeted value from within an input buffer. This is especially useful if you need to change how an object is parsed based on a value within the object.

std::string s = R"({"obj":{"x":5.5}})";
auto z = glz::get_as_json<double, "/obj/x">(s);
expect(z == 5.5);

[!IMPORTANT]

get_as_json does not validate JSON beyond the targeted value. This is to avoid parsing the entire document once the targeted value is reached.

get_sv_json

get_sv_json allows you to get a std::string_view to a targeted value within an input buffer. This can be more efficient to check values and handle custom parsing than constructing a new value with get_as_json.

std::string s = R"({"obj":{"x":5.5}})";
auto view = glz::get_sv_json<"/obj/x">(s);
expect(view == "5.5");

[!IMPORTANT]

get_sv_json does not validate JSON beyond the targeted value. This is to avoid parsing the entire document once the targeted value is reached.

write_at

write_at allows raw text to be written to a specific JSON value pointed at via JSON Pointer syntax.

Compile-time Path (Template Parameter)

When the path is known at compile time, use the template parameter version:

std::string buffer = R"( { "action": "DELETE", "data": { "x": 10, "y": 200 }})";
auto ec = glz::write_at<"/action">(R"("GO!")", buffer);
expect(buffer == R"( { "action": "GO!", "data": { "x": 10, "y": 200 }})");

Another example:

std::string buffer = R"({"str":"hello","number":3.14,"sub":{"target":"X"}})";
auto ec = glz::write_at<"/sub/target">("42", buffer);
expect(not ec);
expect(buffer == R"({"str":"hello","number":3.14,"sub":{"target":42}})");

Runtime Path (String Parameter)

When the path is determined at runtime, pass it as the first argument:

std::string buffer = R"({"action":"DELETE","data":{"x":10,"y":200}})";
std::string path = "/action";
auto ec = glz::write_at(path, R"("GO!")", buffer);
expect(not ec);
expect(buffer == R"({"action":"GO!","data":{"x":10,"y":200}})");

This is useful for dynamic path construction:

std::string buffer = R"({"users":[{"name":"Alice"},{"name":"Bob"}]})";

// Update user at runtime-determined index
size_t user_index = 1;
std::string path = "/users/" + std::to_string(user_index) + "/name";
glz::write_at(path, R"("Charlie")", buffer);
// buffer is now: {"users":[{"name":"Alice"},{"name":"Charlie"}]}

get_view_json

get_view_json returns a std::span pointing to the raw JSON bytes of a value at a given path. This is useful for extracting JSON fragments without parsing.

Compile-time Path

std::string buffer = R"({"obj":{"x":5.5}})";
auto view = glz::get_view_json<"/obj/x">(buffer);
if (view) {
   // view->data() points to "5.5" within buffer
   std::string_view sv{view->data(), view->size()};
   expect(sv == "5.5");
}

Runtime Path

std::string buffer = R"({"action":"DELETE","data":{"x":10,"y":200}})";

auto view = glz::get_view_json("/action", buffer);
if (view) {
   std::string_view sv{view->data(), view->size()};
   expect(sv == R"("DELETE")");
}

// Navigate nested paths
auto view2 = glz::get_view_json("/data/x", buffer);
if (view2) {
   std::string_view sv{view2->data(), view2->size()};
   expect(sv == "10");
}

// Get entire sub-object
auto view3 = glz::get_view_json("/data", buffer);
if (view3) {
   std::string_view sv{view3->data(), view3->size()};
   expect(sv == R"({"x":10,"y":200})");
}

Array Access

std::string buffer = R"({"items":[100,200,300,400,500]})";

auto view = glz::get_view_json("/items/2", buffer);
if (view) {
   std::string_view sv{view->data(), view->size()};
   expect(sv == "300");
}

Error Handling

Both get_view_json and write_at return error information:

std::string buffer = R"({"a":1})";

// Non-existent path
auto view = glz::get_view_json("/nonexistent", buffer);
if (!view) {
   // view.error().ec == glz::error_code::key_not_found
}

// write_at also returns errors
auto ec = glz::write_at("/nonexistent", "42", buffer);
if (ec) {
   // Path not found, buffer unchanged
}

RFC 6901 Escape Sequences

JSON Pointer uses escape sequences for special characters in keys: - ~0 represents ~ - ~1 represents /

Both compile-time and runtime APIs support these escapes:

std::string buffer = R"({"a/b":1,"c~d":2})";

// Key contains "/" - use ~1
auto ec1 = glz::write_at("/a~1b", "10", buffer);
expect(not ec1);

// Key contains "~" - use ~0
auto ec2 = glz::write_at("/c~0d", "20", buffer);
expect(not ec2);

expect(buffer == R"({"a/b":10,"c~d":20})");

Runtime vs Compile-time Comparison

Feature Compile-time Runtime
Path specification Template parameter Function argument
Path validation Compile-time Runtime
Use case Fixed, known paths Dynamic paths
Syntax glz::write_at<"/path">(value, buffer) glz::write_at("/path", value, buffer)

Both versions: - Support the same JSON Pointer syntax - Handle RFC 6901 escape sequences - Work with nested objects and arrays - Return the same error types

Using JSON Pointers with glz::generic

When working with glz::generic, JSON pointers can extract both primitives and containers:

glz::generic json{};
std::string buffer = R"({"names": ["Alice", "Bob"], "count": 2})";
glz::read_json(json, buffer);

// Extract container types - returns expected<T, error_ctx>
auto names = glz::get<std::vector<std::string>>(json, "/names");
if (names) {
  // names->size() == 2, efficient direct conversion
}

// Extract primitives - returns expected<reference_wrapper<T>, error_ctx>
auto count = glz::get<double>(json, "/count");
if (count) {
  // count->get() == 2.0
}

See Generic JSON for more details on the optimized conversion from generic to containers and primitives.

Seek

glz::seek allows you to call a lambda on a nested value.

my_struct s{};
std::any a{};
glz::seek([&](auto& value) { a = value; }, s, "/hello");
expect(a.has_value() && std::any_cast<std::string>(a) == "Hello World");