/** * @file * @author 2012-2013 Stefan Radomski (stefan.radomski@cs.tu-darmstadt.de) * @copyright Simplified BSD * * @cond * This program is free software: you can redistribute it and/or modify * it under the terms of the FreeBSD license as published by the FreeBSD * project. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. * * You should have received a copy of the FreeBSD license along with this * program. If not, see . * @endcond */ #include #include "uscxml/Common.h" #include "uscxml/util/Convenience.h" //#include "uscxml/util/UUID.h" #include "uscxml/util/DOM.h" //#include "uscxml/util/Convenience.h" #include #include #include #include "easylogging++.h" //#include //#include namespace uscxml { using namespace XERCESC_NS; std::ostream& operator<< (std::ostream& os, const DOMNode& node) { DOMImplementation *implementation = DOMImplementationRegistry::getDOMImplementation(X("LS")); DOMLSSerializer *serializer = ((DOMImplementationLS*)implementation)->createLSSerializer(); if (serializer->getDomConfig()->canSetParameter(XMLUni::fgDOMWRTFormatPrettyPrint, true)) serializer->getDomConfig()->setParameter(XMLUni::fgDOMWRTFormatPrettyPrint, true); serializer->setNewLine(XMLString::transcode("\r\n")); X output = serializer->writeToString(&node); os << output; return os; } std::ostream& operator<< (std::ostream& os, const X& xmlString) { os << xmlString.str(); return os; } std::string DOMUtils::idForNode(const DOMNode* node) { std::string nodeId; std::string seperator; const DOMNode* curr = node; while(curr) { switch (curr->getNodeType()) { case DOMNode::ELEMENT_NODE: { const DOMElement* elem = dynamic_cast(curr); if (HAS_ATTR(elem, "id")) { std::string elementId = ATTR(elem, "id"); std::replace( elementId.begin(), elementId.end(), '.', '_'); std::replace( elementId.begin(), elementId.end(), ',', '_'); nodeId.insert(0, elementId + seperator); seperator = "_"; return nodeId; } else { DOMNode* sibling = curr->getPreviousSibling(); int index = 0; while(sibling) { if (sibling->getNodeType() == DOMNode::ELEMENT_NODE) { if (iequals(TAGNAME_CAST(sibling), TAGNAME(elem))) { index++; } } sibling = sibling->getPreviousSibling(); } nodeId.insert(0, TAGNAME(elem) + toStr(index) + seperator); seperator = "_"; } break; } case DOMNode::DOCUMENT_NODE: return nodeId; default: break; } curr = curr->getParentNode(); } return nodeId; } std::string DOMUtils::xPathForNode(const DOMNode* node, const std::string& ns) { std::string xPath; std::string nsPrefix; if (ns.size() > 0) { nsPrefix = ns + ":"; } if (!node || node->getNodeType() != DOMNode::ELEMENT_NODE) return xPath; const DOMNode* curr = node; while(curr) { switch (curr->getNodeType()) { case DOMNode::ELEMENT_NODE: { const DOMElement* elem = dynamic_cast(curr); if (HAS_ATTR(elem, "id")) { // we assume ids to be unique and return immediately if (ns == "*") { xPath.insert(0, "//*[local-name() = \"" + TAGNAME(elem) + "\"][@id=\"" + ATTR(elem, "id") + "\"]"); } else { xPath.insert(0, "//" + nsPrefix + TAGNAME(elem) + "[@id=\"" + ATTR(elem, "id") + "\"]"); } return xPath; } else { // check previous siblings to count our index DOMNode* sibling = curr->getPreviousSibling(); int index = 1; // xpath indices start at 1 while(sibling) { if (sibling->getNodeType() == DOMNode::ELEMENT_NODE) { if (iequals(TAGNAME_CAST(sibling), TAGNAME(elem))) { index++; } } sibling = sibling->getPreviousSibling(); } if (ns == "*") { xPath.insert(0, "/*[local-name() = \"" + TAGNAME(elem) + "\"][" + toStr(index) + "]"); } else { xPath.insert(0, "/" + nsPrefix + TAGNAME(elem) + "[" + toStr(index) + "]"); } } break; } case DOMNode::DOCUMENT_NODE: return xPath; default: LOG(ERROR) << "Only nodes of type element supported for now"; return ""; break; } curr = curr->getParentNode(); } return xPath; } bool DOMUtils::hasIntersection(const std::list& l1, const std::list& l2) { for (auto i = l1.begin(); i != l1.end(); i++) { for (auto j = l2.begin(); j != l2.end(); j++) { if (*i == *j) return true; } } return false; } bool DOMUtils::isMember(const DOMNode* node, const DOMNodeList* list) { for (size_t i = 0; i < list->getLength(); i++) { if (list->item(i) == node) return true; } return false; } bool DOMUtils::isMember(const DOMNode* node, const std::list& list) { for (auto listIter = list.begin(); listIter != list.end(); listIter++) { if ((*listIter) == node) return true; } return false; } bool DOMUtils::isMember(const DOMElement* node, const std::list& list) { for (auto listIter = list.begin(); listIter != list.end(); listIter++) { if ((*listIter) == node) return true; } return false; } const DOMElement* DOMUtils::getNearestAncestor(const DOMNode* node, const std::string tagName) { const DOMNode* parent = node->getParentNode(); while(parent) { if (parent->getNodeType() == DOMNode::ELEMENT_NODE && iequals(TAGNAME_CAST(parent), tagName)) { return static_cast(parent); } parent = parent->getParentNode(); } return NULL; } bool DOMUtils::isDescendant(const DOMNode* s1, const DOMNode* s2) { if (!s1 || !s2) return false; const DOMNode* parent = s1->getParentNode(); while(parent) { if (s2 == parent) return true; parent = parent->getParentNode(); } return false; } std::list DOMUtils::inPostFixOrder(const std::set& elements, const DOMElement* root, const bool includeEmbeddedDoc) { std::list nodes; inPostFixOrder(elements, root, includeEmbeddedDoc, nodes); return nodes; } void DOMUtils::inPostFixOrder(const std::set& elements, const DOMElement* root, const bool includeEmbeddedDoc, std::list& nodes) { if (root == NULL) return; for (auto childElem = root->getFirstElementChild(); childElem; childElem = childElem->getNextElementSibling()) { if (!includeEmbeddedDoc && LOCALNAME(childElem) == "scxml") continue; inPostFixOrder(elements, childElem, includeEmbeddedDoc, nodes); } for (auto childElem = root->getFirstElementChild(); childElem; childElem = childElem->getNextElementSibling()) { if (!includeEmbeddedDoc && TAGNAME(childElem) == XML_PREFIX(root).str() + "scxml") continue; if (elements.find(TAGNAME(childElem)) != elements.end()) { nodes.push_back((DOMElement*)childElem); } } } std::list DOMUtils::inDocumentOrder(const std::set& elements, const DOMElement* root, const bool includeEmbeddedDoc) { std::list nodes; inDocumentOrder(elements, root, includeEmbeddedDoc, nodes); return nodes; } void DOMUtils::inDocumentOrder(const std::set& elements, const DOMElement* root, const bool includeEmbeddedDoc, std::list& nodes) { if (root == NULL) return; if (elements.find(TAGNAME(root)) != elements.end()) { nodes.push_back((DOMElement*)root); } /// @todo: item from getChildNodes is O(N)! DOMElement* child = root->getFirstElementChild(); while(child) { if (includeEmbeddedDoc || TAGNAME(child) != XML_PREFIX(root).str() + "scxml") { inDocumentOrder(elements, child, includeEmbeddedDoc, nodes); } child = child->getNextElementSibling(); } } std::list DOMUtils::getElementsByType(const DOMNode* root, DOMNode::NodeType type) { std::list result; std::list stack; std::list::iterator stackIter; if (!root) return result; stack.push_back((DOMNode*)root); while(stack.size() > 0) { // for(stackIter = stack.begin(); stackIter != stack.end(); stackIter++) { // std::cout << stackIter->getNodeType() << " " << stackIter->getLocalName() << " " << stackIter->getNodeValue() << std::endl; // } // std::cout << std::endl; DOMNode* currNode = stack.back(); if (currNode->hasChildNodes()) { stack.push_back(currNode->getFirstChild()); continue; } // roll back stack and pop everyone without next sibling do { currNode = stack.back(); if (currNode->getNodeType() == type) result.push_back(currNode); stack.pop_back(); if (currNode->getNextSibling()) { stack.push_back(currNode->getNextSibling()); break; } } while(stack.size() > 0); } return result; } std::list DOMUtils::filterChildElements(const std::string& tagName, const std::list& nodeSet, bool recurse) { std::list filteredChildElems; std::list::const_iterator nodeIter = nodeSet.begin(); while(nodeIter != nodeSet.end()) { std::list filtered = filterChildElements(tagName, *nodeIter, recurse); filteredChildElems.merge(filtered); // TODO: guess we want insert? nodeIter++; } return filteredChildElems; } std::list DOMUtils::filterChildElements(const std::string& tagName, const DOMElement* node, bool recurse) { std::list filteredChildElems; if (!node) return filteredChildElems; for (auto childElem = node->getFirstElementChild(); childElem; childElem = childElem->getNextElementSibling()) { // std::cerr << TAGNAME(childs.item(i)) << std::endl; if(iequals(TAGNAME(childElem), tagName)) { filteredChildElems.push_back((DOMElement*)childElem); } if (recurse) { std::list nested = filterChildElements(tagName, childElem, recurse); filteredChildElems.merge(nested); } } return filteredChildElems; } std::list DOMUtils::filterChildType(const DOMNode::NodeType type, const std::list& nodeSet, bool recurse) { std::list filteredChildType; std::list::const_iterator nodeIter = nodeSet.begin(); while(nodeIter != nodeSet.end()) { std::list filtered = filterChildType(type, *nodeIter, recurse); filteredChildType.merge(filtered); nodeIter++; } return filteredChildType; } std::list DOMUtils::filterChildType(const DOMNode::NodeType type, const DOMNode* node, bool recurse) { std::list filteredChildTypes; if (!node) return filteredChildTypes; for (auto child = node->getFirstChild(); child; child = child->getNextSibling()) { if (child->getNodeType() == type) filteredChildTypes.push_back(child); if (recurse) { std::list nested = filterChildType(type, child, recurse); filteredChildTypes.merge(nested); } } return filteredChildTypes; } }