aboutsummaryrefslogtreecommitdiffstats
path: root/include/qpdf
diff options
context:
space:
mode:
authorJay Berkenbilt <ejb@ql.org>2022-01-19 15:31:28 +0100
committerJay Berkenbilt <ejb@ql.org>2022-01-30 19:11:03 +0100
commit37105710ee0b332a3020d4b3220c95b8f4267555 (patch)
treeee4b520b1c141033ba16ba3696def57fc90f60fa /include/qpdf
parenta6df6fdaf724ed5fc6f7e8c021f7804bd5a9c0e2 (diff)
downloadqpdf-37105710ee0b332a3020d4b3220c95b8f4267555.tar.zst
Implement JSONHandler for recursively processing JSON
Diffstat (limited to 'include/qpdf')
-rw-r--r--include/qpdf/JSON.hh25
-rw-r--r--include/qpdf/JSONHandler.hh142
2 files changed, 166 insertions, 1 deletions
diff --git a/include/qpdf/JSON.hh b/include/qpdf/JSON.hh
index 676becbf..3b13b4fe 100644
--- a/include/qpdf/JSON.hh
+++ b/include/qpdf/JSON.hh
@@ -30,7 +30,10 @@
// create temporary JSON objects on the stack, add them to other
// objects, and let them go out of scope safely. It also means that if
// the json JSON object is added in more than one place, all copies
-// share underlying data.
+// share underlying data. This makes them similar in structure and
+// behavior to QPDFObjectHandle and may feel natural within the QPDF
+// codebase, but it is also a good reason not to use this as a
+// general-purpose JSON package.
#include <qpdf/DLL.h>
#include <qpdf/PointerHolder.hh>
@@ -38,6 +41,7 @@
#include <map>
#include <vector>
#include <list>
+#include <functional>
class JSON
{
@@ -77,6 +81,24 @@ class JSON
QPDF_DLL
bool isDictionary() const;
+ // Accessors. Accessor behavior:
+ //
+ // - If argument is wrong type, including null, return false
+ // - If argument is right type, return true and initialize the value
+ QPDF_DLL
+ bool getString(std::string& utf8) const;
+ QPDF_DLL
+ bool getNumber(std::string& value) const;
+ QPDF_DLL
+ bool getBool(bool& value) const;
+ QPDF_DLL
+ bool isNull() const;
+ QPDF_DLL
+ bool forEachDictItem(
+ std::function<void(std::string const& key, JSON value)> fn) const;
+ QPDF_DLL
+ bool forEachArrayItem(std::function<void(JSON value)> fn) const;
+
// Check this JSON object against a "schema". This is not a schema
// according to any standard. It's just a template of what the
// JSON is supposed to contain. The checking does the following:
@@ -129,6 +151,7 @@ class JSON
JSON_string(std::string const& utf8);
virtual ~JSON_string();
virtual std::string unparse(size_t depth) const;
+ std::string utf8;
std::string encoded;
};
struct JSON_number: public JSON_value
diff --git a/include/qpdf/JSONHandler.hh b/include/qpdf/JSONHandler.hh
new file mode 100644
index 00000000..bde134ac
--- /dev/null
+++ b/include/qpdf/JSONHandler.hh
@@ -0,0 +1,142 @@
+// Copyright (c) 2005-2021 Jay Berkenbilt
+//
+// This file is part of qpdf.
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+//
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+//
+// Versions of qpdf prior to version 7 were released under the terms
+// of version 2.0 of the Artistic License. At your option, you may
+// continue to consider qpdf to be licensed under those terms. Please
+// see the manual for additional information.
+
+#ifndef JSONHANDLER_HH
+#define JSONHANDLER_HH
+
+#include <qpdf/DLL.h>
+#include <qpdf/PointerHolder.hh>
+#include <qpdf/JSON.hh>
+#include <string>
+#include <map>
+#include <functional>
+#include <stdexcept>
+#include <memory>
+
+class JSONHandler
+{
+ public:
+ // Error exception is thrown if there are any errors validating
+ // the JSON object.
+ class QPDF_DLL_CLASS Error: public std::runtime_error
+ {
+ public:
+ QPDF_DLL
+ Error(std::string const&);
+ };
+
+ QPDF_DLL
+ JSONHandler();
+
+ QPDF_DLL
+ ~JSONHandler() = default;
+
+ // Based on the type of handler, expect the object to be of a
+ // certain type. JSONHandler::Error is thrown otherwise. Multiple
+ // handlers may be registered, which allows the object to be of
+ // various types. If an anyHandler is added, no other handler will
+ // be called.
+
+ typedef std::function<void(
+ std::string const& path, JSON value)> json_handler_t;
+ typedef std::function<void(
+ std::string const& path)> void_handler_t;
+ typedef std::function<void(
+ std::string const& path, std::string const& value)> string_handler_t;
+ typedef std::function<void(
+ std::string const& path, bool value)> bool_handler_t;
+
+ // If an any handler is added, it will be called for any value
+ // including null, and no other handler will be called.
+ QPDF_DLL
+ void addAnyHandler(json_handler_t fn);
+
+ // If any of the remaining handlers are registered, each
+ // registered handle will be called.
+ QPDF_DLL
+ void addNullHandler(void_handler_t fn);
+ QPDF_DLL
+ void addStringHandler(string_handler_t fn);
+ QPDF_DLL
+ void addNumberHandler(string_handler_t fn);
+ QPDF_DLL
+ void addBoolHandler(bool_handler_t fn);
+
+ // Returns a reference to a map: keys are expected object keys,
+ // and values are handlers for that object.
+ QPDF_DLL
+ std::map<std::string, std::shared_ptr<JSONHandler>>& addDictHandlers();
+
+ // Apply the given handler to any key not explicitly in dict
+ // handlers.
+ QPDF_DLL
+ void addFallbackDictHandler(std::shared_ptr<JSONHandler>);
+
+ // Apply the given handler to each element of the array.
+ QPDF_DLL
+ void addArrayHandler(std::shared_ptr<JSONHandler>);
+
+ // Apply handlers recursively to a JSON object.
+ QPDF_DLL
+ void handle(std::string const& path, JSON j);
+
+ private:
+ JSONHandler(JSONHandler const&) = delete;
+
+ struct Handlers
+ {
+ Handlers() :
+ any_handler(nullptr),
+ null_handler(nullptr),
+ string_handler(nullptr),
+ number_handler(nullptr),
+ bool_handler(nullptr)
+ {
+ }
+
+ json_handler_t any_handler;
+ void_handler_t null_handler;
+ string_handler_t string_handler;
+ string_handler_t number_handler;
+ bool_handler_t bool_handler;
+ std::map<std::string, std::shared_ptr<JSONHandler>> dict_handlers;
+ std::shared_ptr<JSONHandler> fallback_dict_handler;
+ std::shared_ptr<JSONHandler> array_handler;
+ };
+
+ class Members
+ {
+ friend class JSONHandler;
+
+ public:
+ QPDF_DLL
+ ~Members() = default;
+
+ private:
+ Members();
+ Members(Members const&) = delete;
+
+ Handlers h;
+ };
+ PointerHolder<Members> m;
+};
+
+#endif // JSONHANDLER_HH