#include #include #include #include using json = nlohmann::json; // a simple event consumer that collects string representations of the passed // values and their source locations; // note inheriting from json::json_sax_t is not required, but can // help not to forget a required function class sax_event_consumer : public json::json_sax_t { public: std::vector events; std::size_t next_token_start_pos = 0; std::size_t next_token_end_pos = 0; void next_token_start(std::size_t pos) { next_token_start_pos = pos; } void next_token_end(std::size_t pos) { next_token_end_pos = pos; } std::string location_as_str() const { return "at=[" + std::to_string(next_token_start_pos) + "," + std::to_string(next_token_end_pos) + ")"; } bool null() override { events.push_back("null(" + location_as_str() + ")"); return true; } bool boolean(bool val) override { events.push_back("boolean(val=" + std::string(val ? "true" : "false") + ", " + location_as_str() + ")"); return true; } bool number_integer(number_integer_t val) override { events.push_back("number_integer(val=" + std::to_string(val) + ", " + location_as_str() + ")"); return true; } bool number_unsigned(number_unsigned_t val) override { events.push_back("number_unsigned(val=" + std::to_string(val) + ", " + location_as_str() + ")"); return true; } bool number_float(number_float_t val, const string_t& s) override { events.push_back("number_float(val=" + std::to_string(val) + ", s=" + s + ", " + location_as_str() + ")"); return true; } bool string(string_t& val) override { events.push_back("string(val=" + val + ", " + location_as_str() + ")"); return true; } bool start_object(std::size_t elements) override { events.push_back("start_object(elements=" + std::to_string(elements) + ", " + location_as_str() + ")"); return true; } bool end_object() override { events.push_back("end_object(" + location_as_str() + ")"); return true; } bool start_array(std::size_t elements) override { events.push_back("start_array(elements=" + std::to_string(elements) + ", " + location_as_str() + ")"); return true; } bool end_array() override { events.push_back("end_array(" + location_as_str() + ")"); return true; } bool key(string_t& val) override { events.push_back("key(val=" + val + ", " + location_as_str() + ")"); return true; } bool binary(json::binary_t& val) override { events.push_back("binary(val=[...], " + location_as_str() + ")"); return true; } bool parse_error(std::size_t position, const std::string& last_token, const json::exception& ex) override { events.push_back("parse_error(position=" + std::to_string(position) + ", last_token=" + last_token + ",\n ex=" + std::string(ex.what()) + ")"); return false; } }; int main() { // a JSON text auto text = R"( { "Image": { "Width": 800, "Height": 600, "Title": "View from 15th Floor", "Thumbnail": { "Url": "http://www.example.com/image/481989943", "Height": 125, "Width": 100 }, "Animated" : false, "IDs": [116, 943, 234, -38793], "DeletionDate": null, "Distance": 12.723374634 } }] )"; // create a SAX event consumer object sax_event_consumer sec; // parse JSON bool result = json::sax_parse(text, &sec); // output the recorded events for (auto& event : sec.events) { std::cout << event << "\n"; } // output the result of sax_parse std::cout << "\nresult: " << std::boolalpha << result << std::endl; }