2022-01-19 09:31:28 -05:00
|
|
|
#include <qpdf/JSONHandler.hh>
|
|
|
|
#include <qpdf/QUtil.hh>
|
|
|
|
#include <qpdf/QTC.hh>
|
2022-01-28 07:46:04 -05:00
|
|
|
#include <qpdf/QPDFUsage.hh>
|
2022-01-19 09:31:28 -05:00
|
|
|
|
|
|
|
JSONHandler::JSONHandler() :
|
|
|
|
m(new Members())
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
JSONHandler::Members::Members()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2022-01-28 07:46:04 -05:00
|
|
|
void
|
|
|
|
JSONHandler::usage(std::string const& msg)
|
|
|
|
{
|
|
|
|
throw QPDFUsage(msg);
|
|
|
|
}
|
|
|
|
|
2022-01-19 09:31:28 -05:00
|
|
|
void
|
|
|
|
JSONHandler::addAnyHandler(json_handler_t fn)
|
|
|
|
{
|
|
|
|
this->m->h.any_handler = fn;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
JSONHandler::addNullHandler(void_handler_t fn)
|
|
|
|
{
|
|
|
|
this->m->h.null_handler = fn;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
JSONHandler::addStringHandler(string_handler_t fn)
|
|
|
|
{
|
|
|
|
this->m->h.string_handler = fn;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
JSONHandler::addNumberHandler(string_handler_t fn)
|
|
|
|
{
|
|
|
|
this->m->h.number_handler = fn;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
JSONHandler::addBoolHandler(bool_handler_t fn)
|
|
|
|
{
|
|
|
|
this->m->h.bool_handler = fn;
|
|
|
|
}
|
|
|
|
|
2022-01-20 08:53:53 -05:00
|
|
|
void
|
2022-01-30 08:18:04 -05:00
|
|
|
JSONHandler::addDictHandlers(json_handler_t start_fn, void_handler_t end_fn)
|
2022-01-20 08:53:53 -05:00
|
|
|
{
|
|
|
|
this->m->h.dict_start_handler = start_fn;
|
|
|
|
this->m->h.dict_end_handler = end_fn;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
JSONHandler::addDictKeyHandler(
|
|
|
|
std::string const& key, std::shared_ptr<JSONHandler> dkh)
|
2022-01-19 09:31:28 -05:00
|
|
|
{
|
2022-01-20 08:53:53 -05:00
|
|
|
this->m->h.dict_handlers[key] = dkh;
|
2022-01-19 09:31:28 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
JSONHandler::addFallbackDictHandler(std::shared_ptr<JSONHandler> fdh)
|
|
|
|
{
|
|
|
|
this->m->h.fallback_dict_handler = fdh;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2022-01-30 13:45:02 -05:00
|
|
|
JSONHandler::addArrayHandlers(json_handler_t start_fn,
|
2022-01-20 08:53:53 -05:00
|
|
|
void_handler_t end_fn,
|
|
|
|
std::shared_ptr<JSONHandler> ah)
|
2022-01-19 09:31:28 -05:00
|
|
|
{
|
2022-01-20 08:53:53 -05:00
|
|
|
this->m->h.array_start_handler = start_fn;
|
|
|
|
this->m->h.array_end_handler = end_fn;
|
|
|
|
this->m->h.array_item_handler = ah;
|
2022-01-19 09:31:28 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
JSONHandler::handle(std::string const& path, JSON j)
|
|
|
|
{
|
|
|
|
if (this->m->h.any_handler)
|
|
|
|
{
|
|
|
|
this->m->h.any_handler(path, j);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
bool handled = false;
|
|
|
|
bool bvalue = false;
|
2022-01-28 09:05:06 -05:00
|
|
|
std::string s_value;
|
2022-01-19 09:31:28 -05:00
|
|
|
if (this->m->h.null_handler && j.isNull())
|
|
|
|
{
|
|
|
|
this->m->h.null_handler(path);
|
|
|
|
handled = true;
|
|
|
|
}
|
2022-01-28 09:05:06 -05:00
|
|
|
if (this->m->h.string_handler && j.getString(s_value))
|
2022-01-19 09:31:28 -05:00
|
|
|
{
|
2022-01-28 09:05:06 -05:00
|
|
|
this->m->h.string_handler(path, s_value);
|
2022-01-19 09:31:28 -05:00
|
|
|
handled = true;
|
|
|
|
}
|
2022-01-28 09:05:06 -05:00
|
|
|
if (this->m->h.number_handler && j.getNumber(s_value))
|
2022-01-19 09:31:28 -05:00
|
|
|
{
|
2022-01-28 09:05:06 -05:00
|
|
|
this->m->h.number_handler(path, s_value);
|
2022-01-19 09:31:28 -05:00
|
|
|
handled = true;
|
|
|
|
}
|
|
|
|
if (this->m->h.bool_handler && j.getBool(bvalue))
|
|
|
|
{
|
|
|
|
this->m->h.bool_handler(path, bvalue);
|
|
|
|
handled = true;
|
|
|
|
}
|
2022-01-20 08:53:53 -05:00
|
|
|
if (this->m->h.dict_start_handler && j.isDictionary())
|
2022-01-19 09:31:28 -05:00
|
|
|
{
|
2022-01-30 08:18:04 -05:00
|
|
|
this->m->h.dict_start_handler(path, j);
|
2022-01-19 09:31:28 -05:00
|
|
|
std::string path_base = path;
|
|
|
|
if (path_base != ".")
|
|
|
|
{
|
|
|
|
path_base += ".";
|
|
|
|
}
|
|
|
|
j.forEachDictItem([&path, &path_base, this](
|
|
|
|
std::string const& k, JSON v) {
|
|
|
|
auto i = this->m->h.dict_handlers.find(k);
|
|
|
|
if (i == this->m->h.dict_handlers.end())
|
|
|
|
{
|
|
|
|
if (this->m->h.fallback_dict_handler.get())
|
|
|
|
{
|
|
|
|
this->m->h.fallback_dict_handler->handle(
|
|
|
|
path_base + k, v);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
QTC::TC("libtests", "JSONHandler unexpected key");
|
2022-01-28 07:46:04 -05:00
|
|
|
usage("JSON handler found unexpected key " + k +
|
|
|
|
" in object at " + path);
|
2022-01-19 09:31:28 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
i->second->handle(path_base + k, v);
|
|
|
|
}
|
|
|
|
});
|
2022-01-20 08:53:53 -05:00
|
|
|
this->m->h.dict_end_handler(path);
|
2022-01-19 09:31:28 -05:00
|
|
|
handled = true;
|
|
|
|
}
|
2022-01-20 08:53:53 -05:00
|
|
|
if (this->m->h.array_start_handler && j.isArray())
|
2022-01-19 09:31:28 -05:00
|
|
|
{
|
2022-01-30 13:45:02 -05:00
|
|
|
this->m->h.array_start_handler(path, j);
|
2022-01-19 09:31:28 -05:00
|
|
|
size_t i = 0;
|
|
|
|
j.forEachArrayItem([&i, &path, this](JSON v) {
|
2022-01-20 08:53:53 -05:00
|
|
|
this->m->h.array_item_handler->handle(
|
2022-01-19 09:31:28 -05:00
|
|
|
path + "[" + QUtil::uint_to_string(i) + "]", v);
|
|
|
|
++i;
|
|
|
|
});
|
2022-01-20 08:53:53 -05:00
|
|
|
this->m->h.array_end_handler(path);
|
2022-01-19 09:31:28 -05:00
|
|
|
handled = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (! handled)
|
|
|
|
{
|
|
|
|
// It would be nice to include information about what type the
|
|
|
|
// object was and what types were allowed, but we're relying
|
|
|
|
// on schema validation to make sure input is properly
|
|
|
|
// structured before calling the handlers. It would be
|
|
|
|
// different if this code were trying to be part of a
|
|
|
|
// general-purpose JSON package.
|
|
|
|
QTC::TC("libtests", "JSONHandler unhandled value");
|
2022-01-28 07:46:04 -05:00
|
|
|
usage("JSON handler: value at " + path + " is not of expected type");
|
2022-01-19 09:31:28 -05:00
|
|
|
}
|
|
|
|
}
|