922d58315d83d65e0f520c14c6f0e256083460e7
[shibboleth/cpp-xmltooling.git] / xmltooling / signature / impl / KeyInfoImpl.cpp
1 /*
2  *  Copyright 2001-2006 Internet2
3  * 
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
7  *
8  *     http://www.apache.org/licenses/LICENSE-2.0
9  *
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.
15  */
16
17 /**
18  * KeyInfoImpl.cpp
19  * 
20  * Implementation classes for KeyInfo schema
21  */
22
23 #include "internal.h"
24 #include "AbstractChildlessElement.h"
25 #include "AbstractComplexElement.h"
26 #include "AbstractElementProxy.h"
27 #include "AbstractSimpleElement.h"
28 #include "exceptions.h"
29 #include "io/AbstractXMLObjectMarshaller.h"
30 #include "io/AbstractXMLObjectUnmarshaller.h"
31 #include "signature/KeyInfo.h"
32 #include "util/XMLHelper.h"
33
34 #include <xercesc/util/XMLUniDefs.hpp>
35
36 using namespace xmlsignature;
37 using namespace xmltooling;
38 using namespace std;
39
40 #if defined (_MSC_VER)
41     #pragma warning( push )
42     #pragma warning( disable : 4250 4251 )
43 #endif
44
45 namespace xmlsignature {
46     
47     class XMLTOOL_DLLLOCAL DSAKeyValueImpl : public virtual DSAKeyValue,
48         public AbstractComplexElement,
49         public AbstractDOMCachingXMLObject,
50         public AbstractXMLObjectMarshaller,
51         public AbstractXMLObjectUnmarshaller
52     {
53     public:
54         virtual ~DSAKeyValueImpl() {}
55
56         DSAKeyValueImpl(const XMLCh* nsURI, const XMLCh* localName, const XMLCh* prefix, const QName* schemaType)
57             : AbstractXMLObject(nsURI, localName, prefix, schemaType) {
58             init();
59         }
60             
61         DSAKeyValueImpl(const DSAKeyValueImpl& src) : AbstractXMLObject(src), AbstractDOMCachingXMLObject(src) {
62             init();
63             if (src.getP())
64                 setP(src.getP()->cloneP());
65             if (src.getQ())
66                 setQ(src.getQ()->cloneQ());
67             if (src.getG())
68                 setG(src.getG()->cloneG());
69             if (src.getY())
70                 setY(src.getY()->cloneY());
71             if (src.getJ())
72                 setJ(src.getJ()->cloneJ());
73             if (src.getSeed())
74                 setSeed(src.getSeed()->cloneSeed());
75             if (src.getPgenCounter())
76                 setPgenCounter(src.getPgenCounter()->clonePgenCounter());
77         }
78         
79         void init() {
80             m_P=NULL;
81             m_Q=NULL;
82             m_G=NULL;
83             m_Y=NULL;
84             m_J=NULL;
85             m_Seed=NULL;
86             m_PgenCounter=NULL;
87             m_children.push_back(NULL);
88             m_children.push_back(NULL);
89             m_children.push_back(NULL);
90             m_children.push_back(NULL);
91             m_children.push_back(NULL);
92             m_children.push_back(NULL);
93             m_children.push_back(NULL);
94             m_pos_P=m_children.begin();
95             m_pos_Q=m_pos_P;
96             ++m_pos_Q;
97             m_pos_G=m_pos_Q;
98             ++m_pos_G;
99             m_pos_Y=m_pos_G;
100             ++m_pos_Y;
101             m_pos_J=m_pos_Y;
102             ++m_pos_J;
103             m_pos_Seed=m_pos_J;
104             ++m_pos_Seed;
105             m_pos_PgenCounter=m_pos_Seed;
106             ++m_pos_PgenCounter;
107         }
108         
109         IMPL_XMLOBJECT_CLONE(DSAKeyValue);
110         IMPL_TYPED_CHILD(P);
111         IMPL_TYPED_CHILD(Q);
112         IMPL_TYPED_CHILD(G);
113         IMPL_TYPED_CHILD(Y);
114         IMPL_TYPED_CHILD(J);
115         IMPL_TYPED_CHILD(Seed);
116         IMPL_TYPED_CHILD(PgenCounter);
117
118     protected:
119         void processChildElement(XMLObject* childXMLObject, const DOMElement* root) {
120             PROC_TYPED_CHILD(P,XMLConstants::XMLSIG_NS,false);
121             PROC_TYPED_CHILD(Q,XMLConstants::XMLSIG_NS,false);
122             PROC_TYPED_CHILD(G,XMLConstants::XMLSIG_NS,false);
123             PROC_TYPED_CHILD(Y,XMLConstants::XMLSIG_NS,false);
124             PROC_TYPED_CHILD(J,XMLConstants::XMLSIG_NS,false);
125             PROC_TYPED_CHILD(Seed,XMLConstants::XMLSIG_NS,false);
126             PROC_TYPED_CHILD(PgenCounter,XMLConstants::XMLSIG_NS,false);
127             AbstractXMLObjectUnmarshaller::processChildElement(childXMLObject,root);
128         }
129     };
130
131     class XMLTOOL_DLLLOCAL RSAKeyValueImpl : public virtual RSAKeyValue,
132         public AbstractComplexElement,
133         public AbstractDOMCachingXMLObject,
134         public AbstractXMLObjectMarshaller,
135         public AbstractXMLObjectUnmarshaller
136     {
137     public:
138         virtual ~RSAKeyValueImpl() {}
139
140         RSAKeyValueImpl(const XMLCh* nsURI, const XMLCh* localName, const XMLCh* prefix, const QName* schemaType)
141                 : AbstractXMLObject(nsURI, localName, prefix, schemaType) {
142             init();
143         }
144             
145         RSAKeyValueImpl(const RSAKeyValueImpl& src) : AbstractXMLObject(src), AbstractDOMCachingXMLObject(src) {
146             init();
147             if (src.getModulus())
148                 setModulus(src.getModulus()->cloneModulus());
149             if (src.getExponent())
150                 setExponent(src.getExponent()->cloneExponent());
151         }
152         
153         void init() {
154             m_Modulus=NULL;
155             m_Exponent=NULL;
156             m_children.push_back(NULL);
157             m_children.push_back(NULL);
158             m_pos_Modulus=m_children.begin();
159             m_pos_Exponent=m_pos_Modulus;
160             ++m_pos_Exponent;
161         }
162         
163         IMPL_XMLOBJECT_CLONE(RSAKeyValue);
164         IMPL_TYPED_CHILD(Modulus);
165         IMPL_TYPED_CHILD(Exponent);
166
167     protected:
168         void processChildElement(XMLObject* childXMLObject, const DOMElement* root) {
169             PROC_TYPED_CHILD(Modulus,XMLConstants::XMLSIG_NS,false);
170             PROC_TYPED_CHILD(Exponent,XMLConstants::XMLSIG_NS,false);
171             AbstractXMLObjectUnmarshaller::processChildElement(childXMLObject,root);
172         }
173     };
174
175     class XMLTOOL_DLLLOCAL KeyValueImpl : public virtual KeyValue,
176         public AbstractSimpleElement,
177         public AbstractComplexElement,
178         public AbstractDOMCachingXMLObject,
179         public AbstractXMLObjectMarshaller,
180         public AbstractXMLObjectUnmarshaller
181     {
182     public:
183         virtual ~KeyValueImpl() {}
184
185         KeyValueImpl(const XMLCh* nsURI, const XMLCh* localName, const XMLCh* prefix, const QName* schemaType)
186                 : AbstractXMLObject(nsURI, localName, prefix, schemaType) {
187             init();
188         }
189             
190         KeyValueImpl(const KeyValueImpl& src)
191                 : AbstractXMLObject(src), AbstractSimpleElement(src), AbstractDOMCachingXMLObject(src) {
192             init();
193             if (src.getDSAKeyValue())
194                 setDSAKeyValue(src.getDSAKeyValue()->cloneDSAKeyValue());
195             if (src.getRSAKeyValue())
196                 setRSAKeyValue(src.getRSAKeyValue()->cloneRSAKeyValue());
197             if (src.getOtherKeyValue())
198                 setOtherKeyValue(src.getOtherKeyValue()->clone());
199         }
200         
201         void init() {
202             m_DSAKeyValue=NULL;
203             m_RSAKeyValue=NULL;
204             m_OtherKeyValue=NULL;
205             m_children.push_back(NULL);
206             m_children.push_back(NULL);
207             m_children.push_back(NULL);
208             m_pos_DSAKeyValue=m_children.begin();
209             m_pos_RSAKeyValue=m_pos_DSAKeyValue;
210             ++m_pos_RSAKeyValue;
211             m_pos_OtherKeyValue=m_pos_RSAKeyValue;
212             ++m_pos_OtherKeyValue;
213         }
214         
215         IMPL_XMLOBJECT_CLONE(KeyValue);
216         IMPL_TYPED_CHILD(DSAKeyValue);
217         IMPL_TYPED_CHILD(RSAKeyValue);
218         IMPL_XMLOBJECT_CHILD(OtherKeyValue);
219         IMPL_XMLOBJECT_CONTENT;
220
221     protected:
222         void processChildElement(XMLObject* childXMLObject, const DOMElement* root) {
223             PROC_TYPED_CHILD(DSAKeyValue,XMLConstants::XMLSIG_NS,false);
224             PROC_TYPED_CHILD(RSAKeyValue,XMLConstants::XMLSIG_NS,false);
225             
226             // Unknown child.
227             const XMLCh* nsURI=root->getNamespaceURI();
228             if (!XMLString::equals(nsURI,XMLConstants::XMLSIG_NS) && nsURI && *nsURI) {
229                 setOtherKeyValue(childXMLObject);
230                 return;
231             }
232             
233             AbstractXMLObjectUnmarshaller::processChildElement(childXMLObject,root);
234         }
235     };
236
237     class XMLTOOL_DLLLOCAL TransformImpl : public virtual Transform,
238         public AbstractDOMCachingXMLObject,
239         public AbstractElementProxy,
240         public AbstractXMLObjectMarshaller,
241         public AbstractXMLObjectUnmarshaller
242     {
243     public:
244         virtual ~TransformImpl() {
245             XMLString::release(&m_Algorithm);
246         }
247
248         TransformImpl(const XMLCh* nsURI, const XMLCh* localName, const XMLCh* prefix, const QName* schemaType)
249             : AbstractXMLObject(nsURI, localName, prefix, schemaType), m_Algorithm(NULL) {
250         }
251             
252         TransformImpl(const TransformImpl& src)
253                 : AbstractXMLObject(src), AbstractDOMCachingXMLObject(src), AbstractElementProxy(src),
254                     m_Algorithm(XMLString::replicate(src.m_Algorithm)) {
255             for (list<XMLObject*>::const_iterator i=src.m_children.begin(); i!=src.m_children.end(); i++) {
256                 if (*i) {
257                     XPath* x=dynamic_cast<XPath*>(*i);
258                     if (x) {
259                         getXPaths().push_back(x->cloneXPath());
260                         continue;
261                     }
262                     getXMLObjects().push_back((*i)->clone());
263                 }
264             }
265         }
266         
267         IMPL_XMLOBJECT_CLONE(Transform);
268         IMPL_STRING_ATTRIB(Algorithm);
269         IMPL_TYPED_CHILDREN(XPath,m_children.end());
270         IMPL_XMLOBJECT_CONTENT;
271
272     protected:
273         void marshallAttributes(DOMElement* domElement) const {
274             MARSHALL_STRING_ATTRIB(Algorithm,ALGORITHM,NULL);
275         }
276
277         void processChildElement(XMLObject* childXMLObject, const DOMElement* root) {
278             PROC_TYPED_CHILDREN(XPath,XMLConstants::XMLSIG_NS,false);
279             
280             // Unknown child.
281             const XMLCh* nsURI=root->getNamespaceURI();
282             if (!XMLString::equals(nsURI,XMLConstants::XMLSIG_NS) && nsURI && *nsURI) {
283                 getXMLObjects().push_back(childXMLObject);
284                 return;
285             }
286             
287             AbstractXMLObjectUnmarshaller::processChildElement(childXMLObject,root);
288         }
289
290         void processAttribute(const DOMAttr* attribute) {
291             PROC_STRING_ATTRIB(Algorithm,ALGORITHM,NULL);
292             AbstractXMLObjectUnmarshaller::processAttribute(attribute);
293         }
294     };
295
296     class XMLTOOL_DLLLOCAL TransformsImpl : public virtual Transforms,
297         public AbstractComplexElement,
298         public AbstractDOMCachingXMLObject,
299         public AbstractXMLObjectMarshaller,
300         public AbstractXMLObjectUnmarshaller
301     {
302     public:
303         virtual ~TransformsImpl() {}
304
305         TransformsImpl(const XMLCh* nsURI, const XMLCh* localName, const XMLCh* prefix, const QName* schemaType)
306             : AbstractXMLObject(nsURI, localName, prefix, schemaType) {
307         }
308             
309         TransformsImpl(const TransformsImpl& src) : AbstractXMLObject(src), AbstractDOMCachingXMLObject(src) {
310             VectorOf(Transform) v=getTransforms();
311             for (vector<Transform*>::const_iterator i=src.m_Transforms.begin(); i!=src.m_Transforms.end(); i++) {
312                 if (*i) {
313                     v.push_back((*i)->cloneTransform());
314                 }
315             }
316         }
317         
318         IMPL_XMLOBJECT_CLONE(Transforms);
319         IMPL_TYPED_CHILDREN(Transform,m_children.end());
320
321     protected:
322         void processChildElement(XMLObject* childXMLObject, const DOMElement* root) {
323             PROC_TYPED_CHILDREN(Transform,XMLConstants::XMLSIG_NS,false);
324             AbstractXMLObjectUnmarshaller::processChildElement(childXMLObject,root);
325         }
326     };
327
328     class XMLTOOL_DLLLOCAL RetrievalMethodImpl : public virtual RetrievalMethod,
329         public AbstractComplexElement,
330         public AbstractDOMCachingXMLObject,
331         public AbstractXMLObjectMarshaller,
332         public AbstractXMLObjectUnmarshaller
333     {
334     public:
335         virtual ~RetrievalMethodImpl() {
336             XMLString::release(&m_URI);
337             XMLString::release(&m_Type);
338         }
339
340         RetrievalMethodImpl(const XMLCh* nsURI, const XMLCh* localName, const XMLCh* prefix, const QName* schemaType)
341             : AbstractXMLObject(nsURI, localName, prefix, schemaType) {
342             init();
343         }
344             
345         RetrievalMethodImpl(const RetrievalMethodImpl& src) : AbstractXMLObject(src), AbstractDOMCachingXMLObject(src) {
346             init();
347             setURI(getURI());
348             setType(getType());
349             if (src.getTransforms())
350                 setTransforms(src.getTransforms()->cloneTransforms());
351         }
352         
353         void init() {
354             m_URI=m_Type=NULL;
355             m_Transforms=NULL;
356             m_children.push_back(NULL);
357             m_pos_Transforms=m_children.begin();
358         }
359         
360         IMPL_XMLOBJECT_CLONE(RetrievalMethod);
361         IMPL_STRING_ATTRIB(URI);
362         IMPL_STRING_ATTRIB(Type);
363         IMPL_TYPED_CHILD(Transforms);
364
365     protected:
366         void marshallAttributes(DOMElement* domElement) const {
367             MARSHALL_STRING_ATTRIB(URI,URI,NULL);
368             MARSHALL_STRING_ATTRIB(Type,TYPE,NULL);
369         }
370
371         void processChildElement(XMLObject* childXMLObject, const DOMElement* root) {
372             PROC_TYPED_CHILD(Transforms,XMLConstants::XMLSIG_NS,false);
373             AbstractXMLObjectUnmarshaller::processChildElement(childXMLObject,root);
374         }
375
376         void processAttribute(const DOMAttr* attribute) {
377             PROC_STRING_ATTRIB(URI,URI,NULL);
378             PROC_STRING_ATTRIB(Type,TYPE,NULL);
379             AbstractXMLObjectUnmarshaller::processAttribute(attribute);
380         }
381     };
382
383     class XMLTOOL_DLLLOCAL X509IssuerSerialImpl : public virtual X509IssuerSerial,
384         public AbstractComplexElement,
385         public AbstractDOMCachingXMLObject,
386         public AbstractXMLObjectMarshaller,
387         public AbstractXMLObjectUnmarshaller
388     {
389     public:
390         virtual ~X509IssuerSerialImpl() {}
391
392         X509IssuerSerialImpl(const XMLCh* nsURI, const XMLCh* localName, const XMLCh* prefix, const QName* schemaType)
393                 : AbstractXMLObject(nsURI, localName, prefix, schemaType) {
394             init();
395         }
396             
397         X509IssuerSerialImpl(const X509IssuerSerialImpl& src) : AbstractXMLObject(src), AbstractDOMCachingXMLObject(src) {
398             init();
399             if (src.getX509IssuerName())
400                 setX509IssuerName(src.getX509IssuerName()->cloneX509IssuerName());
401             if (src.getX509SerialNumber())
402                 setX509SerialNumber(src.getX509SerialNumber()->cloneX509SerialNumber());
403         }
404         
405         void init() {
406             m_X509IssuerName=NULL;
407             m_X509SerialNumber=NULL;
408             m_children.push_back(NULL);
409             m_children.push_back(NULL);
410             m_pos_X509IssuerName=m_children.begin();
411             m_pos_X509SerialNumber=m_pos_X509IssuerName;
412             ++m_pos_X509SerialNumber;
413         }
414         
415         IMPL_XMLOBJECT_CLONE(X509IssuerSerial);
416         IMPL_TYPED_CHILD(X509IssuerName);
417         IMPL_TYPED_CHILD(X509SerialNumber);
418
419     protected:
420         void processChildElement(XMLObject* childXMLObject, const DOMElement* root) {
421             PROC_TYPED_CHILD(X509IssuerName,XMLConstants::XMLSIG_NS,false);
422             PROC_TYPED_CHILD(X509SerialNumber,XMLConstants::XMLSIG_NS,false);
423             AbstractXMLObjectUnmarshaller::processChildElement(childXMLObject,root);
424         }
425     };
426
427     class XMLTOOL_DLLLOCAL X509DataImpl : public virtual X509Data,
428         public AbstractComplexElement,
429         public AbstractDOMCachingXMLObject,
430         public AbstractXMLObjectMarshaller,
431         public AbstractXMLObjectUnmarshaller
432     {
433     public:
434         virtual ~X509DataImpl() {}
435
436         X509DataImpl(const XMLCh* nsURI, const XMLCh* localName, const XMLCh* prefix, const QName* schemaType)
437             : AbstractXMLObject(nsURI, localName, prefix, schemaType) {
438         }
439             
440         X509DataImpl(const X509DataImpl& src) : AbstractXMLObject(src), AbstractDOMCachingXMLObject(src) {
441             for (list<XMLObject*>::const_iterator i=src.m_children.begin(); i!=src.m_children.end(); i++) {
442                 if (*i) {
443                     X509Certificate* xcert=dynamic_cast<X509Certificate*>(*i);
444                     if (xcert) {
445                         getX509Certificates().push_back(xcert->cloneX509Certificate());
446                         continue;
447                     }
448
449                     X509CRL* xcrl=dynamic_cast<X509CRL*>(*i);
450                     if (xcrl) {
451                         getX509CRLs().push_back(xcrl->cloneX509CRL());
452                         continue;
453                     }
454
455                     X509SubjectName* xsn=dynamic_cast<X509SubjectName*>(*i);
456                     if (xsn) {
457                         getX509SubjectNames().push_back(xsn->cloneX509SubjectName());
458                         continue;
459                     }
460
461                     X509IssuerSerial* xis=dynamic_cast<X509IssuerSerial*>(*i);
462                     if (xis) {
463                         getX509IssuerSerials().push_back(xis->cloneX509IssuerSerial());
464                         continue;
465                     }
466
467                     X509SKI* xski=dynamic_cast<X509SKI*>(*i);
468                     if (xski) {
469                         getX509SKIs().push_back(xski->cloneX509SKI());
470                         continue;
471                     }
472
473                     getOtherX509Datas().push_back((*i)->clone());
474                 }
475             }
476         }
477         
478         IMPL_XMLOBJECT_CLONE(X509Data);
479         IMPL_TYPED_CHILDREN(X509IssuerSerial,m_children.end());
480         IMPL_TYPED_CHILDREN(X509SKI,m_children.end());
481         IMPL_TYPED_CHILDREN(X509SubjectName,m_children.end());
482         IMPL_TYPED_CHILDREN(X509Certificate,m_children.end());
483         IMPL_TYPED_CHILDREN(X509CRL,m_children.end());
484         IMPL_XMLOBJECT_CHILDREN(OtherX509Data,m_children.end());
485
486     protected:
487         void processChildElement(XMLObject* childXMLObject, const DOMElement* root) {
488             PROC_TYPED_CHILDREN(X509IssuerSerial,XMLConstants::XMLSIG_NS,false);
489             PROC_TYPED_CHILDREN(X509SKI,XMLConstants::XMLSIG_NS,false);
490             PROC_TYPED_CHILDREN(X509SubjectName,XMLConstants::XMLSIG_NS,false);
491             PROC_TYPED_CHILDREN(X509Certificate,XMLConstants::XMLSIG_NS,false);
492             PROC_TYPED_CHILDREN(X509CRL,XMLConstants::XMLSIG_NS,false);
493             
494             // Unknown child.
495             const XMLCh* nsURI=root->getNamespaceURI();
496             if (!XMLString::equals(nsURI,XMLConstants::XMLSIG_NS) && nsURI && *nsURI) {
497                 getOtherX509Datas().push_back(childXMLObject);
498                 return;
499             }
500             
501             AbstractXMLObjectUnmarshaller::processChildElement(childXMLObject,root);
502         }
503     };
504
505     class XMLTOOL_DLLLOCAL SPKIDataImpl : public virtual SPKIData,
506         public AbstractComplexElement,
507         public AbstractDOMCachingXMLObject,
508         public AbstractXMLObjectMarshaller,
509         public AbstractXMLObjectUnmarshaller
510     {
511     public:
512         virtual ~SPKIDataImpl() {}
513
514         SPKIDataImpl(const XMLCh* nsURI, const XMLCh* localName, const XMLCh* prefix, const QName* schemaType)
515             : AbstractXMLObject(nsURI, localName, prefix, schemaType) {
516         }
517             
518         SPKIDataImpl(const SPKIDataImpl& src) : AbstractXMLObject(src), AbstractDOMCachingXMLObject(src) {
519             VectorOfPairs(SPKISexp,XMLObject) v=getSPKISexps();
520             for (vector< pair<SPKISexp*,XMLObject*> >::const_iterator i=src.m_SPKISexps.begin(); i!=src.m_SPKISexps.end(); i++) {
521                 if (i->first) {
522                     v.push_back(make_pair(i->first->cloneSPKISexp(),(i->second ? i->second->clone() : (XMLObject*)NULL)));
523                 }
524             }
525         }
526         
527         IMPL_XMLOBJECT_CLONE(SPKIData);
528
529     private:
530         vector< pair<SPKISexp*,XMLObject*> > m_SPKISexps;
531
532     public:
533         VectorOfPairs(SPKISexp,XMLObject) getSPKISexps() {
534             return VectorOfPairs(SPKISexp,XMLObject)(this, m_SPKISexps, &m_children, m_children.end());
535         }
536         
537         const vector< pair<SPKISexp*,XMLObject*> >& getSPKISexps() const {
538             return m_SPKISexps;
539         }
540         
541     protected:
542         void processChildElement(XMLObject* childXMLObject, const DOMElement* root) {
543             if (XMLHelper::isNodeNamed(root,XMLConstants::XMLSIG_NS,SPKISexp::LOCAL_NAME)) {
544                 SPKISexp* typesafe=dynamic_cast<SPKISexp*>(childXMLObject);
545                 if (typesafe) {
546                     getSPKISexps().push_back(make_pair(typesafe,(XMLObject*)NULL));
547                     return;
548                 }
549             }
550
551             // Unknown child (has to be paired with the last SPKISexp processed.
552             const XMLCh* nsURI=root->getNamespaceURI();
553             if (!XMLString::equals(nsURI,XMLConstants::XMLSIG_NS) && nsURI && *nsURI) {
554                 // Update second half of pair in vector, and in master list.
555                 if (!m_SPKISexps.empty() && m_SPKISexps.back().second==NULL) {
556                     m_SPKISexps.back().second=childXMLObject;
557                     m_children.back()=childXMLObject;
558                     return;
559                 }
560                 else
561                     throw UnmarshallingException("Extension element must follow ds:SPKISexp element.");
562             }
563             
564             AbstractXMLObjectUnmarshaller::processChildElement(childXMLObject,root);
565         }
566     };
567
568     class XMLTOOL_DLLLOCAL PGPDataImpl : public virtual PGPData,
569         public AbstractComplexElement,
570         public AbstractDOMCachingXMLObject,
571         public AbstractXMLObjectMarshaller,
572         public AbstractXMLObjectUnmarshaller
573     {
574     public:
575         virtual ~PGPDataImpl() {}
576
577         PGPDataImpl(const XMLCh* nsURI, const XMLCh* localName, const XMLCh* prefix, const QName* schemaType)
578                 : AbstractXMLObject(nsURI, localName, prefix, schemaType) {
579             init();
580         }
581             
582         PGPDataImpl(const PGPDataImpl& src) : AbstractXMLObject(src), AbstractDOMCachingXMLObject(src) {
583             init();
584             if (src.getPGPKeyID())
585                 setPGPKeyID(src.getPGPKeyID()->clonePGPKeyID());
586             if (src.getPGPKeyPacket())
587                 setPGPKeyPacket(src.getPGPKeyPacket()->clonePGPKeyPacket());
588             VectorOf(XMLObject) v=getPGPDataExtensions();
589             for (vector<XMLObject*>::const_iterator i=src.m_PGPDataExtensions.begin(); i!=src.m_PGPDataExtensions.end(); i++) {
590                 if (*i) {
591                     v.push_back((*i)->clone());
592                 }
593             }
594         }
595         
596         void init() {
597             m_PGPKeyID=NULL;
598             m_PGPKeyPacket=NULL;
599             m_children.push_back(NULL);
600             m_children.push_back(NULL);
601             m_pos_PGPKeyID=m_children.begin();
602             m_pos_PGPKeyPacket=m_pos_PGPKeyID;
603             ++m_pos_PGPKeyPacket;
604         }
605         
606         IMPL_XMLOBJECT_CLONE(PGPData);
607         IMPL_TYPED_CHILD(PGPKeyID);
608         IMPL_TYPED_CHILD(PGPKeyPacket);
609         IMPL_XMLOBJECT_CHILDREN(PGPDataExtension,m_children.end());
610
611     protected:
612         void processChildElement(XMLObject* childXMLObject, const DOMElement* root) {
613             PROC_TYPED_CHILD(PGPKeyID,XMLConstants::XMLSIG_NS,false);
614             PROC_TYPED_CHILD(PGPKeyPacket,XMLConstants::XMLSIG_NS,false);
615
616             // Unknown child.
617             const XMLCh* nsURI=root->getNamespaceURI();
618             if (!XMLString::equals(nsURI,XMLConstants::XMLSIG_NS) && nsURI && *nsURI) {
619                 getPGPDataExtensions().push_back(childXMLObject);
620                 return;
621             }
622
623             AbstractXMLObjectUnmarshaller::processChildElement(childXMLObject,root);
624         }
625     };
626
627     class XMLTOOL_DLLLOCAL KeyInfoImpl : public virtual KeyInfo,
628         public AbstractComplexElement,
629         public AbstractSimpleElement,
630         public AbstractDOMCachingXMLObject,
631         public AbstractXMLObjectMarshaller,
632         public AbstractXMLObjectUnmarshaller
633     {
634     public:
635         virtual ~KeyInfoImpl() {
636             XMLString::release(&m_Id);
637         }
638
639         KeyInfoImpl(const XMLCh* nsURI, const XMLCh* localName, const XMLCh* prefix, const QName* schemaType)
640             : AbstractXMLObject(nsURI, localName, prefix, schemaType), m_Id(NULL) {
641         }
642             
643         KeyInfoImpl(const KeyInfoImpl& src)
644                 : AbstractXMLObject(src), AbstractSimpleElement(src), AbstractDOMCachingXMLObject(src),
645                     m_Id(XMLString::replicate(src.m_Id)) {
646
647             for (list<XMLObject*>::const_iterator i=src.m_children.begin(); i!=src.m_children.end(); i++) {
648                 if (*i) {
649                     X509Data* xd=dynamic_cast<X509Data*>(*i);
650                     if (xd) {
651                         getX509Datas().push_back(xd->cloneX509Data());
652                         continue;
653                     }
654
655                     KeyName* kn=dynamic_cast<KeyName*>(*i);
656                     if (kn) {
657                         getKeyNames().push_back(kn->cloneKeyName());
658                         continue;
659                     }
660
661                     KeyValue* kv=dynamic_cast<KeyValue*>(*i);
662                     if (kv) {
663                         getKeyValues().push_back(kv->cloneKeyValue());
664                         continue;
665                     }
666
667                     RetrievalMethod* rm=dynamic_cast<RetrievalMethod*>(*i);
668                     if (rm) {
669                         getRetrievalMethods().push_back(rm->cloneRetrievalMethod());
670                         continue;
671                     }
672
673                     MgmtData* md=dynamic_cast<MgmtData*>(*i);
674                     if (md) {
675                         getMgmtDatas().push_back(md->cloneMgmtData());
676                         continue;
677                     }
678
679                     SPKIData* sd=dynamic_cast<SPKIData*>(*i);
680                     if (sd) {
681                         getSPKIDatas().push_back(sd->cloneSPKIData());
682                         continue;
683                     }
684
685                     PGPData* pd=dynamic_cast<PGPData*>(*i);
686                     if (pd) {
687                         getPGPDatas().push_back(pd->clonePGPData());
688                         continue;
689                     }
690
691                     getOthers().push_back((*i)->clone());
692                 }
693             }
694         }
695         
696         IMPL_XMLOBJECT_CLONE(KeyInfo);
697         IMPL_ID_ATTRIB(Id);
698         IMPL_TYPED_CHILDREN(KeyName,m_children.end());
699         IMPL_TYPED_CHILDREN(KeyValue,m_children.end());
700         IMPL_TYPED_CHILDREN(RetrievalMethod,m_children.end());
701         IMPL_TYPED_CHILDREN(X509Data,m_children.end());
702         IMPL_TYPED_CHILDREN(MgmtData,m_children.end());
703         IMPL_TYPED_CHILDREN(SPKIData,m_children.end());
704         IMPL_TYPED_CHILDREN(PGPData,m_children.end());
705         IMPL_XMLOBJECT_CHILDREN(Other,m_children.end());
706         IMPL_XMLOBJECT_CONTENT;
707
708     protected:
709         void marshallAttributes(DOMElement* domElement) const {
710             MARSHALL_ID_ATTRIB(Id,ID,NULL);
711         }
712
713         void processChildElement(XMLObject* childXMLObject, const DOMElement* root) {
714             PROC_TYPED_CHILDREN(X509Data,XMLConstants::XMLSIG_NS,false);
715             PROC_TYPED_CHILDREN(KeyName,XMLConstants::XMLSIG_NS,false);
716             PROC_TYPED_CHILDREN(KeyValue,XMLConstants::XMLSIG_NS,false);
717             PROC_TYPED_CHILDREN(RetrievalMethod,XMLConstants::XMLSIG_NS,false);
718             PROC_TYPED_CHILDREN(MgmtData,XMLConstants::XMLSIG_NS,false);
719             PROC_TYPED_CHILDREN(SPKIData,XMLConstants::XMLSIG_NS,false);
720             PROC_TYPED_CHILDREN(PGPData,XMLConstants::XMLSIG_NS,false);
721             
722             // Unknown child.
723             const XMLCh* nsURI=root->getNamespaceURI();
724             if (!XMLString::equals(nsURI,XMLConstants::XMLSIG_NS) && nsURI && *nsURI) {
725                 getOthers().push_back(childXMLObject);
726                 return;
727             }
728             
729             AbstractXMLObjectUnmarshaller::processChildElement(childXMLObject,root);
730         }
731
732         void processAttribute(const DOMAttr* attribute) {
733             PROC_ID_ATTRIB(Id,ID,NULL);
734             AbstractXMLObjectUnmarshaller::processAttribute(attribute);
735         }
736     };
737     
738     DECL_XMLOBJECTIMPL_SIMPLE(XMLTOOL_DLLLOCAL,KeyName);
739     DECL_XMLOBJECTIMPL_SIMPLE(XMLTOOL_DLLLOCAL,MgmtData);
740     DECL_XMLOBJECTIMPL_SIMPLE(XMLTOOL_DLLLOCAL,Modulus);
741     DECL_XMLOBJECTIMPL_SIMPLE(XMLTOOL_DLLLOCAL,Exponent);
742     DECL_XMLOBJECTIMPL_SIMPLE(XMLTOOL_DLLLOCAL,Seed);
743     DECL_XMLOBJECTIMPL_SIMPLE(XMLTOOL_DLLLOCAL,PgenCounter);
744     DECL_XMLOBJECTIMPL_SIMPLE(XMLTOOL_DLLLOCAL,P);
745     DECL_XMLOBJECTIMPL_SIMPLE(XMLTOOL_DLLLOCAL,Q);
746     DECL_XMLOBJECTIMPL_SIMPLE(XMLTOOL_DLLLOCAL,G);
747     DECL_XMLOBJECTIMPL_SIMPLE(XMLTOOL_DLLLOCAL,Y);
748     DECL_XMLOBJECTIMPL_SIMPLE(XMLTOOL_DLLLOCAL,J);
749     DECL_XMLOBJECTIMPL_SIMPLE(XMLTOOL_DLLLOCAL,XPath);
750     DECL_XMLOBJECTIMPL_SIMPLE(XMLTOOL_DLLLOCAL,X509IssuerName);
751     DECL_XMLOBJECTIMPL_SIMPLE(XMLTOOL_DLLLOCAL,X509SerialNumber);
752     DECL_XMLOBJECTIMPL_SIMPLE(XMLTOOL_DLLLOCAL,X509SKI);
753     DECL_XMLOBJECTIMPL_SIMPLE(XMLTOOL_DLLLOCAL,X509SubjectName);
754     DECL_XMLOBJECTIMPL_SIMPLE(XMLTOOL_DLLLOCAL,X509Certificate);
755     DECL_XMLOBJECTIMPL_SIMPLE(XMLTOOL_DLLLOCAL,X509CRL);
756     DECL_XMLOBJECTIMPL_SIMPLE(XMLTOOL_DLLLOCAL,SPKISexp);
757     DECL_XMLOBJECTIMPL_SIMPLE(XMLTOOL_DLLLOCAL,PGPKeyID);
758     DECL_XMLOBJECTIMPL_SIMPLE(XMLTOOL_DLLLOCAL,PGPKeyPacket);
759 };
760
761 #if defined (_MSC_VER)
762     #pragma warning( pop )
763 #endif
764
765 // Builder Implementations
766
767 IMPL_XMLOBJECTBUILDER(X509IssuerSerial);
768 IMPL_XMLOBJECTBUILDER(X509IssuerName);
769 IMPL_XMLOBJECTBUILDER(X509SerialNumber);
770 IMPL_XMLOBJECTBUILDER(X509SKI);
771 IMPL_XMLOBJECTBUILDER(X509SubjectName);
772 IMPL_XMLOBJECTBUILDER(X509Certificate);
773 IMPL_XMLOBJECTBUILDER(X509CRL);
774 IMPL_XMLOBJECTBUILDER(X509Data);
775 IMPL_XMLOBJECTBUILDER(XPath);
776 IMPL_XMLOBJECTBUILDER(Transform);
777 IMPL_XMLOBJECTBUILDER(Transforms);
778 IMPL_XMLOBJECTBUILDER(RetrievalMethod);
779 IMPL_XMLOBJECTBUILDER(KeyName);
780 IMPL_XMLOBJECTBUILDER(MgmtData);
781 IMPL_XMLOBJECTBUILDER(Modulus);
782 IMPL_XMLOBJECTBUILDER(Exponent);
783 IMPL_XMLOBJECTBUILDER(Seed);
784 IMPL_XMLOBJECTBUILDER(PgenCounter);
785 IMPL_XMLOBJECTBUILDER(P);
786 IMPL_XMLOBJECTBUILDER(Q);
787 IMPL_XMLOBJECTBUILDER(G);
788 IMPL_XMLOBJECTBUILDER(Y);
789 IMPL_XMLOBJECTBUILDER(J);
790 IMPL_XMLOBJECTBUILDER(DSAKeyValue);
791 IMPL_XMLOBJECTBUILDER(RSAKeyValue);
792 IMPL_XMLOBJECTBUILDER(KeyValue);
793 IMPL_XMLOBJECTBUILDER(KeyInfo);
794 IMPL_XMLOBJECTBUILDER(SPKISexp);
795 IMPL_XMLOBJECTBUILDER(SPKIData);
796 IMPL_XMLOBJECTBUILDER(PGPKeyID);
797 IMPL_XMLOBJECTBUILDER(PGPKeyPacket);
798 IMPL_XMLOBJECTBUILDER(PGPData);
799
800 // Unicode literals
801
802 const XMLCh KeyInfo::LOCAL_NAME[] =             UNICODE_LITERAL_7(K,e,y,I,n,f,o);
803 const XMLCh KeyInfo::TYPE_NAME[] =              UNICODE_LITERAL_11(K,e,y,I,n,f,o,T,y,p,e);
804 const XMLCh KeyInfo::ID_ATTRIB_NAME[] =         UNICODE_LITERAL_2(I,d);
805 const XMLCh KeyValue::LOCAL_NAME[] =            UNICODE_LITERAL_8(K,e,y,V,a,l,u,e);
806 const XMLCh KeyValue::TYPE_NAME[] =             UNICODE_LITERAL_12(K,e,y,V,a,l,u,e,T,y,p,e);
807 const XMLCh DSAKeyValue::LOCAL_NAME[] =         UNICODE_LITERAL_11(D,S,A,K,e,y,V,a,l,u,e);
808 const XMLCh DSAKeyValue::TYPE_NAME[] =          UNICODE_LITERAL_15(D,S,A,K,e,y,V,a,l,u,e,T,y,p,e);
809 const XMLCh RSAKeyValue::LOCAL_NAME[] =         UNICODE_LITERAL_11(R,S,A,K,e,y,V,a,l,u,e);
810 const XMLCh RSAKeyValue::TYPE_NAME[] =          UNICODE_LITERAL_15(R,S,A,K,e,y,V,a,l,u,e,T,y,p,e);
811 const XMLCh MgmtData::LOCAL_NAME[] =            UNICODE_LITERAL_8(M,g,m,t,D,a,t,a);
812 const XMLCh KeyName::LOCAL_NAME[] =             UNICODE_LITERAL_7(K,e,y,N,a,m,e);
813 const XMLCh Modulus::LOCAL_NAME[] =             UNICODE_LITERAL_7(M,o,d,u,l,u,s);
814 const XMLCh Exponent::LOCAL_NAME[] =            UNICODE_LITERAL_8(E,x,p,o,n,e,n,t);
815 const XMLCh Seed::LOCAL_NAME[] =                UNICODE_LITERAL_4(S,e,e,d);
816 const XMLCh PgenCounter::LOCAL_NAME[] =         UNICODE_LITERAL_11(P,g,e,n,C,o,u,n,t,e,r);
817 const XMLCh P::LOCAL_NAME[] =                   UNICODE_LITERAL_1(P);
818 const XMLCh Q::LOCAL_NAME[] =                   UNICODE_LITERAL_1(Q);
819 const XMLCh G::LOCAL_NAME[] =                   UNICODE_LITERAL_1(G);
820 const XMLCh Y::LOCAL_NAME[] =                   UNICODE_LITERAL_1(Y);
821 const XMLCh J::LOCAL_NAME[] =                   UNICODE_LITERAL_1(J);
822 const XMLCh XPath::LOCAL_NAME[] =               UNICODE_LITERAL_5(X,P,a,t,h);
823 const XMLCh Transform::LOCAL_NAME[] =           UNICODE_LITERAL_9(T,r,a,n,s,f,o,r,m);
824 const XMLCh Transform::TYPE_NAME[] =            UNICODE_LITERAL_13(T,r,a,n,s,f,o,r,m,T,y,p,e);
825 const XMLCh Transform::ALGORITHM_ATTRIB_NAME[] = UNICODE_LITERAL_9(A,l,g,o,r,i,t,h,m);
826 const XMLCh Transforms::LOCAL_NAME[] =          UNICODE_LITERAL_10(T,r,a,n,s,f,o,r,m,s);
827 const XMLCh Transforms::TYPE_NAME[] =           UNICODE_LITERAL_14(T,r,a,n,s,f,o,r,m,s,T,y,p,e);
828 const XMLCh RetrievalMethod::LOCAL_NAME[] =     UNICODE_LITERAL_15(R,e,t,r,i,e,v,a,l,M,e,t,h,o,d);
829 const XMLCh RetrievalMethod::TYPE_NAME[] =      UNICODE_LITERAL_19(R,e,t,r,i,e,v,a,l,M,e,t,h,o,d,T,y,p,e);
830 const XMLCh RetrievalMethod::URI_ATTRIB_NAME[] = UNICODE_LITERAL_3(U,R,I);
831 const XMLCh RetrievalMethod::TYPE_ATTRIB_NAME[] = UNICODE_LITERAL_4(T,y,p,e);
832 const XMLCh SPKISexp::LOCAL_NAME[] =            UNICODE_LITERAL_8(S,P,K,I,S,e,x,p);
833 const XMLCh SPKIData::LOCAL_NAME[] =            UNICODE_LITERAL_8(S,P,K,I,D,a,t,a);
834 const XMLCh SPKIData::TYPE_NAME[] =             UNICODE_LITERAL_12(S,P,K,I,D,a,t,a,T,y,p,e);
835 const XMLCh PGPKeyID::LOCAL_NAME[] =            UNICODE_LITERAL_8(P,G,P,K,e,y,I,D);
836 const XMLCh PGPKeyPacket::LOCAL_NAME[] =        UNICODE_LITERAL_12(P,G,P,K,e,y,P,a,c,k,e,t);
837 const XMLCh PGPData::LOCAL_NAME[] =             UNICODE_LITERAL_7(P,G,P,D,a,t,a);
838 const XMLCh PGPData::TYPE_NAME[] =              UNICODE_LITERAL_11(P,G,P,D,a,t,a,T,y,p,e);
839
840 #define XCH(ch) chLatin_##ch
841 #define XNUM(d) chDigit_##d
842
843 const XMLCh X509Data::LOCAL_NAME[] = {
844     XCH(X), XNUM(5), XNUM(0), XNUM(9), XCH(D), XCH(a), XCH(t), XCH(a), chNull
845     };
846 const XMLCh X509Data::TYPE_NAME[] = {
847     XCH(X), XNUM(5), XNUM(0), XNUM(9), XCH(D), XCH(a), XCH(t), XCH(a), XCH(T), XCH(y), XCH(p), XCH(e), chNull
848     };
849 const XMLCh X509IssuerSerial::LOCAL_NAME[] = {
850     XCH(X), XNUM(5), XNUM(0), XNUM(9), XCH(I), XCH(s), XCH(s), XCH(u), XCH(e), XCH(r),
851     XCH(S), XCH(e), XCH(r), XCH(i), XCH(a), XCH(l), chNull
852     };
853 const XMLCh X509IssuerSerial::TYPE_NAME[] = {
854     XCH(X), XNUM(5), XNUM(0), XNUM(9), XCH(I), XCH(s), XCH(s), XCH(u), XCH(e), XCH(r),
855     XCH(S), XCH(e), XCH(r), XCH(i), XCH(a), XCH(l), XCH(T), XCH(y), XCH(p), XCH(e), chNull
856     };
857 const XMLCh X509IssuerName::LOCAL_NAME[] = {
858     XCH(X), XNUM(5), XNUM(0), XNUM(9), XCH(I), XCH(s), XCH(s), XCH(u), XCH(e), XCH(r),
859     XCH(N), XCH(a), XCH(m), XCH(e), chNull
860     };
861 const XMLCh X509SerialNumber::LOCAL_NAME[] = {
862     XCH(X), XNUM(5), XNUM(0), XNUM(9), XCH(S), XCH(e), XCH(r), XCH(i), XCH(a), XCH(l),
863     XCH(N), XCH(u), XCH(m), XCH(b), XCH(e), XCH(r), chNull
864     };
865 const XMLCh X509SKI::LOCAL_NAME[] = { XCH(X), XNUM(5), XNUM(0), XNUM(9), XCH(S), XCH(K), XCH(I), chNull };
866 const XMLCh X509SubjectName::LOCAL_NAME[] = {
867     XCH(X), XNUM(5), XNUM(0), XNUM(9), XCH(S), XCH(u), XCH(b), XCH(j), XCH(e), XCH(c), XCH(t),
868     XCH(N), XCH(a), XCH(m), XCH(e), chNull
869     };
870 const XMLCh X509Certificate::LOCAL_NAME[] = {
871     XCH(X), XNUM(5), XNUM(0), XNUM(9),
872     XCH(C), XCH(e), XCH(r), XCH(t), XCH(i), XCH(f), XCH(i), XCH(c), XCH(a), XCH(t), XCH(e), chNull
873     };
874 const XMLCh X509CRL::LOCAL_NAME[] = { XCH(X), XNUM(5), XNUM(0), XNUM(9), XCH(C), XCH(R), XCH(L), chNull };
875