#include "xml.h" #include "journal.h" #include "error.h" #include #include #include namespace ledger { namespace xml { document_t::document_t(node_t * _top, const char ** _builtins, const int _builtins_size) : builtins(_builtins), builtins_size(_builtins_size), top(new terminal_node_t(this)) {} void document_t::set_top(node_t * _top) { if (top) delete top; top = _top; } int document_t::register_name(const std::string& name) { int index = lookup_name_id(name); if (index != -1) return index; names.push_back(name); index = names.size() - 1; DEBUG_PRINT("xml.lookup", this << " Inserting name: " << names.back()); std::pair result = names_index.insert(names_pair(names.back(), index)); assert(result.second); return index + 1000; } int document_t::lookup_name_id(const std::string& name) const { if (builtins) { int first = 0; int last = builtins_size; while (first <= last) { int mid = (first + last) / 2; // compute mid point. int result; if ((result = (int)name[0] - (int)builtins[mid][0]) == 0) result = std::strcmp(name.c_str(), builtins[mid]); if (result > 0) first = mid + 1; // repeat search in top half. else if (result < 0) last = mid - 1; // repeat search in bottom half. else return mid; } } DEBUG_PRINT("xml.lookup", this << " Finding name: " << name); names_map::const_iterator i = names_index.find(name); if (i != names_index.end()) return (*i).second + 1000; return -1; } const char * document_t::lookup_name(int id) const { if (id < 1000) { switch (id) { case CURRENT: return "CURRENT"; case PARENT: return "PARENT"; case ROOT: return "ROOT"; case ALL: return "ALL"; default: assert(id >= 10); assert(builtins); return builtins[id - 10]; } } else { return names[id - 1000].c_str(); } } void document_t::write(std::ostream& out) const { if (top) { out << "\n"; top->write(out); } } #ifndef THREADSAFE document_t * node_t::document; #endif node_t::node_t(document_t * _document, parent_node_t * _parent, unsigned int _flags) : name_id(0), parent(_parent), next(NULL), prev(NULL), flags(_flags), info(NULL), attrs(NULL) { TRACE_CTOR("node_t(document_t *, node_t *)"); document = _document; if (! document->top) document->set_top(this); if (parent) parent->add_child(this); } void node_t::extract() { if (prev) prev->next = next; if (parent) { if (parent->_children == this) parent->_children = next; if (parent->_last_child == this) parent->_last_child = prev; parent = NULL; } if (next) next->prev = prev; next = NULL; prev = NULL; } void parent_node_t::clear() { node_t * child = _children; while (child) { node_t * tnext = child->next; delete child; child = tnext; } } void parent_node_t::add_child(node_t * node) { // It is important that this node is not called before children(), // otherwise, this node will not get auto-populated. if (_children == NULL) { assert(_last_child == NULL); _children = node; node->prev = NULL; } else { assert(_last_child != NULL); _last_child->next = node; node->prev = _last_child; } node->parent = this; while (node->next) { node_t * next_node = node->next; assert(next_node->prev == node); next_node->parent = this; node = next_node; } _last_child = node; } void parent_node_t::write(std::ostream& out, int depth) const { for (int i = 0; i < depth; i++) out << " "; out << '<' << name() << ">\n"; for (node_t * child = children(); child; child = child->next) child->write(out, depth + 1); for (int i = 0; i < depth; i++) out << " "; out << "\n"; } void terminal_node_t::write(std::ostream& out, int depth) const { for (int i = 0; i < depth; i++) out << " "; if (data.empty()) { out << '<' << name() << " />\n"; } else { out << '<' << name() << ">" << text() << "\n"; } } #if defined(HAVE_EXPAT) || defined(HAVE_XMLPARSE) template inline T * create_node(parser_t * parser) { T * node = new T(parser->document, parser->node_stack.empty() ? NULL : parser->node_stack.front()); node->set_name(parser->pending); node->attrs = parser->pending_attrs; parser->pending = NULL; parser->pending_attrs = NULL; return node; } static void startElement(void *userData, const char *name, const char **attrs) { parser_t * parser = static_cast(userData); DEBUG_PRINT("xml.parse", "startElement(" << name << ")"); if (parser->pending) { parent_node_t * node = create_node(parser); if (parser->node_stack.empty()) parser->document->top = node; parser->node_stack.push_front(node); } parser->pending = name; if (attrs) { for (const char ** p = attrs; *p; p += 2) { if (! parser->pending_attrs) parser->pending_attrs = new node_t::attrs_map; std::pair result = parser->pending_attrs->insert(node_t::attrs_pair(*p, *(p + 1))); assert(result.second); } } } static void endElement(void *userData, const char *name) { parser_t * parser = static_cast(userData); DEBUG_PRINT("xml.parse", "endElement(" << name << ")"); if (parser->pending) { terminal_node_t * node = create_node(parser); if (parser->node_stack.empty()) { parser->document->top = node; return; } } else if (! parser->handled_data) { assert(! parser->node_stack.empty()); parser->node_stack.pop_front(); } else { parser->handled_data = false; } } static void dataHandler(void *userData, const char *s, int len) { parser_t * parser = static_cast(userData); DEBUG_PRINT("xml.parse", "dataHandler(" << std::string(s, len) << ")"); bool all_whitespace = true; for (int i = 0; i < len; i++) { if (! std::isspace(s[i])) { all_whitespace = false; break; } } // jww (2006-09-28): I currently do not support text nodes within a // node that has children. if (! all_whitespace) { terminal_node_t * node = create_node(parser); node->set_text(std::string(s, len)); parser->handled_data = true; if (parser->node_stack.empty()) { parser->document->top = node; return; } } } bool parser_t::test(std::istream& in) const { char buf[80]; in.getline(buf, 79); if (std::strncmp(buf, " doc(new document_t(NULL, builtins, builtins_size)); document = doc.get(); parser = XML_ParserCreate(NULL); XML_SetElementHandler(parser, startElement, endElement); XML_SetCharacterDataHandler(parser, dataHandler); XML_SetUserData(parser, this); char buf[BUFSIZ]; while (! in.eof()) { in.getline(buf, BUFSIZ - 1); std::strcat(buf, "\n"); bool result; try { result = XML_Parse(parser, buf, std::strlen(buf), in.eof()); } catch (const std::exception& err) { //unsigned long line = XML_GetCurrentLineNumber(parser) - offset++; XML_ParserFree(parser); throw new parse_error(err.what()); } if (! have_error.empty()) { //unsigned long line = XML_GetCurrentLineNumber(parser) - offset++; parse_error err(have_error); std::cerr << "Error: " << err.what() << std::endl; have_error = ""; } if (! result) { //unsigned long line = XML_GetCurrentLineNumber(parser) - offset++; const char * err = XML_ErrorString(XML_GetErrorCode(parser)); XML_ParserFree(parser); throw new parse_error(err); } } XML_ParserFree(parser); document = NULL; return doc.release(); } node_t * commodity_node_t::children() const { // jww (2007-04-19): Need to report the commodity and its details return NULL; } node_t * amount_node_t::children() const { // jww (2007-04-19): Need to report the quantity and commodity return NULL; } node_t * transaction_node_t::children() const { if (! _children) { terminal_node_t * account_node = new terminal_node_t(document, const_cast(this)); account_node->set_name("account"); account_node->set_text(transaction->account->fullname()); } return parent_node_t::children(); } node_t * transaction_node_t::lookup_child(int _name_id) { if (_name_id == payee_id) { payee_virtual_node = new terminal_node_t(document); payee_virtual_node->set_text(transaction->entry->payee); return payee_virtual_node; } } value_t transaction_node_t::to_value() const { return transaction->amount; } node_t * entry_node_t::children() const { if (! _children) { if (! entry->code.empty()) { terminal_node_t * code_node = new terminal_node_t(document, const_cast(this)); code_node->set_name("code"); code_node->set_text(entry->code); } if (! entry->payee.empty()) { terminal_node_t * payee_node = new terminal_node_t(document, const_cast(this)); payee_node->set_name("payee"); payee_node->set_text(entry->payee); } for (transactions_list::iterator i = entry->transactions.begin(); i != entry->transactions.end(); i++) new transaction_node_t(document, *i, const_cast(this)); } return parent_node_t::children(); } node_t * account_node_t::children() const { if (! _children) { if (! account->name.empty()) { terminal_node_t * name_node = new terminal_node_t(document, const_cast(this)); name_node->set_name("name"); name_node->set_text(account->name); } if (! account->note.empty()) { terminal_node_t * note_node = new terminal_node_t(document, const_cast(this)); note_node->set_name("note"); note_node->set_text(account->note); } for (accounts_map::iterator i = account->accounts.begin(); i != account->accounts.end(); i++) new account_node_t(document, (*i).second, const_cast(this)); } return parent_node_t::children(); } node_t * journal_node_t::children() const { if (! _children) { #if 0 account_node_t * master_account = new account_node_t(document, journal->master, const_cast(this)); #endif parent_node_t * entries = new parent_node_t(document, const_cast(this)); entries->set_name("entries"); for (entries_list::iterator i = journal->entries.begin(); i != journal->entries.end(); i++) new entry_node_t(document, *i, const_cast(this)); } return parent_node_t::children(); } #endif // defined(HAVE_EXPAT) || defined(HAVE_XMLPARSE) void output_xml_string(std::ostream& out, const std::string& str) { for (const char * s = str.c_str(); *s; s++) { switch (*s) { case '<': out << "<"; break; case '>': out << "&rt;"; break; case '&': out << "&"; break; default: out << *s; break; } } } } // namespace xml } // namespace ledger