Set fourth file version digit to signify rebuild.
[shibboleth/cpp-xmltooling.git] / xmltoolingtest / XMLObjectBaseTestCase.h
index 68c3b3e..c51a4ff 100644 (file)
@@ -1,37 +1,48 @@
-/*
- *  Copyright 2001-2005 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
+/**
+ * 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.
  *
- *     http://www.apache.org/licenses/LICENSE-2.0
+ * 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
  *
- * 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.
+ * 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.
  */
 
 #include <cxxtest/TestSuite.h>
 #include <xmltooling/AbstractAttributeExtensibleXMLObject.h>
-#include <xmltooling/AbstractDOMCachingXMLObject.h>
-#include <xmltooling/AbstractExtensibleXMLObject.h>
+#include <xmltooling/AbstractComplexElement.h>
+#include <xmltooling/ElementProxy.h>
 #include <xmltooling/exceptions.h>
 #include <xmltooling/XMLObjectBuilder.h>
 #include <xmltooling/XMLToolingConfig.h>
 #include <xmltooling/io/AbstractXMLObjectMarshaller.h>
 #include <xmltooling/io/AbstractXMLObjectUnmarshaller.h>
+#include <xmltooling/impl/AnyElement.h>
+#include <xmltooling/impl/UnknownElement.h>
 #include <xmltooling/util/ParserPool.h>
-#include <xmltooling/util/XMLObjectChildrenList.h>
+#include <xmltooling/util/XMLConstants.h>
 #include <xmltooling/util/XMLHelper.h>
+#include <xmltooling/util/XMLObjectChildrenList.h>
+
+#ifndef XMLTOOLING_NO_XMLSEC
+    #include <xmltooling/signature/Signature.h>
+    using namespace xmlsignature;
+#endif
 
 using namespace xmltooling;
+using namespace xercesc;
 using namespace std;
 
-extern ParserPool* validatingPool;
-extern ParserPool* nonvalidatingPool;
 extern string data_path;
 
 #if defined (_MSC_VER)
@@ -39,31 +50,49 @@ extern string data_path;
     #pragma warning( disable : 4250 4251 )
 #endif
 
