2 * Copyright 2001-2005 Internet2
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
8 * http://www.apache.org/licenses/LICENSE-2.0
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
17 #include <cxxtest/TestSuite.h>
18 #include <xmltooling/AbstractAttributeExtensibleXMLObject.h>
19 #include <xmltooling/AbstractDOMCachingXMLObject.h>
20 #include <xmltooling/AbstractElementProxy.h>
21 #include <xmltooling/exceptions.h>
22 #include <xmltooling/XMLObjectBuilder.h>
23 #include <xmltooling/XMLToolingConfig.h>
24 #include <xmltooling/io/AbstractXMLObjectMarshaller.h>
25 #include <xmltooling/io/AbstractXMLObjectUnmarshaller.h>
26 #ifndef XMLTOOLING_NO_XMLSEC
27 #include <xmltooling/signature/Signature.h>
29 #include <xmltooling/util/ParserPool.h>
30 #include <xmltooling/util/XMLConstants.h>
31 #include <xmltooling/util/XMLHelper.h>
32 #include <xmltooling/util/XMLObjectChildrenList.h>
34 using namespace xmltooling;
37 extern ParserPool* validatingPool;
38 extern ParserPool* nonvalidatingPool;
39 extern string data_path;
41 #if defined (_MSC_VER)
42 #pragma warning( push )
43 #pragma warning( disable : 4250 4251 )
46 class SimpleXMLObject : public AbstractDOMCachingXMLObject
49 static const XMLCh NAMESPACE[];
50 static const XMLCh NAMESPACE_PREFIX[];
51 static const XMLCh LOCAL_NAME[];
52 static const XMLCh ID_ATTRIB_NAME[];
54 SimpleXMLObject() : AbstractDOMCachingXMLObject(NAMESPACE, LOCAL_NAME, NAMESPACE_PREFIX), m_id(NULL), m_value(NULL) {
55 #ifndef XMLTOOLING_NO_XMLSEC
56 m_children.push_back(NULL);
57 m_signature=m_children.begin();
61 virtual ~SimpleXMLObject() {
62 XMLString::release(&m_id);
63 XMLString::release(&m_value);
66 const XMLCh* getId() const { return m_id; }
67 void setId(const XMLCh* id) { m_id=prepareForAssignment(m_id,id); }
69 const XMLCh* getValue() const { return m_value; }
70 void setValue(const XMLCh* value) { m_value=prepareForAssignment(m_value,value); }
72 #ifndef XMLTOOLING_NO_XMLSEC
73 Signature* getSignature() const {
74 return dynamic_cast<Signature*>(*m_signature);
77 void setSignature(Signature* sig) {
78 *m_signature=prepareForAssignment(*m_signature,sig);
82 VectorOf(SimpleXMLObject) getSimpleXMLObjects() {
83 return VectorOf(SimpleXMLObject)(this, m_simples, &m_children, m_children.end());
86 SimpleXMLObject* clone() const {
87 auto_ptr<XMLObject> domClone(AbstractDOMCachingXMLObject::clone());
88 SimpleXMLObject* ret=dynamic_cast<SimpleXMLObject*>(domClone.get());
94 ret=new SimpleXMLObject();
95 ret->m_namespaces=m_namespaces;
97 ret->setValue(m_value);
98 xmltooling::clone(m_children, ret->m_children);
105 vector<SimpleXMLObject*> m_simples;
106 #ifndef XMLTOOLING_NO_XMLSEC
107 list<XMLObject*>::iterator m_signature;
111 class SimpleXMLObjectBuilder : public XMLObjectBuilder
114 SimpleXMLObject* buildObject() const {
115 return new SimpleXMLObject();
119 class SimpleXMLObjectMarshaller : public AbstractXMLObjectMarshaller
122 SimpleXMLObjectMarshaller() {}
125 void marshallAttributes(const XMLObject& xmlObject, DOMElement* domElement) const {
126 const SimpleXMLObject& simpleXMLObject = dynamic_cast<const SimpleXMLObject&>(xmlObject);
128 if(simpleXMLObject.getId()) {
129 domElement->setAttributeNS(NULL, SimpleXMLObject::ID_ATTRIB_NAME, simpleXMLObject.getId());
130 domElement->setIdAttributeNS(NULL, SimpleXMLObject::ID_ATTRIB_NAME);
134 void marshallElementContent(const XMLObject& xmlObject, DOMElement* domElement) const {
135 const SimpleXMLObject& simpleXMLObject = dynamic_cast<const SimpleXMLObject&>(xmlObject);
137 if(simpleXMLObject.getValue()) {
138 domElement->setTextContent(simpleXMLObject.getValue());
143 class SimpleXMLObjectUnmarshaller : public AbstractXMLObjectUnmarshaller
146 SimpleXMLObjectUnmarshaller() {}
149 void processChildElement(XMLObject& parentXMLObject, XMLObject* childXMLObject, const DOMElement* root) const {
150 SimpleXMLObject& simpleXMLObject = dynamic_cast<SimpleXMLObject&>(parentXMLObject);
152 if (XMLHelper::isNodeNamed(root, SimpleXMLObject::NAMESPACE, SimpleXMLObject::LOCAL_NAME))
153 simpleXMLObject.getSimpleXMLObjects().push_back(dynamic_cast<SimpleXMLObject*>(childXMLObject));
154 #ifndef XMLTOOLING_NO_XMLSEC
155 else if (XMLHelper::isNodeNamed(root, XMLConstants::XMLSIG_NS, Signature::LOCAL_NAME))
156 simpleXMLObject.setSignature(dynamic_cast<Signature*>(childXMLObject));
159 throw UnmarshallingException("Unknown child element cannot be added to parent object.");
162 void processAttribute(XMLObject& xmlObject, const DOMAttr* attribute) const {
163 SimpleXMLObject& simpleXMLObject = dynamic_cast<SimpleXMLObject&>(xmlObject);
165 if (XMLHelper::isNodeNamed(attribute, NULL, SimpleXMLObject::ID_ATTRIB_NAME))
166 simpleXMLObject.setId(attribute->getValue());
168 throw UnmarshallingException("Unknown attribute cannot be processed by parent object.");
171 void processElementContent(XMLObject& xmlObject, const XMLCh* elementContent) const {
172 SimpleXMLObject& simpleXMLObject = dynamic_cast<SimpleXMLObject&>(xmlObject);
174 simpleXMLObject.setValue(elementContent);
179 class WildcardXMLObjectMarshaller;
181 class WildcardXMLObject : public AbstractElementProxy, public AbstractAttributeExtensibleXMLObject
183 friend class WildcardXMLObjectMarshaller;
185 WildcardXMLObject(const XMLCh* nsURI, const XMLCh* localName, const XMLCh* prefix)
186 : AbstractDOMCachingXMLObject(nsURI, localName, prefix),
187 AbstractElementProxy(nsURI, localName, prefix),
188 AbstractAttributeExtensibleXMLObject(nsURI, localName, prefix) {}
189 virtual ~WildcardXMLObject() {}
191 WildcardXMLObject* clone() const {
192 auto_ptr<XMLObject> domClone(AbstractDOMCachingXMLObject::clone());
193 WildcardXMLObject* ret=dynamic_cast<WildcardXMLObject*>(domClone.get());
199 ret=new WildcardXMLObject(
200 getElementQName().getNamespaceURI(),getElementQName().getLocalPart(),getElementQName().getPrefix()
202 ret->m_namespaces=m_namespaces;
203 for (map<QName,XMLCh*>::const_iterator i=m_attributeMap.begin(); i!=m_attributeMap.end(); i++) {
204 ret->m_attributeMap[i->first]=XMLString::replicate(i->second);
206 ret->setTextContent(getTextContent());
207 xmltooling::clone(m_children, ret->m_children);
212 class WildcardXMLObjectBuilder : public XMLObjectBuilder
215 XMLObject* buildObject() const {
216 throw XMLObjectException("No default builder available.");
219 WildcardXMLObject* buildObject(const XMLCh* nsURI, const XMLCh* localName, const XMLCh* prefix) const {
220 return new WildcardXMLObject(nsURI,localName,prefix);
224 class WildcardXMLObjectMarshaller : public AbstractXMLObjectMarshaller
227 WildcardXMLObjectMarshaller() : AbstractXMLObjectMarshaller() {}
230 void marshallAttributes(const XMLObject& xmlObject, DOMElement* domElement) const {
231 const WildcardXMLObject& wcXMLObject = dynamic_cast<const WildcardXMLObject&>(xmlObject);
233 for (map<QName,XMLCh*>::const_iterator i=wcXMLObject.m_attributeMap.begin(); i!=wcXMLObject.m_attributeMap.end(); i++) {
234 DOMAttr* attr=domElement->getOwnerDocument()->createAttributeNS(i->first.getNamespaceURI(),i->first.getLocalPart());
235 if (i->first.hasPrefix())
236 attr->setPrefix(i->first.getPrefix());
237 attr->setNodeValue(i->second);
238 domElement->setAttributeNode(attr);
242 void marshallElementContent(const XMLObject& xmlObject, DOMElement* domElement) const {
243 const WildcardXMLObject& wcXMLObject = dynamic_cast<const WildcardXMLObject&>(xmlObject);
245 if(wcXMLObject.getTextContent()) {
246 domElement->appendChild(domElement->getOwnerDocument()->createTextNode(wcXMLObject.getTextContent()));
251 class WildcardXMLObjectUnmarshaller : public AbstractXMLObjectUnmarshaller
254 WildcardXMLObjectUnmarshaller() {}
257 XMLObject* buildXMLObject(const DOMElement* domElement) const {
258 const WildcardXMLObjectBuilder* builder =
259 dynamic_cast<const WildcardXMLObjectBuilder*>(XMLObjectBuilder::getBuilder(domElement));
261 return builder->buildObject(domElement->getNamespaceURI(),domElement->getLocalName(),domElement->getPrefix());
262 throw UnmarshallingException("Failed to locate WildcardObjectBuilder for element.");
265 void processChildElement(XMLObject& parentXMLObject, XMLObject* childXMLObject, const DOMElement* root) const {
266 WildcardXMLObject& wcXMLObject = dynamic_cast<WildcardXMLObject&>(parentXMLObject);
268 wcXMLObject.getXMLObjects().push_back(childXMLObject);
271 void processAttribute(XMLObject& xmlObject, const DOMAttr* attribute) const {
272 WildcardXMLObject& wcXMLObject = dynamic_cast<WildcardXMLObject&>(xmlObject);
274 QName q(attribute->getNamespaceURI(),attribute->getLocalName(),attribute->getPrefix());
275 wcXMLObject.setAttribute(q,attribute->getNodeValue());
278 void processElementContent(XMLObject& xmlObject, const XMLCh* elementContent) const {
279 WildcardXMLObject& wcXMLObject = dynamic_cast<WildcardXMLObject&>(xmlObject);
281 wcXMLObject.setTextContent(elementContent);
286 #if defined (_MSC_VER)
287 #pragma warning( pop )