blob: 340097a541f3d59c04a11d6020b39dd8705f0883 [file] [log] [blame]
/**
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
/*
* XSEC
*
* XKMSRegisterRequestImpl := Implementation for RegisterRequest Messages
*
* $Id$
*
*/
// XSEC Includes
#include <xsec/dsig/DSIGReference.hpp>
#include <xsec/framework/XSECDefs.hpp>
#include <xsec/framework/XSECEnv.hpp>
#include <xsec/framework/XSECError.hpp>
#ifdef XSEC_XKMS_ENABLED
#include "../../utils/XSECDOMUtils.hpp"
#include "XKMSRegisterRequestImpl.hpp"
#include "XKMSAuthenticationImpl.hpp"
#include "XKMSPrototypeKeyBindingImpl.hpp"
#include <xsec/xkms/XKMSConstants.hpp>
#include <xercesc/dom/DOM.hpp>
#include <xercesc/util/XMLUniDefs.hpp>
XERCES_CPP_NAMESPACE_USE
// --------------------------------------------------------------------------------
// Construct/Destruct
// --------------------------------------------------------------------------------
XKMSRegisterRequestImpl::XKMSRegisterRequestImpl(const XSECEnv * env) :
m_request(env),
m_msg(m_request.m_msg),
mp_authentication(NULL),
mp_prototypeKeyBinding(NULL),
mp_proofOfPossessionSignature(NULL) {
}
XKMSRegisterRequestImpl::XKMSRegisterRequestImpl(const XSECEnv * env, DOMElement * node) :
m_request(env, node),
m_msg(m_request.m_msg),
mp_authentication(NULL),
mp_prototypeKeyBinding(NULL),
mp_proofOfPossessionSignature(NULL) {
}
XKMSRegisterRequestImpl::~XKMSRegisterRequestImpl() {
if (mp_authentication != NULL)
delete mp_authentication;
if (mp_prototypeKeyBinding != NULL)
delete mp_prototypeKeyBinding;
// Provider will take care of the proofOfPossession signature
}
// --------------------------------------------------------------------------------
// Load
// --------------------------------------------------------------------------------
void XKMSRegisterRequestImpl::load(void) {
if (m_msg.mp_messageAbstractTypeElement == NULL) {
// Attempt to load an empty element
throw XSECException(XSECException::XKMSError,
"XKMSRegisterRequest::load - called on empty DOM");
}
if (!strEquals(getXKMSLocalName(m_msg.mp_messageAbstractTypeElement),
XKMSConstants::s_tagRegisterRequest)) {
throw XSECException(XSECException::XKMSError,
"XKMSRegisterRequest::load - called on incorrect node");
}
// Load the base message
m_request.load();
// Now check for any PrototypeKeyBinding elements
DOMElement * tmpElt = findFirstElementChild(m_msg.mp_messageAbstractTypeElement);
while (tmpElt != NULL && !strEquals(getXKMSLocalName(tmpElt), XKMSConstants::s_tagPrototypeKeyBinding)) {
tmpElt = findNextElementChild(tmpElt);
}
if (tmpElt != NULL) {
XSECnew(mp_prototypeKeyBinding, XKMSPrototypeKeyBindingImpl(m_msg.mp_env, tmpElt));
mp_prototypeKeyBinding->load();
tmpElt = findNextElementChild(tmpElt);
}
else {
throw XSECException(XSECException::ExpectedXKMSChildNotFound,
"XKMSRegisterRequest::load - Expected PrototypeKeyBinding node");
}
// Authentication Element
if (tmpElt != NULL && strEquals(getXKMSLocalName(tmpElt), XKMSConstants::s_tagAuthentication)) {
XSECnew(mp_authentication, XKMSAuthenticationImpl(m_msg.mp_env, tmpElt));
mp_authentication->load(mp_prototypeKeyBinding->getId());
tmpElt = findNextElementChild(tmpElt);
}
else {
throw XSECException(XSECException::ExpectedXKMSChildNotFound,
"XKMSRegisterRequest::load - Expected Authentication node");
}
if (tmpElt != NULL && strEquals(getXKMSLocalName(tmpElt), XKMSConstants::s_tagProofOfPossession)) {
// Find the signature
DOMElement * sigElt = (DOMElement *) findFirstElementChild(tmpElt);
if (sigElt == NULL || !strEquals(getDSIGLocalName(sigElt),
XKMSConstants::s_tagSignature)) {
throw XSECException(XSECException::ExpectedXKMSChildNotFound,
"XKMSRegisterRequest::load - Expected Signature child of ProofOfPossession");
}
// The provider will take care of cleaning this up later.
mp_proofOfPossessionSignature = m_prov.newSignatureFromDOM(m_msg.mp_env->getParentDocument(),
sigElt);
mp_proofOfPossessionSignature->load();
// Check the signature is across the correct input
DSIGReferenceList * rl =
mp_proofOfPossessionSignature->getReferenceList();
if (rl->getSize() != 1) {
throw XSECException(XSECException::XKMSError,
"XKMSRegisterRequestImpl::load - ProofOfPossession Signature with incorrect number of references found (should be 1)");
}
safeBuffer sb;
sb.sbXMLChIn(DSIGConstants::s_unicodeStrEmpty);
sb.sbXMLChAppendCh(chPound);
sb.sbXMLChCat(mp_prototypeKeyBinding->getId());
if (!strEquals(rl->item(0)->getURI(), sb.rawXMLChBuffer())) {
throw XSECException(XSECException::XKMSError,
"XKMSRegisterRequestImpl::load - ProofOfPossession Signature refers to incorrect Id (should be for PrototypeKeyBinding)");
}
// We don't actually check the signature as we have no key material to do so!
}
}
// --------------------------------------------------------------------------------
// Create
// --------------------------------------------------------------------------------
DOMElement * XKMSRegisterRequestImpl::
createBlankRegisterRequest(const XMLCh * service, const XMLCh * id) {
return m_request.createBlankRequestAbstractType(
XKMSConstants::s_tagRegisterRequest, service, id);
}
// --------------------------------------------------------------------------------
// MessageType
// --------------------------------------------------------------------------------
XKMSMessageAbstractType::messageType XKMSRegisterRequestImpl::getMessageType(void) {
return XKMSMessageAbstractTypeImpl::RegisterRequest;
}
// --------------------------------------------------------------------------------
// Get Methods
// --------------------------------------------------------------------------------
XKMSPrototypeKeyBinding * XKMSRegisterRequestImpl::getPrototypeKeyBinding(void) const {
return mp_prototypeKeyBinding;
}
XKMSAuthentication * XKMSRegisterRequestImpl::getAuthentication (void) const {
return mp_authentication;
}
DSIGSignature * XKMSRegisterRequestImpl::getProofOfPossessionSignature(void) const {
return mp_proofOfPossessionSignature;
}
// --------------------------------------------------------------------------------
// Set Methods
// --------------------------------------------------------------------------------
XKMSPrototypeKeyBinding * XKMSRegisterRequestImpl::addPrototypeKeyBinding(void) {
if (mp_prototypeKeyBinding != NULL)
return mp_prototypeKeyBinding;
// OK - Nothing exists, so we need to create from scratch
XSECnew(mp_prototypeKeyBinding, XKMSPrototypeKeyBindingImpl(m_msg.mp_env));
DOMElement * elt = mp_prototypeKeyBinding->createBlankPrototypeKeyBinding();
// Insert
DOMElement * be = findFirstElementChild(m_msg.mp_messageAbstractTypeElement);
while (be != NULL &&
!strEquals(getXKMSLocalName(be), XKMSConstants::s_tagAuthentication) &&
!strEquals(getXKMSLocalName(be), XKMSConstants::s_tagProofOfPossession)) {
be = findNextElementChild(be);
}
if (be == NULL) {
m_msg.mp_env->doPrettyPrint(m_msg.mp_messageAbstractTypeElement);
m_msg.mp_messageAbstractTypeElement->appendChild(elt);
m_msg.mp_env->doPrettyPrint(m_msg.mp_messageAbstractTypeElement);
return mp_prototypeKeyBinding;
}
m_msg.mp_messageAbstractTypeElement->insertBefore(elt, be);
if (m_msg.mp_env->getPrettyPrintFlag() == true) {
m_msg.mp_messageAbstractTypeElement->insertBefore(
m_msg.mp_env->getParentDocument()->createTextNode(DSIGConstants::s_unicodeStrNL),
be);
}
return mp_prototypeKeyBinding;
}
XKMSAuthentication * XKMSRegisterRequestImpl::addAuthentication(void) {
if (mp_authentication != NULL)
return mp_authentication;
if (mp_prototypeKeyBinding == NULL) {
throw XSECException(XSECException::XKMSError,
"XKMSRegisterRequestImpl::addAuthentication - called prior to key infos being added");
}
XSECnew(mp_authentication, XKMSAuthenticationImpl(m_msg.mp_env));
DOMElement * e =
mp_authentication->createBlankAuthentication(mp_prototypeKeyBinding->getId());
DOMElement * be = findFirstElementChild(m_msg.mp_messageAbstractTypeElement);
while (be != NULL && !strEquals(getXKMSLocalName(be), XKMSConstants::s_tagProofOfPossession))
be = findNextElementChild(be);
if (be == NULL) {
m_msg.mp_env->doPrettyPrint(m_msg.mp_messageAbstractTypeElement);
m_msg.mp_messageAbstractTypeElement->appendChild(e);
m_msg.mp_env->doPrettyPrint(m_msg.mp_messageAbstractTypeElement);
return mp_authentication;
}
m_msg.mp_messageAbstractTypeElement->insertBefore(e, be);
if (m_msg.mp_env->getPrettyPrintFlag() == true) {
m_msg.mp_messageAbstractTypeElement->insertBefore(
m_msg.mp_env->getParentDocument()->createTextNode(DSIGConstants::s_unicodeStrNL),
be);
}
return mp_authentication;
}
DSIGSignature * XKMSRegisterRequestImpl::addProofOfPossessionSignature(
const XMLCh* c14nAlgorithm,
const XMLCh* signatureAlgorithm,
const XMLCh* hashAlgorithm) {
DSIGSignature * ret = m_prov.newSignature();
DOMElement * elt = ret->createBlankSignature(m_msg.mp_env->getParentDocument(), c14nAlgorithm, signatureAlgorithm);
/* Create the enveloping reference */
safeBuffer sb;
sb.sbXMLChIn(DSIGConstants::s_unicodeStrEmpty);
sb.sbXMLChAppendCh(chPound);
sb.sbXMLChCat(mp_prototypeKeyBinding->getId());
DSIGReference *ref = ret->createReference(sb.rawXMLChBuffer(), hashAlgorithm);
ref->appendCanonicalizationTransform(DSIGConstants::s_unicodeStrURIEXC_C14N_COM);
/* Embed the signature in the document inside a KeyBindingAuthentication element */
safeBuffer str;
DOMDocument *doc = m_msg.mp_env->getParentDocument();
const XMLCh * prefix = m_msg.mp_env->getXKMSNSPrefix();
makeQName(str, prefix, XKMSConstants::s_tagProofOfPossession);
DOMElement * t = doc->createElementNS(XKMSConstants::s_unicodeStrURIXKMS,
str.rawXMLChBuffer());
m_msg.mp_env->doPrettyPrint(t);
t->appendChild(elt);
m_msg.mp_env->doPrettyPrint(t);
// Now append into the RegisterRequest
m_msg.mp_messageAbstractTypeElement->appendChild(t);
m_msg.mp_env->doPrettyPrint(m_msg.mp_messageAbstractTypeElement);
return ret;
}
#endif /* XSEC_XKMS_ENABLED */