Fixes associated with wildcard test classes.
[shibboleth/cpp-xmltooling.git] / xmltoolingtest / XMLObjectBaseTestCase.h
1 /*\r
2  *  Copyright 2001-2005 Internet2\r
3  * \r
4  * Licensed under the Apache License, Version 2.0 (the "License");\r
5  * you may not use this file except in compliance with the License.\r
6  * You may obtain a copy of the License at\r
7  *\r
8  *     http://www.apache.org/licenses/LICENSE-2.0\r
9  *\r
10  * Unless required by applicable law or agreed to in writing, software\r
11  * distributed under the License is distributed on an "AS IS" BASIS,\r
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\r
13  * See the License for the specific language governing permissions and\r
14  * limitations under the License.\r
15  */\r
16 \r
17 #include <cxxtest/TestSuite.h>\r
18 #include <xmltooling/AbstractAttributeExtensibleXMLObject.h>\r
19 #include <xmltooling/AbstractDOMCachingXMLObject.h>\r
20 #include <xmltooling/AbstractExtensibleXMLObject.h>\r
21 #include <xmltooling/exceptions.h>\r
22 #include <xmltooling/XMLObjectBuilder.h>\r
23 #include <xmltooling/XMLToolingConfig.h>\r
24 #include <xmltooling/io/AbstractXMLObjectMarshaller.h>\r
25 #include <xmltooling/io/AbstractXMLObjectUnmarshaller.h>\r
26 #include <xmltooling/util/ParserPool.h>\r
27 #include <xmltooling/util/XMLObjectChildrenList.h>\r
28 #include <xmltooling/util/XMLHelper.h>\r
29 \r
30 using namespace xmltooling;\r
31 using namespace std;\r
32 \r
33 extern ParserPool* validatingPool;\r
34 extern ParserPool* nonvalidatingPool;\r
35 extern string data_path;\r
36 \r
37 #if defined (_MSC_VER)\r
38     #pragma warning( push )\r
39     #pragma warning( disable : 4250 4251 )\r
40 #endif\r
41 \r
42 class SimpleXMLObject : public AbstractDOMCachingXMLObject\r
43 {\r
44 public:\r
45     static const XMLCh NAMESPACE[];\r
46     static const XMLCh NAMESPACE_PREFIX[];\r
47     static const XMLCh LOCAL_NAME[];\r
48     static const XMLCh ID_ATTRIB_NAME[];\r
49 \r
50     SimpleXMLObject() : AbstractDOMCachingXMLObject(NAMESPACE, LOCAL_NAME, NAMESPACE_PREFIX), m_id(NULL), m_value(NULL) {}\r
51     virtual ~SimpleXMLObject() {\r
52         XMLString::release(&m_id);\r
53         XMLString::release(&m_value);\r
54     }\r
55     \r
56     const XMLCh* getId() const { return m_id; }\r
57     void setId(const XMLCh* id) { m_id=prepareForAssignment(m_id,id); }\r
58 \r
59     const XMLCh* getValue() const { return m_value; }\r
60     void setValue(const XMLCh* value) { m_value=prepareForAssignment(m_value,value); }\r
61     \r
62     VectorOf(SimpleXMLObject) getSimpleXMLObjects() {\r
63         return VectorOf(SimpleXMLObject)(this, m_simples, &m_children, m_children.end());\r
64     }\r
65     \r
66     SimpleXMLObject* clone() const {\r
67         auto_ptr<XMLObject> domClone(AbstractDOMCachingXMLObject::clone());\r
68         SimpleXMLObject* ret=dynamic_cast<SimpleXMLObject*>(domClone.get());\r
69         if (ret) {\r
70             domClone.release();\r
71             return ret;\r
72         }\r
73 \r
74         ret=new SimpleXMLObject();\r
75         ret->setId(m_id);\r
76         ret->setValue(m_value);\r
77         xmltooling::clone(m_children, ret->m_children);\r
78         return ret;\r
79     }\r
80 \r
81 private:\r
82     XMLCh* m_id;\r
83     XMLCh* m_value;\r
84     vector<SimpleXMLObject*> m_simples;\r
85 };\r
86 \r
87 class SimpleXMLObjectBuilder : public XMLObjectBuilder\r
88 {\r
89 public:\r
90     SimpleXMLObject* buildObject() const {\r
91         return new SimpleXMLObject();\r
92     }\r
93 };\r
94 \r
95 class SimpleXMLObjectMarshaller : public AbstractXMLObjectMarshaller\r
96 {\r
97 public:\r
98     SimpleXMLObjectMarshaller() {}\r
99 \r
100 private:\r
101     void marshallAttributes(const XMLObject& xmlObject, DOMElement* domElement) const {\r
102         const SimpleXMLObject& simpleXMLObject = dynamic_cast<const SimpleXMLObject&>(xmlObject);\r
103         \r
104         if(simpleXMLObject.getId()) {\r
105             domElement->setAttributeNS(NULL, SimpleXMLObject::ID_ATTRIB_NAME, simpleXMLObject.getId());\r
106             domElement->setIdAttributeNS(NULL, SimpleXMLObject::ID_ATTRIB_NAME);\r
107         }\r
108     }\r
109 \r
110     void marshallElementContent(const XMLObject& xmlObject, DOMElement* domElement) const {\r
111         const SimpleXMLObject& simpleXMLObject = dynamic_cast<const SimpleXMLObject&>(xmlObject);\r
112 \r
113         if(simpleXMLObject.getValue()) {\r
114             domElement->setTextContent(simpleXMLObject.getValue());\r
115         }\r
116     }\r
117 };\r
118 \r
119 class SimpleXMLObjectUnmarshaller : public AbstractXMLObjectUnmarshaller\r
120 {\r
121 public:\r
122     SimpleXMLObjectUnmarshaller() {}\r
123 \r
124 private:\r
125     void processChildElement(XMLObject& parentXMLObject, XMLObject* childXMLObject) const {\r
126         SimpleXMLObject& simpleXMLObject = dynamic_cast<SimpleXMLObject&>(parentXMLObject);\r
127 \r
128         SimpleXMLObject* child = dynamic_cast<SimpleXMLObject*>(childXMLObject);\r
129         if (child) {\r
130             simpleXMLObject.getSimpleXMLObjects().push_back(child);\r
131         }\r
132         else {\r
133             throw UnmarshallingException("Unknown child element cannot be added to parent object.");\r
134         }\r
135     }\r
136 \r
137     void processAttribute(XMLObject& xmlObject, const DOMAttr* attribute) const {\r
138         SimpleXMLObject& simpleXMLObject = dynamic_cast<SimpleXMLObject&>(xmlObject);\r
139 \r
140         if (XMLString::equals(attribute->getLocalName(),SimpleXMLObject::ID_ATTRIB_NAME)) {\r
141             simpleXMLObject.setId(attribute->getValue());\r
142         }\r
143         else {\r
144             throw UnmarshallingException("Unknown attribute cannot be processed by parent object.");\r
145         }\r
146     }\r
147 \r
148     void processElementContent(XMLObject& xmlObject, const XMLCh* elementContent) const {\r
149         SimpleXMLObject& simpleXMLObject = dynamic_cast<SimpleXMLObject&>(xmlObject);\r
150         \r
151         simpleXMLObject.setValue(elementContent);\r
152     }\r
153 \r
154 };\r
155 \r
156 class WildcardXMLObjectMarshaller;\r
157 \r
158 class WildcardXMLObject : public AbstractExtensibleXMLObject, public AbstractAttributeExtensibleXMLObject\r
159 {\r
160     friend class WildcardXMLObjectMarshaller;\r
161 public:\r
162     WildcardXMLObject(const XMLCh* nsURI, const XMLCh* localName, const XMLCh* prefix)\r
163         : AbstractDOMCachingXMLObject(nsURI, localName, prefix),\r
164         AbstractExtensibleXMLObject(nsURI, localName, prefix),\r
165         AbstractAttributeExtensibleXMLObject(nsURI, localName, prefix) {}\r
166     virtual ~WildcardXMLObject() {}\r
167     \r
168     WildcardXMLObject* clone() const {\r
169         auto_ptr<XMLObject> domClone(AbstractDOMCachingXMLObject::clone());\r
170         WildcardXMLObject* ret=dynamic_cast<WildcardXMLObject*>(domClone.get());\r
171         if (ret) {\r
172             domClone.release();\r
173             return ret;\r
174         }\r
175 \r
176         ret=new WildcardXMLObject(\r
177             getElementQName().getNamespaceURI(),getElementQName().getLocalPart(),getElementQName().getPrefix()\r
178             );\r
179         for (map<QName,XMLCh*>::const_iterator i=m_attributeMap.begin(); i!=m_attributeMap.end(); i++) {\r
180             ret->m_attributeMap[i->first]=XMLString::replicate(i->second);\r
181         }\r
182         xmltooling::clone(m_children, ret->m_children);\r
183         return ret;\r
184     }\r
185 };\r
186 \r
187 class WildcardXMLObjectBuilder : public XMLObjectBuilder\r
188 {\r
189 public:\r
190     XMLObject* buildObject() const {\r
191         throw XMLObjectException("No default builder available.");\r
192     }\r
193 \r
194     WildcardXMLObject* buildObject(const XMLCh* nsURI, const XMLCh* localName, const XMLCh* prefix) const {\r
195         return new WildcardXMLObject(nsURI,localName,prefix);\r
196     }\r
197 };\r
198 \r
199 class WildcardXMLObjectMarshaller : public AbstractXMLObjectMarshaller\r
200 {\r
201 public:\r
202     WildcardXMLObjectMarshaller() : AbstractXMLObjectMarshaller() {}\r
203 \r
204 private:\r
205     void marshallAttributes(const XMLObject& xmlObject, DOMElement* domElement) const {\r
206         const WildcardXMLObject& wcXMLObject = dynamic_cast<const WildcardXMLObject&>(xmlObject);\r
207 \r
208         for (map<QName,XMLCh*>::const_iterator i=wcXMLObject.m_attributeMap.begin(); i!=wcXMLObject.m_attributeMap.end(); i++) {\r
209             DOMAttr* attr=domElement->getOwnerDocument()->createAttributeNS(i->first.getNamespaceURI(),i->first.getLocalPart());\r
210             if (i->first.hasPrefix())\r
211                 attr->setPrefix(i->first.getPrefix());\r
212             attr->setNodeValue(i->second);\r
213             domElement->setAttributeNode(attr);\r
214         }\r
215     }\r
216 \r
217     void marshallElementContent(const XMLObject& xmlObject, DOMElement* domElement) const {\r
218         const WildcardXMLObject& wcXMLObject = dynamic_cast<const WildcardXMLObject&>(xmlObject);\r
219 \r
220         if(wcXMLObject.getTextContent()) {\r
221             domElement->appendChild(domElement->getOwnerDocument()->createTextNode(wcXMLObject.getTextContent()));\r
222         }\r
223     }\r
224 };\r
225 \r
226 class WildcardXMLObjectUnmarshaller : public AbstractXMLObjectUnmarshaller\r
227 {\r
228 public:\r
229     WildcardXMLObjectUnmarshaller() {}\r
230 \r
231 private:\r
232     XMLObject* buildXMLObject(const DOMElement* domElement) const {\r
233         const WildcardXMLObjectBuilder* builder =\r
234             dynamic_cast<const WildcardXMLObjectBuilder*>(XMLObjectBuilder::getBuilder(domElement));\r
235         if (builder)\r
236             return builder->buildObject(domElement->getNamespaceURI(),domElement->getLocalName(),domElement->getPrefix());\r
237         throw UnmarshallingException("Failed to locate WildcardObjectBuilder for element.");\r
238     }\r
239 \r
240     void processChildElement(XMLObject& parentXMLObject, XMLObject* childXMLObject) const {\r
241         WildcardXMLObject& wcXMLObject = dynamic_cast<WildcardXMLObject&>(parentXMLObject);\r
242 \r
243         wcXMLObject.getXMLObjects().push_back(childXMLObject);\r
244     }\r
245 \r
246     void processAttribute(XMLObject& xmlObject, const DOMAttr* attribute) const {\r
247         WildcardXMLObject& wcXMLObject = dynamic_cast<WildcardXMLObject&>(xmlObject);\r
248         \r
249         wcXMLObject.setAttribute(\r
250             QName(attribute->getNamespaceURI(),attribute->getLocalName(),attribute->getPrefix()),\r
251             attribute->getNodeValue()\r
252             );\r
253     }\r
254 \r
255     void processElementContent(XMLObject& xmlObject, const XMLCh* elementContent) const {\r
256         WildcardXMLObject& wcXMLObject = dynamic_cast<WildcardXMLObject&>(xmlObject);\r
257         \r
258         wcXMLObject.setTextContent(elementContent);\r
259     }\r
260 \r
261 };\r
262 \r
263 #if defined (_MSC_VER)\r
264     #pragma warning( pop )\r
265 #endif\r