Reducing header overuse, non-inlining selected methods (CPPOST-35).
[shibboleth/cpp-opensaml.git] / saml / saml2 / profile / impl / BrowserSSOProfile20Validator.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  * BrowserSSOProfile20Validator.cpp
19  *
20  * SAML 2.0 Browser SSO Profile Assertion Validator
21  */
22
23 #include "internal.h"
24 #include "saml2/core/Assertions.h"
25 #include "saml2/profile/BrowserSSOProfileValidator.h"
26
27 #include <xmltooling/logging.h>
28 #include <xmltooling/XMLToolingConfig.h>
29 #include <xmltooling/util/NDC.h>
30
31 using namespace opensaml::saml2;
32 using namespace xmltooling::logging;
33 using namespace xmltooling;
34 using namespace std;
35
36 void BrowserSSOProfileValidator::validateAssertion(const Assertion& assertion) const
37 {
38 #ifdef _DEBUG
39     xmltooling::NDC ndc("validate");
40 #endif
41     Category& log = Category::getInstance(SAML_LOGCAT".AssertionValidator");
42
43     // The assertion MUST have proper confirmation requirements.
44     const char* msg="assertion is missing bearer SubjectConfirmation";
45     const Subject* subject = assertion.getSubject();
46     if (subject) {
47         const vector<SubjectConfirmation*>& confs = subject->getSubjectConfirmations();
48         for (vector<SubjectConfirmation*>::const_iterator sc = confs.begin(); sc!=confs.end(); ++sc) {
49             if (XMLString::equals((*sc)->getMethod(), SubjectConfirmation::BEARER)) {
50                 const SubjectConfirmationDataType* data = dynamic_cast<const SubjectConfirmationDataType*>((*sc)->getSubjectConfirmationData());
51
52                 if (m_destination.get()) {
53                     if (!XMLString::equals(m_destination.get(), data ? data->getRecipient() : NULL)) {
54                         msg = "bearer confirmation failed with recipient mismatch";
55                         continue;
56                     }
57                 }
58
59                 if (m_requestID.get()) {
60                     if (!XMLString::equals(m_requestID.get(), data ? data->getInResponseTo() : NULL)) {
61                         msg = "bearer confirmation failed with request correlation mismatch";
62                         continue;
63                     }
64                 }
65
66                 if (m_ts) {
67                     if (!data || !data->getNotOnOrAfter()) {
68                         msg = "bearer confirmation missing NotOnOrAfter attribute";
69                         continue;
70                     }
71                     else if (data->getNotOnOrAfterEpoch() <= m_ts - XMLToolingConfig::getConfig().clock_skew_secs) {
72                         msg = "bearer confirmation has expired";
73                         continue;
74                     }
75                 }
76
77                 // Save off client address.
78                 if (data) {
79                     auto_ptr_char ip(data->getAddress());
80                     if (ip.get())
81                         m_address = ip.get();
82                 }
83
84                 // Pass up for additional checking.
85                 return AssertionValidator::validateAssertion(assertion);
86             }
87         }
88     }
89
90     log.error(msg ? msg : "no error message");
91     throw ValidationException("Unable to locate satisfiable bearer SubjectConfirmation in assertion.");
92 }