summaryrefslogtreecommitdiff
path: root/src/data/node.h
diff options
context:
space:
mode:
Diffstat (limited to 'src/data/node.h')
-rw-r--r--src/data/node.h352
1 files changed, 352 insertions, 0 deletions
diff --git a/src/data/node.h b/src/data/node.h
new file mode 100644
index 00000000..b0324ca0
--- /dev/null
+++ b/src/data/node.h
@@ -0,0 +1,352 @@
+/*
+ * Copyright (c) 2003-2007, John Wiegley. All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * - Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ *
+ * - Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * - Neither the name of New Artisans LLC nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+#ifndef _NODE_H
+#define _NODE_H
+
+#include "value.h"
+
+namespace ledger {
+namespace xml {
+
+#define XML_NODE_IS_PARENT 0x1
+
+DECLARE_EXCEPTION(conversion_error);
+
+class document_t;
+class parent_node_t;
+class terminal_node_t;
+
+class node_t : public supports_flags<>, public noncopyable
+{
+public:
+ typedef uint_fast16_t nameid_t;
+
+ // This has to be public so that multi_index_container can reference
+ // it in parent_node_t.
+ nameid_t name_id_;
+
+protected:
+ document_t& document_;
+ optional<parent_node_t&> parent_;
+
+#if 1
+ typedef std::map<const nameid_t, value_t> attributes_t;
+ typedef std::pair<const nameid_t, value_t> attr_pair;
+#else
+ typedef std::pair<nameid_t, value_t> attr_pair;
+
+ typedef multi_index_container<
+ attr_pair,
+ multi_index::indexed_by<
+ multi_index::sequenced<>,
+ multi_index::hashed_unique<
+ multi_index::member<attr_pair, nameid_t, &attr_pair::first> >
+ >
+ > attributes_t;
+
+ typedef attributes_t::nth_index<0>::type attributes_by_order;
+ typedef attributes_t::nth_index<1>::type attributes_hashed;
+#endif
+
+ optional<attributes_t> attributes;
+
+public:
+ bool compiled; // so that compile_node() can access it
+
+ node_t(nameid_t _name_id, document_t& _document,
+ const optional<parent_node_t&>& _parent = none, flags_t _flags = 0)
+ : supports_flags<>(_flags), name_id_(_name_id),
+ document_(_document), parent_(_parent) {
+ TRACE_CTOR(node_t, "document_t&, parent_node_t&, nameid_t, flags_t");
+ }
+
+ virtual ~node_t() {
+ TRACE_DTOR(node_t);
+ }
+
+ void extract();
+
+ bool is_compiled() const {
+ return compiled;
+ }
+
+ bool is_parent_node() const {
+ return has_flags(XML_NODE_IS_PARENT);
+ }
+ parent_node_t& as_parent_node() {
+ if (! is_parent_node())
+ throw_(std::logic_error, "Request to cast leaf node to a parent node");
+ return downcast<parent_node_t>(*this);
+ }
+ const parent_node_t& as_parent_node() const {
+ return const_cast<node_t *>(this)->as_parent_node();
+ }
+
+ bool is_terminal_node() const {
+ return ! has_flags(XML_NODE_IS_PARENT);
+ }
+ terminal_node_t& as_terminal_node() {
+ if (! is_terminal_node())
+ throw_(std::logic_error, "Request to cast parent node to a leaf node");
+ return downcast<terminal_node_t>(*this);
+ }
+ const terminal_node_t& as_terminal_node() const {
+ return const_cast<node_t *>(this)->as_terminal_node();
+ }
+
+ virtual value_t to_value() const = 0;
+ virtual void print(std::ostream& out) const = 0;
+ virtual void print_attributes(std::ostream& out) const;
+
+ const char * name() const;
+ nameid_t name_id() const {
+ return name_id_;
+ }
+
+ document_t& document() const {
+ return document_;
+ }
+ optional<parent_node_t&> parent() const {
+ return parent_;
+ }
+
+ value_t& set_attr(const string& _name, const char * value);
+ value_t& set_attr(const string& _name, const value_t& value);
+
+ value_t& set_attr(const nameid_t _name_id, const char * value) {
+ return set_attr(_name_id, string_value(value));
+ }
+ value_t& set_attr(const nameid_t _name_id, const value_t& value) {
+ if (! attributes)
+ attributes = attributes_t();
+
+ attributes_t::iterator i = attributes->find(_name_id);
+ if (i == attributes->end()) {
+ std::pair<attributes_t::iterator, bool> result =
+ attributes->insert(attr_pair(_name_id, value));
+ assert(result.second);
+ return (*result.first).second;
+ } else {
+ i->second = value;
+ return i->second;
+ }
+ }
+
+ optional<value_t&> get_attr(const string& _name);
+ optional<value_t&> get_attr(const nameid_t _name_id) {
+ if (attributes) {
+#if 1
+ attributes_t::iterator i = attributes->find(_name_id);
+ if (i != attributes->end())
+ return (*i).second;
+#else
+ typedef attributes_t::nth_index<1>::type attributes_by_name;
+
+ const attributes_by_name& name_index = attributes->get<1>();
+ attributes_by_name::iterator i = name_index.find(_name_id);
+ if (i != name_index.end())
+ return (*i).second;
+#endif
+ }
+ return none;
+ }
+
+ optional<const value_t&> get_attr(const string& _name) const {
+ if (optional<value_t&> value =
+ const_cast<node_t *>(this)->get_attr(_name))
+ return *value;
+ else
+ return none;
+ }
+ optional<const value_t&> get_attr(const nameid_t _name_id) const {
+ if (optional<value_t&> value =
+ const_cast<node_t *>(this)->get_attr(_name_id))
+ return *value;
+ else
+ return none;
+ }
+};
+
+class parent_node_t : public node_t
+{
+ typedef multi_index_container<
+ node_t *,
+ multi_index::indexed_by<
+ multi_index::sequenced<>,
+ multi_index::hashed_non_unique<
+ multi_index::member<node_t, nameid_t, &node_t::name_id_> >,
+ multi_index::hashed_unique<multi_index::identity<node_t *> >
+ >
+ > children_t;
+
+ children_t children;
+
+public:
+ typedef children_t::nth_index<0>::type children_by_order;
+ typedef children_t::nth_index<1>::type children_by_nameid;
+ typedef children_t::nth_index<2>::type children_by_ptr;
+
+ parent_node_t(nameid_t _name_id, document_t& _document,
+ const optional<parent_node_t&>& _parent = none)
+ : node_t(_name_id, _document, _parent, XML_NODE_IS_PARENT) {
+ TRACE_CTOR(parent_node_t, "document_t *, parent_node_t *");
+ }
+ virtual ~parent_node_t() {
+ TRACE_DTOR(parent_node_t);
+ clear_children();
+ }
+
+ template <typename T>
+ T * create_child(nameid_t _name_id) {
+ T * child = new T(_name_id, document(), *this);
+ children.push_back(child);
+ return child;
+ }
+
+ void remove_child(node_t * child) {
+ children_by_ptr& ptr_index = children.get<2>();
+ children_by_ptr::iterator i = ptr_index.find(child);
+ if (i == ptr_index.end())
+ throw_(std::logic_error, "Request to delete node which is not a child");
+ ptr_index.erase(i);
+ }
+
+ void delete_child(node_t * child) {
+ remove_child(child);
+ checked_delete(child);
+ }
+
+ struct match_nameid {
+ nameid_t nameid;
+ match_nameid(nameid_t _nameid) : nameid(_nameid) {}
+ bool operator()(const node_t * node) const {
+ return node->name_id() == nameid;
+ }
+ };
+
+ typedef children_by_order::iterator iterator;
+ typedef children_by_order::iterator const_iterator;
+
+ children_by_order::iterator begin() {
+ return children.get<0>().begin();
+ }
+ children_by_order::const_iterator begin() const {
+ return children.get<0>().begin();
+ }
+ children_by_order::iterator end() {
+ return children.get<0>().end();
+ }
+ children_by_order::const_iterator end() const {
+ return children.get<0>().end();
+ }
+
+ std::size_t size() const {
+ return children.get<0>().size();
+ }
+
+ children_by_nameid::iterator begin(nameid_t _name_id) {
+ return std::find_if(children.get<1>().begin(),
+ children.get<1>().end(), match_nameid(_name_id));
+ }
+ children_by_nameid::const_iterator begin(nameid_t _name_id) const {
+ return std::find_if(children.get<1>().begin(),
+ children.get<1>().end(), match_nameid(_name_id));
+ }
+ children_by_nameid::iterator end(nameid_t) {
+ return children.get<1>().end();
+ }
+ children_by_nameid::const_iterator end(nameid_t) const {
+ return children.get<1>().end();
+ }
+
+ void clear_children() {
+ typedef children_t::nth_index<0>::type children_by_index;
+
+ children_by_index& child_index = children.get<0>();
+ for (children_by_index::iterator i = child_index.begin();
+ i != child_index.end();
+ i++)
+ checked_delete(*i);
+
+ children.clear();
+ }
+
+ virtual value_t to_value() const {
+ throw_(std::logic_error, "Cannot convert parent node to a value");
+ return NULL_VALUE;
+ }
+
+ void print(std::ostream& out) const;
+};
+
+inline void node_t::extract()
+{
+ if (parent_)
+ parent_->remove_child(this);
+}
+
+class terminal_node_t : public node_t
+{
+ string data;
+
+public:
+ terminal_node_t(nameid_t _name_id, document_t& _document,
+ const optional<parent_node_t&>& _parent = none)
+ : node_t(_name_id, _document, _parent)
+ {
+ TRACE_CTOR(terminal_node_t, "document_t *, parent_node_t *");
+ }
+ virtual ~terminal_node_t() {
+ TRACE_DTOR(terminal_node_t);
+ }
+
+ const char * text() const {
+ return data.c_str();
+ }
+ void set_text(const string& _data) {
+ data = _data;
+ }
+ void set_text(const char * _data) {
+ data = _data;
+ }
+
+ virtual value_t to_value() const {
+ return value_t(text(), true);
+ }
+
+ void print(std::ostream& out) const;
+};
+
+} // namespace xml
+} // namespace ledger
+
+#endif // _NODE_H