#ifndef _BUILDER_H #define _BUILDER_H #include "document.h" namespace ledger { namespace xml { /** * @class builder_t * * @brief Represents an interface for building a data hierarchy. * * This interface is much like .NET's XmlWriter facility. It * abstracts the kind of hierarchy we're building, instead focusing * only on the relationships. */ class builder_t { public: struct position_t { typedef uint_least32_t file_pos_t; typedef uint_least32_t file_line_t; file_pos_t offset; file_line_t linenum; position_t() : offset(0), linenum(0) {} explicit position_t(file_pos_t _offset, file_line_t _linenum) : offset(_offset), linenum(_linenum) {} }; protected: position_t current_position; public: virtual ~builder_t() {} virtual void set_start_position(std::istream& in) {} virtual void set_position(const position_t& position) {} virtual position_t& position() { return current_position; } virtual void push_attr(const string& name, const string& value) = 0; virtual void push_attr(const node_t::nameid_t name_id, const string& value) = 0; virtual void begin_node(const string& name, bool terminal = false) = 0; virtual void begin_node(const node_t::nameid_t name_id, bool terminal = false) = 0; virtual void push_node(const string& name, const optional& end_pos = none) = 0; virtual void push_node(const node_t::nameid_t name_id, const optional& end_pos = none) = 0; virtual node_t * current_node() = 0; virtual void append_text(const string& text) = 0; virtual node_t * end_node(const string& name, const optional& end_pos = none) = 0; virtual node_t * end_node(const node_t::nameid_t name_id, const optional& end_pos = none) = 0; }; /** * @class xml_builder_t * * @brief Build a generic node_t hierarchy. * * This builder can be used to parse ordinary XML into a document * object structure which can then be traversed in memory. */ class document_builder_t : public builder_t { typedef std::list > attrs_list; document_t& document_; attrs_list current_attrs; node_t * current; string current_text; public: document_builder_t(document_t& _document) : document_(_document), current(&document_) {} virtual void push_attr(const string& name, const string& value) { push_attr(document().register_name(name), value); } virtual void push_attr(const node_t::nameid_t name_id, const string& value) { current_attrs.push_back(attrs_list::value_type(name_id, value.c_str())); } virtual void begin_node(const string& name, bool terminal = false) { begin_node(document().register_name(name), terminal); } virtual void begin_node(const node_t::nameid_t name_id, bool terminal = false) { if (terminal) current = current->as_parent_node().create_child(name_id); else current = current->as_parent_node().create_child(name_id); foreach (const attrs_list::value_type& pair, current_attrs) current->set_attr(pair.first, pair.second.c_str()); current_attrs.clear(); } virtual void push_node(const string& name, const optional& end_pos = none) { begin_node(name, true); end_node(name, end_pos); } virtual void push_node(const node_t::nameid_t name_id, const optional& end_pos = none) { begin_node(name_id, true); end_node(name_id, end_pos); } virtual document_t& document() { return document_; } virtual node_t * current_node() { return current; } virtual void append_text(const string& text) { assert(! current->is_parent_node()); polymorphic_downcast(current)->set_text(text); } virtual node_t * end_node(const string& name, const optional& end_pos = none) { return current = &*current->parent(); } virtual node_t * end_node(const node_t::nameid_t name_id, const optional& end_pos = none) { return current = &*current->parent(); } }; /** * @class journal_builder_t * * @brief This custom builder creates an XML-mirrored Ledger journal. * * Rather than simply creating a node_t hierarchy, as xml_builder_t * does, this code creates the associated journal elements referred to * by those nodes, and then refers to those elements via minimalist * "shadow nodes". * * Thus, after building a element, the element itself * will have no children, but instead will point to a transaction_t * object. If later an XPath expression desires to traverse the * element, all of the appropriate child nodes will be * constructed on the fly, as if they'd been created in the first * place by a regular xml_builder_t. */ class journal_builder_t : public document_builder_t { public: virtual void set_start_position(std::istream& in) { set_position(position_t(in.tellg(), 1)); } virtual void set_position(const position_t& position) { current_position = position; } }; /** * @class xml_writer_t * * @brief Create textual XML on the given output stream. * * This builder, rather than manipulating data structures in memory, * simply streams its contents on the fly to the given output stream. * It uses only enough memory to remember the currently push * attributes and text. */ class xml_writer_t : public builder_t { typedef std::list > attrs_list; attrs_list current_attrs; std::ostream& outs; public: xml_writer_t(std::ostream& _outs) : outs(_outs) { outs << ""; begin_node("ledger"); } ~xml_writer_t() { end_node("ledger"); } virtual void push_attr(const string& name, const string& value) { current_attrs.push_back(attrs_list::value_type(name, value)); } virtual void push_attr(const node_t::nameid_t name_id, const string& value) { push_attr("hello", value); } virtual void begin_node(const string& name, bool terminal = false) { outs << '<' << name; foreach (const attrs_list::value_type& attr, current_attrs) outs << ' ' << attr.first << "=\"" << attr.second << "\""; current_attrs.clear(); outs << '>'; } virtual void begin_node(const node_t::nameid_t name_id, bool terminal = false) { begin_node("hello"); } virtual void push_node(const string& name, const optional& end_pos = none) { begin_node(name, true); end_node(name, end_pos); } virtual void push_node(const node_t::nameid_t name_id, const optional& end_pos = none) { push_node("hello", end_pos); } virtual node_t * current_node() { return NULL; } virtual void append_text(const string& text) { outs << text; } virtual node_t * end_node(const string& name, const optional& end_pos = none) { outs << "'; return NULL; } virtual node_t * end_node(const node_t::nameid_t name_id, const optional& end_pos = none) { end_node("hello", end_pos); return NULL; } }; } // namespace xml } // namespace ledger #endif // _BUILDER_H