-class SimpleXMLObject : public AbstractDOMCachingXMLObject
+class SimpleXMLObject
+    : public AbstractAttributeExtensibleXMLObject,
+        public AbstractComplexElement,
+        public AbstractDOMCachingXMLObject,
+        public AbstractXMLObjectMarshaller,
+        public AbstractXMLObjectUnmarshaller
 {
+protected:
+    SimpleXMLObject(const SimpleXMLObject& src)
+            : AbstractXMLObject(src), AbstractAttributeExtensibleXMLObject(src), AbstractComplexElement(src), AbstractDOMCachingXMLObject(src),
+                m_id(XMLString::replicate(src.m_id)) {
+#ifndef XMLTOOLING_NO_XMLSEC
+        m_children.push_back(nullptr);
+        m_signature=m_children.begin();
+#endif
+        VectorOf(SimpleXMLObject) mine=getSimpleXMLObjects();
+        for (vector<SimpleXMLObject*>::const_iterator i=src.m_simples.begin(); i!=src.m_simples.end(); i++) {
+            mine.push_back(dynamic_cast<SimpleXMLObject*>((*i)->clone()));
+        }
+    }
+
 public:
     static const XMLCh NAMESPACE[];
     static const XMLCh NAMESPACE_PREFIX[];
     static const XMLCh LOCAL_NAME[];
+    static const XMLCh DERIVED_NAME[];
+    static const XMLCh TYPE_NAME[];
     static const XMLCh ID_ATTRIB_NAME[];
 
-    SimpleXMLObject() : AbstractDOMCachingXMLObject(NAMESPACE, LOCAL_NAME, NAMESPACE_PREFIX), m_id(NULL), m_value(NULL) {}
+    SimpleXMLObject(
+        const XMLCh* nsURI=nullptr, const XMLCh* localName=nullptr, const XMLCh* prefix=nullptr, const xmltooling::QName* schemaType=nullptr
+        ) : AbstractXMLObject(nsURI, localName, prefix, schemaType), m_id(nullptr) {
+#ifndef XMLTOOLING_NO_XMLSEC
+        m_children.push_back(nullptr);
+        m_signature=m_children.begin();
+#endif
+    }
+
     virtual ~SimpleXMLObject() {
         XMLString::release(&m_id);
-        XMLString::release(&m_value);
     }
-    
-    const XMLCh* getId() const { return m_id; }
-    void setId(const XMLCh* id) { m_id=prepareForAssignment(m_id,id); }
 
-    const XMLCh* getValue() const { return m_value; }
-    void setValue(const XMLCh* value) { m_value=prepareForAssignment(m_value,value); }
-    
-    VectorOf(SimpleXMLObject) getSimpleXMLObjects() {
-        return VectorOf(SimpleXMLObject)(this, m_simples, &m_children, m_children.end());
-    }
-    
-    SimpleXMLObject* clone() const {
+    XMLObject* clone() const {
         auto_ptr<XMLObject> domClone(AbstractDOMCachingXMLObject::clone());
         SimpleXMLObject* ret=dynamic_cast<SimpleXMLObject*>(domClone.get());
         if (ret) {
@@ -71,191 +100,102 @@ public:
             return ret;
         }
 
-        ret=new SimpleXMLObject();
-        ret->setId(m_id);
-        ret->setValue(m_value);
-        xmltooling::clone(m_children, ret->m_children);
-        return ret;
+        return new SimpleXMLObject(*this);
     }
 
-private:
-    XMLCh* m_id;
-    XMLCh* m_value;
-    vector<SimpleXMLObject*> m_simples;
-};
-
-class SimpleXMLObjectBuilder : public XMLObjectBuilder
-{
-public:
-    SimpleXMLObject* buildObject() const {
-        return new SimpleXMLObject();
-    }
-};
+    const XMLCh* getXMLID() const { return getId(); }
+    const XMLCh* getId() const { return m_id; }
+    void setId(const XMLCh* id) { m_id=prepareForAssignment(m_id,id); }
 
-class SimpleXMLObjectMarshaller : public AbstractXMLObjectMarshaller
-{
-public:
-    SimpleXMLObjectMarshaller() {}
+    const XMLCh* getValue() const { return getTextContent(); }
+    void setValue(const XMLCh* value) { setTextContent(value); }
 
-private:
-    void marshallAttributes(const XMLObject& xmlObject, DOMElement* domElement) const {
-        const SimpleXMLObject& simpleXMLObject = dynamic_cast<const SimpleXMLObject&>(xmlObject);
-        
-        if(simpleXMLObject.getId()) {
-            domElement->setAttributeNS(NULL, SimpleXMLObject::ID_ATTRIB_NAME, simpleXMLObject.getId());
-            domElement->setIdAttributeNS(NULL, SimpleXMLObject::ID_ATTRIB_NAME);
-        }
+#ifndef XMLTOOLING_NO_XMLSEC    
+    Signature* getSignature() const {
+        return dynamic_cast<Signature*>(*m_signature);
     }
 
-    void marshallElementContent(const XMLObject& xmlObject, DOMElement* domElement) const {
-        const SimpleXMLObject& simpleXMLObject = dynamic_cast<const SimpleXMLObject&>(xmlObject);
-
-        if(simpleXMLObject.getValue()) {
-            domElement->setTextContent(simpleXMLObject.getValue());
-        }
+    void setSignature(Signature* sig) {
+        *m_signature=prepareForAssignment(*m_signature,sig);
     }
-};
-
-class SimpleXMLObjectUnmarshaller : public AbstractXMLObjectUnmarshaller
-{
-public:
-    SimpleXMLObjectUnmarshaller() {}
-
-private:
-    void processChildElement(XMLObject& parentXMLObject, XMLObject* childXMLObject) const {
-        SimpleXMLObject& simpleXMLObject = dynamic_cast<SimpleXMLObject&>(parentXMLObject);
+#endif
 
-        SimpleXMLObject* child = dynamic_cast<SimpleXMLObject*>(childXMLObject);
-        if (child) {
-            simpleXMLObject.getSimpleXMLObjects().push_back(child);
-        }
-        else {
-            throw UnmarshallingException("Unknown child element cannot be added to parent object.");
-        }
+    VectorOf(SimpleXMLObject) getSimpleXMLObjects() {
+        return VectorOf(SimpleXMLObject)(this, m_simples, &m_children, m_children.end());
     }
-
-    void processAttribute(XMLObject& xmlObject, const DOMAttr* attribute) const {
-        SimpleXMLObject& simpleXMLObject = dynamic_cast<SimpleXMLObject&>(xmlObject);
-
-        if (XMLString::equals(attribute->getLocalName(),SimpleXMLObject::ID_ATTRIB_NAME)) {
-            simpleXMLObject.setId(attribute->getValue());
-        }
-        else {
-            throw UnmarshallingException("Unknown attribute cannot be processed by parent object.");
+    
+    const std::vector<SimpleXMLObject*>& getSimpleXMLObjects() const {
+        return m_simples;
+    }
+
+protected:
+    void marshallAttributes(xercesc::DOMElement* domElement) const {
+        if(getId()) {
+            domElement->setAttributeNS(nullptr, SimpleXMLObject::ID_ATTRIB_NAME, getId());
+#ifdef XMLTOOLING_XERCESC_BOOLSETIDATTRIBUTE
+            domElement->setIdAttributeNS(nullptr, SimpleXMLObject::ID_ATTRIB_NAME, true);
+#else
+            domElement->setIdAttributeNS(nullptr, SimpleXMLObject::ID_ATTRIB_NAME);
+#endif
         }
+        marshallExtensionAttributes(domElement);
     }
 
-    void processElementContent(XMLObject& xmlObject, const XMLCh* elementContent) const {
-        SimpleXMLObject& simpleXMLObject = dynamic_cast<SimpleXMLObject&>(xmlObject);
-        
-        simpleXMLObject.setValue(elementContent);
-    }
-
-};
-
-class WildcardXMLObjectMarshaller;
-
-class WildcardXMLObject : public AbstractExtensibleXMLObject, public AbstractAttributeExtensibleXMLObject
-{
-    friend class WildcardXMLObjectMarshaller;
-public:
-    WildcardXMLObject(const XMLCh* nsURI, const XMLCh* localName, const XMLCh* prefix)
-        : AbstractDOMCachingXMLObject(nsURI, localName, prefix),
-        AbstractExtensibleXMLObject(nsURI, localName, prefix),
-        AbstractAttributeExtensibleXMLObject(nsURI, localName, prefix) {}
-    virtual ~WildcardXMLObject() {}
-    
-    WildcardXMLObject* clone() const {
-        auto_ptr<XMLObject> domClone(AbstractDOMCachingXMLObject::clone());
-        WildcardXMLObject* ret=dynamic_cast<WildcardXMLObject*>(domClone.get());
-        if (ret) {
-            domClone.release();
-            return ret;
+    void processChildElement(XMLObject* childXMLObject, const xercesc::DOMElement* root) {
+        SimpleXMLObject* simple=dynamic_cast<SimpleXMLObject*>(childXMLObject);
+        if (simple) {
+            getSimpleXMLObjects().push_back(simple);
+            return;
         }
-
-        ret=new WildcardXMLObject(
-            getElementQName().getNamespaceURI(),getElementQName().getLocalPart(),getElementQName().getPrefix()
-            );
-        for (map<QName,XMLCh*>::const_iterator i=m_attributeMap.begin(); i!=m_attributeMap.end(); i++) {
-            ret->m_attributeMap[i->first]=XMLString::replicate(i->second);
+        
+#ifndef XMLTOOLING_NO_XMLSEC
+        Signature* sig=dynamic_cast<Signature*>(childXMLObject);
+        if (sig) {
+            setSignature(sig);
+            return;
         }
-        xmltooling::clone(m_children, ret->m_children);
-        return ret;
-    }
-};
-
-class WildcardXMLObjectBuilder : public XMLObjectBuilder
-{
-public:
-    XMLObject* buildObject() const {
-        throw XMLObjectException("No default builder available.");
-    }
+#endif
 
-    WildcardXMLObject* buildObject(const XMLCh* nsURI, const XMLCh* localName, const XMLCh* prefix) const {
-        return new WildcardXMLObject(nsURI,localName,prefix);
+        throw UnmarshallingException("Unknown child element cannot be added to parent object.");
     }
-};
 
-class WildcardXMLObjectMarshaller : public AbstractXMLObjectMarshaller
-{
-public:
-    WildcardXMLObjectMarshaller() : AbstractXMLObjectMarshaller() {}
-
-private:
-    void marshallAttributes(const XMLObject& xmlObject, DOMElement* domElement) const {
-        const WildcardXMLObject& wcXMLObject = dynamic_cast<const WildcardXMLObject&>(xmlObject);
-
-        for (map<QName,XMLCh*>::const_iterator i=wcXMLObject.m_attributeMap.begin(); i!=wcXMLObject.m_attributeMap.end(); i++) {
-            DOMAttr* attr=domElement->getOwnerDocument()->createAttributeNS(i->first.getNamespaceURI(),i->first.getLocalPart());
-            if (i->first.hasPrefix())
-                attr->setPrefix(i->first.getPrefix());
-            attr->setNodeValue(i->second);
-            domElement->setAttributeNode(attr);
+    void processAttribute(const xercesc::DOMAttr* attribute) {
+        if (XMLHelper::isNodeNamed(attribute, nullptr, SimpleXMLObject::ID_ATTRIB_NAME)) {
+            setId(attribute->getValue());
+            return;
         }
+        unmarshallExtensionAttribute(attribute);
     }
 
-    void marshallElementContent(const XMLObject& xmlObject, DOMElement* domElement) const {
-        const WildcardXMLObject& wcXMLObject = dynamic_cast<const WildcardXMLObject&>(xmlObject);
-
-        if(wcXMLObject.getTextContent()) {
-            domElement->appendChild(domElement->getOwnerDocument()->createTextNode(wcXMLObject.getTextContent()));
-        }
-    }
+private:
+    XMLCh* m_id;
+    vector<SimpleXMLObject*> m_simples;
+#ifndef XMLTOOLING_NO_XMLSEC
+    list<XMLObject*>::iterator m_signature;
+#endif
 };
 
-class WildcardXMLObjectUnmarshaller : public AbstractXMLObjectUnmarshaller
+class SimpleXMLObjectBuilder : public XMLObjectBuilder
 {
 public:
-    WildcardXMLObjectUnmarshaller() {}
-
-private:
-    XMLObject* buildXMLObject(const DOMElement* domElement) const {
-        const WildcardXMLObjectBuilder* builder =
-            dynamic_cast<const WildcardXMLObjectBuilder*>(XMLObjectBuilder::getBuilder(domElement));
-        if (builder)
-            return builder->buildObject(domElement->getNamespaceURI(),domElement->getLocalName(),domElement->getPrefix());
-        throw UnmarshallingException("Failed to locate WildcardObjectBuilder for element.");
-    }
-
-    void processChildElement(XMLObject& parentXMLObject, XMLObject* childXMLObject) const {
-        WildcardXMLObject& wcXMLObject = dynamic_cast<WildcardXMLObject&>(parentXMLObject);
-
-        wcXMLObject.getXMLObjects().push_back(childXMLObject);
+    XMLObject* buildObject() const {
+        return buildObject(SimpleXMLObject::NAMESPACE, SimpleXMLObject::LOCAL_NAME, SimpleXMLObject::NAMESPACE_PREFIX);
     }
 
-    void processAttribute(XMLObject& xmlObject, const DOMAttr* attribute) const {
-        WildcardXMLObject& wcXMLObject = dynamic_cast<WildcardXMLObject&>(xmlObject);
-       
-        QName q(attribute->getNamespaceURI(),attribute->getLocalName(),attribute->getPrefix()); 
-        wcXMLObject.setAttribute(q,attribute->getNodeValue());
+    XMLObject* buildObject(
+        const XMLCh* nsURI, const XMLCh* localName, const XMLCh* prefix=nullptr, const xmltooling::QName* schemaType=nullptr
+        ) const {
+        return new SimpleXMLObject(nsURI, localName, prefix, schemaType);
     }
 
-    void processElementContent(XMLObject& xmlObject, const XMLCh* elementContent) const {
-        WildcardXMLObject& wcXMLObject = dynamic_cast<WildcardXMLObject&>(xmlObject);
-        
-        wcXMLObject.setTextContent(elementContent);
+    static SimpleXMLObject* buildSimpleXMLObject() {
+        const SimpleXMLObjectBuilder* b = dynamic_cast<const SimpleXMLObjectBuilder*>(
+            XMLObjectBuilder::getBuilder(xmltooling::QName(SimpleXMLObject::NAMESPACE,SimpleXMLObject::LOCAL_NAME))
+            );
+        if (b)
+            return dynamic_cast<SimpleXMLObject*>(b->buildObject());
+        throw XMLObjectException("Unable to obtain typed builder for SimpleXMLObject.");
     }
-
 };
 
 #if defined (_MSC_VER)