Update copyright.
[shibboleth/cpp-opensaml.git] / saml / saml2 / metadata / impl / XMLMetadataProvider.cpp
1 /*
2  *  Copyright 2001-2007 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  * XMLMetadataProvider.cpp
19  * 
20  * Supplies metadata from an XML file
21  */
22
23 #include "internal.h"
24 #include "saml2/metadata/Metadata.h"
25 #include "saml2/metadata/AbstractMetadataProvider.h"
26
27 #include <log4cpp/Category.hh>
28 #include <xmltooling/util/NDC.h>
29 #include <xmltooling/util/ReloadableXMLFile.h>
30
31 using namespace opensaml::saml2md;
32 using namespace xmltooling;
33 using namespace log4cpp;
34 using namespace std;
35
36 #if defined (_MSC_VER)
37     #pragma warning( push )
38     #pragma warning( disable : 4250 )
39 #endif
40
41 namespace opensaml {
42     namespace saml2md {
43
44         class SAML_DLLLOCAL XMLMetadataProvider : public AbstractMetadataProvider, public ReloadableXMLFile
45         {
46         public:
47             XMLMetadataProvider(const DOMElement* e) : AbstractMetadataProvider(e), ReloadableXMLFile(e), m_object(NULL) {}
48             virtual ~XMLMetadataProvider() {
49                 delete m_object;
50             }
51
52             void init() {
53                 load(); // guarantees an exception or the metadata is loaded
54             }
55             
56             const XMLObject* getMetadata() const {
57                 return m_object;
58             }
59
60         protected:
61             pair<bool,DOMElement*> load();
62
63             bool isValid() const {
64                 const TimeBoundSAMLObject* bound=dynamic_cast<const TimeBoundSAMLObject*>(m_object);
65                 return bound ? bound->isValid() : false;
66             }
67
68         private:
69             void index();
70         
71             XMLObject* m_object;
72         }; 
73
74         MetadataProvider* SAML_DLLLOCAL XMLMetadataProviderFactory(const DOMElement* const & e)
75         {
76             return new XMLMetadataProvider(e);
77         }
78
79     };
80 };
81
82 #if defined (_MSC_VER)
83     #pragma warning( pop )
84 #endif
85
86 pair<bool,DOMElement*> XMLMetadataProvider::load()
87 {
88     // Load from source using base class.
89     pair<bool,DOMElement*> raw = ReloadableXMLFile::load();
90     
91     // If we own it, wrap it for now.
92     XercesJanitor<DOMDocument> docjanitor(raw.first ? raw.second->getOwnerDocument() : NULL);
93             
94     // Unmarshall objects, binding the document.
95     auto_ptr<XMLObject> xmlObject(XMLObjectBuilder::buildOneFromElement(raw.second, true));
96     docjanitor.release();
97
98     if (!dynamic_cast<const EntitiesDescriptor*>(xmlObject.get()) && !dynamic_cast<const EntityDescriptor*>(xmlObject.get()))
99         throw MetadataException(
100             "Root of metadata instance not recognized: $1", params(1,xmlObject->getElementQName().toString().c_str())
101             );
102     
103     // Preprocess the metadata.
104     doFilters(*xmlObject.get());
105     xmlObject->releaseThisAndChildrenDOM();
106     xmlObject->setDocument(NULL);
107     
108     // Swap it in.
109     bool changed = m_object!=NULL;
110     delete m_object;
111     m_object = xmlObject.release();
112     index();
113     if (changed)
114         emitChangeEvent();
115     return make_pair(false,(DOMElement*)NULL);
116 }
117
118 void XMLMetadataProvider::index()
119 {
120     clearDescriptorIndex();
121     EntitiesDescriptor* group=dynamic_cast<EntitiesDescriptor*>(m_object);
122     if (group) {
123         AbstractMetadataProvider::index(group, SAMLTIME_MAX);
124         return;
125     }
126     EntityDescriptor* site=dynamic_cast<EntityDescriptor*>(m_object);
127     AbstractMetadataProvider::index(site, SAMLTIME_MAX);
128 }