X-Git-Url: http://www.project-moonshot.org/gitweb/?a=blobdiff_plain;f=xmltooling%2FAbstractDOMCachingXMLObject.cpp;h=061df1a0373997d464516b00a9453dd3163a6ba0;hb=d1c1478d26cfd9b87a4deb80c3638dc42e111735;hp=8a23062a7f031b145bb72199600c4397b5791943;hpb=5b17688631ba903af821a15221614b4423bb1dc9;p=shibboleth%2Fcpp-xmltooling.git diff --git a/xmltooling/AbstractDOMCachingXMLObject.cpp b/xmltooling/AbstractDOMCachingXMLObject.cpp index 8a23062..061df1a 100644 --- a/xmltooling/AbstractDOMCachingXMLObject.cpp +++ b/xmltooling/AbstractDOMCachingXMLObject.cpp @@ -1,133 +1,193 @@ -/* - * Copyright 2001-2006 Internet2 - * - * 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. - */ - -/** - * AbstractDOMCachingXMLObject.cpp - * - * Extension of AbstractXMLObject that implements a DOMCachingXMLObject. - */ - -#include "internal.h" -#include "exceptions.h" -#include "AbstractDOMCachingXMLObject.h" - -#include -#include -#include - -using namespace xmltooling; -using namespace log4cpp; -using namespace std; - -AbstractDOMCachingXMLObject::~AbstractDOMCachingXMLObject() -{ - if (m_document) - m_document->release(); -} - -void AbstractDOMCachingXMLObject::setDOM(DOMElement* dom, bool bindDocument) -{ - m_dom=dom; - if (dom) { - if (bindDocument) { - DOMDocument* tmp=setDocument(dom->getOwnerDocument()); - if (tmp) - tmp->release(); - } - } - else if (m_document) { - m_document->release(); - m_document=NULL; - } -} - -void AbstractDOMCachingXMLObject::releaseDOM() -{ - Category& log=Category::getInstance(XMLTOOLING_LOGCAT".DOM"); - if (log.isDebugEnabled()) - log.debug("releasing cached DOM reprsentation for %s", getElementQName().toString().c_str()); - setDOM(NULL); -} - -void AbstractDOMCachingXMLObject::releaseParentDOM(bool propagateRelease) -{ - Category& log=Category::getInstance(XMLTOOLING_LOGCAT".DOM"); - if (log.isDebugEnabled()) { - log.debug( - "releasing cached DOM representation for parent of %s with propagation set to %s", - getElementQName().toString().c_str(), propagateRelease ? "true" : "false" - ); - } - - DOMCachingXMLObject* domCachingParent = dynamic_cast(getParent()); - if (domCachingParent) { - domCachingParent->releaseDOM(); - if (propagateRelease) - domCachingParent->releaseParentDOM(propagateRelease); - } -} - -class _release : public binary_function { -public: - void operator()(XMLObject* obj, bool propagate) const { - DOMCachingXMLObject* domCaching = dynamic_cast(obj); - if (domCaching) { - domCaching->releaseDOM(); - if (propagate) - domCaching->releaseChildrenDOM(propagate); - } - } -}; - -void AbstractDOMCachingXMLObject::releaseChildrenDOM(bool propagateRelease) -{ - Category& log=Category::getInstance(XMLTOOLING_LOGCAT".DOM"); - if (log.isDebugEnabled()) { - log.debug( - "releasing cached DOM representation for children of %s with propagation set to %s", - getElementQName().toString().c_str(), propagateRelease ? "true" : "false" - ); - } - - vector children; - if (getOrderedChildren(children)) - for_each(children.begin(),children.end(),bind2nd(_release(),propagateRelease)); -} - -XMLObject* AbstractDOMCachingXMLObject::prepareForAssignment(const XMLObject* oldValue, XMLObject* newValue) { - - if (newValue && newValue->hasParent()) - throw XMLObjectException("child XMLObject cannot be added - it is already the child of another XMLObject"); - - if (!oldValue) { - if (newValue) { - releaseThisandParentDOM(); - newValue->setParent(this); - return newValue; - } - else { - return NULL; - } - } - - if (oldValue != newValue) { - delete oldValue; - releaseThisandParentDOM(); - newValue->setParent(this); - } - - return newValue; -} +/** + * Licensed to the University Corporation for Advanced Internet + * Development, Inc. (UCAID) under one or more contributor license + * agreements. See the NOTICE file distributed with this work for + * additional information regarding copyright ownership. + * + * UCAID licenses this file to you 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. + */ + +/** + * AbstractDOMCachingXMLObject.cpp + * + * Extension of AbstractXMLObject that implements a DOMCachingXMLObject. + */ + +#include "internal.h" +#include "AbstractDOMCachingXMLObject.h" +#include "exceptions.h" +#include "XMLObjectBuilder.h" +#include "util/XMLHelper.h" + +#include +#include + +using namespace xmltooling; +using namespace xercesc; +using namespace std; + +AbstractDOMCachingXMLObject::AbstractDOMCachingXMLObject() : m_dom(nullptr), m_document(nullptr) +{ +} + +AbstractDOMCachingXMLObject::AbstractDOMCachingXMLObject(const AbstractDOMCachingXMLObject& src) + : AbstractXMLObject(src), m_dom(nullptr), m_document(nullptr) +{ +} + +AbstractDOMCachingXMLObject::~AbstractDOMCachingXMLObject() +{ + if (m_document) + m_document->release(); +} + +DOMElement* AbstractDOMCachingXMLObject::getDOM() const +{ + return m_dom; +} + +void AbstractDOMCachingXMLObject::setDOM(DOMElement* dom, bool bindDocument) const +{ + m_dom = dom; + if (dom && bindDocument) { + DOMDocument* doc = dom->getOwnerDocument(); + setDocument(doc); + DOMElement* documentRoot = doc->getDocumentElement(); + if (!documentRoot) + doc->appendChild(dom); + else if (documentRoot != dom) + doc->replaceChild(dom, documentRoot); + } +} + +void AbstractDOMCachingXMLObject::setDocument(DOMDocument* doc) const +{ + if (m_document != doc) { + if (m_document) + m_document->release(); + m_document=doc; + } +} + +void AbstractDOMCachingXMLObject::releaseDOM() const +{ + if (m_dom) { + if (m_log.isDebugEnabled()) { + string qname=getElementQName().toString(); + m_log.debug("releasing cached DOM representation for (%s)", qname.empty() ? "unknown" : qname.c_str()); + } + setDOM(nullptr); + } +} + +void AbstractDOMCachingXMLObject::releaseParentDOM(bool propagateRelease) const +{ + if (getParent() && getParent()->getDOM()) { + m_log.debug( + "releasing cached DOM representation for parent object with propagation set to %s", + propagateRelease ? "true" : "false" + ); + getParent()->releaseDOM(); + if (propagateRelease) + getParent()->releaseParentDOM(propagateRelease); + } +} + +namespace { + class _release : public binary_function { + public: + void operator()(XMLObject* obj, bool propagate) const { + if (obj) { + obj->releaseDOM(); + if (propagate) + obj->releaseChildrenDOM(propagate); + } + } + }; +}; + +void AbstractDOMCachingXMLObject::releaseChildrenDOM(bool propagateRelease) const +{ + if (hasChildren()) { + m_log.debug( + "releasing cached DOM representation for children with propagation set to %s", + propagateRelease ? "true" : "false" + ); + const list& children=getOrderedChildren(); + for_each(children.begin(),children.end(),bind2nd(_release(),propagateRelease)); + } +} + +DOMElement* AbstractDOMCachingXMLObject::cloneDOM(DOMDocument* doc) const +{ + if (getDOM()) { + DOMDocument* cloneDoc = doc; + if (!cloneDoc) + cloneDoc=DOMImplementationRegistry::getDOMImplementation(nullptr)->createDocument(); + try { + return static_cast(cloneDoc->importNode(getDOM(),true)); + } + catch (XMLException& ex) { + if (!doc) + cloneDoc->release(); + auto_ptr_char temp(ex.getMessage()); + m_log.error("DOM clone failed: %s", temp.get()); + } + } + return nullptr; +} + +XMLObject* AbstractDOMCachingXMLObject::clone() const +{ + // See if we can clone via the DOM. + DOMElement* domCopy=cloneDOM(); + if (domCopy) { + // Seemed to work, so now we unmarshall the DOM to produce the clone. + const XMLObjectBuilder* b=XMLObjectBuilder::getBuilder(domCopy); + if (!b) { + auto_ptr q(XMLHelper::getNodeQName(domCopy)); + m_log.error( + "DOM clone failed, unable to locate builder for element (%s)", q->toString().c_str() + ); + domCopy->getOwnerDocument()->release(); + throw UnmarshallingException("Unable to locate builder for cloned element."); + } + XercesJanitor janitor(domCopy->getOwnerDocument()); + XMLObject* ret = b->buildFromElement(domCopy,true); // bind document + janitor.release(); // safely transferred + return ret; + } + return nullptr; +} + +void AbstractDOMCachingXMLObject::detach() +{ + // This is an override that duplicates some of the checking in the base class but + // adds document management in preparation for deletion of the parent. + + if (!getParent()) + return; + + if (getParent()->hasParent()) + throw XMLObjectException("Cannot detach an object whose parent is itself a child."); + + AbstractDOMCachingXMLObject* parent = dynamic_cast(getParent()); + if (parent && parent->m_document) { + // Transfer control of document to me... + setDocument(parent->m_document); + parent->m_document = nullptr; + } + // The rest is done by the base. + AbstractXMLObject::detach(); +}