2 * Copyright 2001-2010 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.
18 * shibsp/attribute/Attribute.cpp
20 * A resolved attribute.
24 #include "exceptions.h"
27 # include "attribute/AttributeDecoder.h"
29 #include "attribute/SimpleAttribute.h"
30 #include "attribute/ScopedAttribute.h"
31 #include "attribute/NameIDAttribute.h"
32 #include "attribute/ExtensibleAttribute.h"
33 #include "attribute/XMLAttribute.h"
34 #include "util/SPConstants.h"
36 #include <xercesc/util/XMLUniDefs.hpp>
37 #include <xmltooling/security/SecurityHelper.h>
39 using namespace shibsp;
40 using namespace xmltooling;
44 SHIBSP_DLLLOCAL Attribute* SimpleAttributeFactory(DDF& in);
45 SHIBSP_DLLLOCAL Attribute* ScopedAttributeFactory(DDF& in);
46 SHIBSP_DLLLOCAL Attribute* NameIDAttributeFactory(DDF& in);
47 SHIBSP_DLLLOCAL Attribute* ExtensibleAttributeFactory(DDF& in);
48 SHIBSP_DLLLOCAL Attribute* XMLAttributeFactory(DDF& in);
51 SHIBSP_DLLLOCAL PluginManager<AttributeDecoder,xmltooling::QName,const DOMElement*>::Factory StringAttributeDecoderFactory;
52 SHIBSP_DLLLOCAL PluginManager<AttributeDecoder,xmltooling::QName,const DOMElement*>::Factory ScopedAttributeDecoderFactory;
53 SHIBSP_DLLLOCAL PluginManager<AttributeDecoder,xmltooling::QName,const DOMElement*>::Factory NameIDAttributeDecoderFactory;
54 SHIBSP_DLLLOCAL PluginManager<AttributeDecoder,xmltooling::QName,const DOMElement*>::Factory NameIDFromScopedAttributeDecoderFactory;
55 SHIBSP_DLLLOCAL PluginManager<AttributeDecoder,xmltooling::QName,const DOMElement*>::Factory KeyInfoAttributeDecoderFactory;
56 SHIBSP_DLLLOCAL PluginManager<AttributeDecoder,xmltooling::QName,const DOMElement*>::Factory DOMAttributeDecoderFactory;
57 SHIBSP_DLLLOCAL PluginManager<AttributeDecoder,xmltooling::QName,const DOMElement*>::Factory XMLAttributeDecoderFactory;
59 static const XMLCh _StringAttributeDecoder[] = UNICODE_LITERAL_22(S,t,r,i,n,g,A,t,t,r,i,b,u,t,e,D,e,c,o,d,e,r);
60 static const XMLCh _ScopedAttributeDecoder[] = UNICODE_LITERAL_22(S,c,o,p,e,d,A,t,t,r,i,b,u,t,e,D,e,c,o,d,e,r);
61 static const XMLCh _NameIDAttributeDecoder[] = UNICODE_LITERAL_22(N,a,m,e,I,D,A,t,t,r,i,b,u,t,e,D,e,c,o,d,e,r);
62 static const XMLCh _NameIDFromScopedAttributeDecoder[] = UNICODE_LITERAL_32(N,a,m,e,I,D,F,r,o,m,S,c,o,p,e,d,A,t,t,r,i,b,u,t,e,D,e,c,o,d,e,r);
63 static const XMLCh _KeyInfoAttributeDecoder[] =UNICODE_LITERAL_23(K,e,y,I,n,f,o,A,t,t,r,i,b,u,t,e,D,e,c,o,d,e,r);
64 static const XMLCh _DOMAttributeDecoder[] = UNICODE_LITERAL_19(D,O,M,A,t,t,r,i,b,u,t,e,D,e,c,o,d,e,r);
65 static const XMLCh _XMLAttributeDecoder[] = UNICODE_LITERAL_19(X,M,L,A,t,t,r,i,b,u,t,e,D,e,c,o,d,e,r);
67 static const XMLCh caseSensitive[] = UNICODE_LITERAL_13(c,a,s,e,S,e,n,s,i,t,i,v,e);
68 static const XMLCh hashAlg[] = UNICODE_LITERAL_7(h,a,s,h,A,l,g);
69 static const XMLCh internal[] = UNICODE_LITERAL_8(i,n,t,e,r,n,a,l);
74 xmltooling::QName shibsp::StringAttributeDecoderType(shibspconstants::SHIB2ATTRIBUTEMAP_NS, _StringAttributeDecoder);
75 xmltooling::QName shibsp::ScopedAttributeDecoderType(shibspconstants::SHIB2ATTRIBUTEMAP_NS, _ScopedAttributeDecoder);
76 xmltooling::QName shibsp::NameIDAttributeDecoderType(shibspconstants::SHIB2ATTRIBUTEMAP_NS, _NameIDAttributeDecoder);
77 xmltooling::QName shibsp::NameIDFromScopedAttributeDecoderType(shibspconstants::SHIB2ATTRIBUTEMAP_NS, _NameIDFromScopedAttributeDecoder);
78 xmltooling::QName shibsp::KeyInfoAttributeDecoderType(shibspconstants::SHIB2ATTRIBUTEMAP_NS, _KeyInfoAttributeDecoder);
79 xmltooling::QName shibsp::DOMAttributeDecoderType(shibspconstants::SHIB2ATTRIBUTEMAP_NS, _DOMAttributeDecoder);
80 xmltooling::QName shibsp::XMLAttributeDecoderType(shibspconstants::SHIB2ATTRIBUTEMAP_NS, _XMLAttributeDecoder);
82 void shibsp::registerAttributeDecoders()
84 SPConfig& conf = SPConfig::getConfig();
85 conf.AttributeDecoderManager.registerFactory(StringAttributeDecoderType, StringAttributeDecoderFactory);
86 conf.AttributeDecoderManager.registerFactory(ScopedAttributeDecoderType, ScopedAttributeDecoderFactory);
87 conf.AttributeDecoderManager.registerFactory(NameIDAttributeDecoderType, NameIDAttributeDecoderFactory);
88 conf.AttributeDecoderManager.registerFactory(NameIDFromScopedAttributeDecoderType, NameIDFromScopedAttributeDecoderFactory);
89 conf.AttributeDecoderManager.registerFactory(KeyInfoAttributeDecoderType, KeyInfoAttributeDecoderFactory);
90 conf.AttributeDecoderManager.registerFactory(DOMAttributeDecoderType, DOMAttributeDecoderFactory);
91 conf.AttributeDecoderManager.registerFactory(XMLAttributeDecoderType, XMLAttributeDecoderFactory);
94 AttributeDecoder::AttributeDecoder(const DOMElement *e)
95 : m_caseSensitive(true), m_internal(false), m_hashAlg(e ? e->getAttributeNS(nullptr, hashAlg) : nullptr)
98 const XMLCh* flag = e->getAttributeNS(nullptr, caseSensitive);
99 if (flag && (*flag == chLatin_f || *flag == chDigit_0))
100 m_caseSensitive = false;
102 flag = e->getAttributeNS(nullptr, internal);
103 if (flag && (*flag == chLatin_t || *flag == chDigit_1))
108 AttributeDecoder::~AttributeDecoder()
112 Attribute* AttributeDecoder::_decode(Attribute* attr) const
115 attr->setCaseSensitive(m_caseSensitive);
116 attr->setInternal(m_internal);
118 if (m_hashAlg.get() && *m_hashAlg.get()) {
119 // We turn the values into strings using the supplied hash algorithm and return a SimpleAttribute instead.
120 auto_ptr<SimpleAttribute> simple(new SimpleAttribute(attr->getAliases()));
121 simple->setCaseSensitive(false);
122 simple->setInternal(m_internal);
123 vector<string>& newdest = simple->getValues();
124 const vector<string>& serialized = attr->getSerializedValues();
125 for (vector<string>::const_iterator ser = serialized.begin(); ser != serialized.end(); ++ser) {
126 newdest.push_back(SecurityHelper::doHash(m_hashAlg.get(), ser->data(), ser->length()));
127 if (newdest.back().empty())
131 return newdest.empty() ? nullptr : simple.release();
139 void shibsp::registerAttributeFactories()
141 Attribute::registerFactory("", SimpleAttributeFactory);
142 Attribute::registerFactory("Simple", SimpleAttributeFactory);
143 Attribute::registerFactory("Scoped", ScopedAttributeFactory);
144 Attribute::registerFactory("NameID", NameIDAttributeFactory);
145 Attribute::registerFactory("Extensible", ExtensibleAttributeFactory);
146 Attribute::registerFactory("XML", XMLAttributeFactory);
149 map<string,Attribute::AttributeFactory*> Attribute::m_factoryMap;
151 void Attribute::registerFactory(const char* type, AttributeFactory* factory)
153 m_factoryMap[type] = factory;
156 void Attribute::deregisterFactory(const char* type)
158 m_factoryMap.erase(type);
161 void Attribute::deregisterFactories()
163 m_factoryMap.clear();
166 Attribute::Attribute(const vector<string>& ids) : m_id(ids), m_caseSensitive(true), m_internal(false)
170 Attribute::Attribute(DDF& in) : m_caseSensitive(in["case_insensitive"].isnull()), m_internal(!in["internal"].isnull())
172 const char* id = in.first().name();
176 throw AttributeException("No id found in marshalled attribute content.");
177 DDF aliases = in["aliases"];
178 if (aliases.islist()) {
179 DDF alias = aliases.first();
180 while (alias.isstring()) {
181 m_id.push_back(alias.string());
182 alias = aliases.next();
187 Attribute::~Attribute()
191 const char* Attribute::getId() const
193 return m_id.front().c_str();
196 const vector<string>& Attribute::getAliases() const
201 vector<string>& Attribute::getAliases()
206 void Attribute::setCaseSensitive(bool caseSensitive)
208 m_caseSensitive = caseSensitive;
211 void Attribute::setInternal(bool internal)
213 m_internal = internal;
216 bool Attribute::isCaseSensitive() const
218 return m_caseSensitive;
221 bool Attribute::isInternal() const
226 size_t Attribute::valueCount() const
228 return m_serialized.size();
231 const vector<string>& Attribute::getSerializedValues() const
236 const char* Attribute::getString(size_t index) const
238 return m_serialized[index].c_str();
241 const char* Attribute::getScope(size_t index) const
246 void Attribute::removeValue(size_t index)
248 if (index < m_serialized.size())
249 m_serialized.erase(m_serialized.begin() + index);
252 DDF Attribute::marshall() const
255 ddf.structure().addmember(m_id.front().c_str()).list();
256 if (!m_caseSensitive)
257 ddf.addmember("case_insensitive");
259 ddf.addmember("internal");
260 if (m_id.size() > 1) {
262 DDF aliases = ddf.addmember("aliases").list();
263 for (std::vector<std::string>::const_iterator a = m_id.begin() + 1; a != m_id.end(); ++a) {
264 alias = DDF(nullptr).string(a->c_str());
271 Attribute* Attribute::unmarshall(DDF& in)
273 map<string,AttributeFactory*>::const_iterator i = m_factoryMap.find(in.name() ? in.name() : "");
274 if (i == m_factoryMap.end())
275 throw AttributeException("No registered factory for Attribute of type ($1).", params(1,in.name()));
276 return (i->second)(in);