|
|
|
|
|
|
|
*/ |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
*/ |
|
package com.sun.org.apache.xml.internal.security.keys.keyresolver.implementations; |
|
|
|
import java.io.IOException; |
|
import java.security.PrivateKey; |
|
import java.security.PublicKey; |
|
import java.security.cert.X509Certificate; |
|
|
|
import javax.crypto.SecretKey; |
|
import javax.xml.namespace.QName; |
|
import javax.xml.parsers.ParserConfigurationException; |
|
|
|
import com.sun.org.apache.xml.internal.security.c14n.CanonicalizationException; |
|
import com.sun.org.apache.xml.internal.security.exceptions.XMLSecurityException; |
|
import com.sun.org.apache.xml.internal.security.keys.KeyInfo; |
|
import com.sun.org.apache.xml.internal.security.keys.content.KeyInfoReference; |
|
import com.sun.org.apache.xml.internal.security.keys.keyresolver.KeyResolverException; |
|
import com.sun.org.apache.xml.internal.security.keys.keyresolver.KeyResolverSpi; |
|
import com.sun.org.apache.xml.internal.security.keys.storage.StorageResolver; |
|
import com.sun.org.apache.xml.internal.security.signature.XMLSignatureInput; |
|
import com.sun.org.apache.xml.internal.security.utils.Constants; |
|
import com.sun.org.apache.xml.internal.security.utils.XMLUtils; |
|
import com.sun.org.apache.xml.internal.security.utils.resolver.ResourceResolver; |
|
import org.w3c.dom.Attr; |
|
import org.w3c.dom.Element; |
|
import org.xml.sax.SAXException; |
|
|
|
|
|
|
|
|
|
|
|
*/ |
|
public class KeyInfoReferenceResolver extends KeyResolverSpi { |
|
|
|
private static final com.sun.org.slf4j.internal.Logger LOG = |
|
com.sun.org.slf4j.internal.LoggerFactory.getLogger(KeyInfoReferenceResolver.class); |
|
|
|
|
|
public boolean engineCanResolve(Element element, String baseURI, StorageResolver storage) { |
|
return XMLUtils.elementIsInSignature11Space(element, Constants._TAG_KEYINFOREFERENCE); |
|
} |
|
|
|
|
|
public PublicKey engineLookupAndResolvePublicKey(Element element, String baseURI, StorageResolver storage) |
|
throws KeyResolverException { |
|
|
|
LOG.debug("Can I resolve {}", element.getTagName()); |
|
|
|
if (!engineCanResolve(element, baseURI, storage)) { |
|
return null; |
|
} |
|
|
|
try { |
|
KeyInfo referent = resolveReferentKeyInfo(element, baseURI, storage); |
|
if (referent != null) { |
|
return referent.getPublicKey(); |
|
} |
|
} catch (XMLSecurityException e) { |
|
LOG.debug("XMLSecurityException", e); |
|
} |
|
|
|
return null; |
|
} |
|
|
|
|
|
public X509Certificate engineLookupResolveX509Certificate(Element element, String baseURI, StorageResolver storage) |
|
throws KeyResolverException { |
|
|
|
LOG.debug("Can I resolve {}", element.getTagName()); |
|
|
|
if (!engineCanResolve(element, baseURI, storage)) { |
|
return null; |
|
} |
|
|
|
try { |
|
KeyInfo referent = resolveReferentKeyInfo(element, baseURI, storage); |
|
if (referent != null) { |
|
return referent.getX509Certificate(); |
|
} |
|
} catch (XMLSecurityException e) { |
|
LOG.debug("XMLSecurityException", e); |
|
} |
|
|
|
return null; |
|
} |
|
|
|
|
|
public SecretKey engineLookupAndResolveSecretKey(Element element, String baseURI, StorageResolver storage) |
|
throws KeyResolverException { |
|
|
|
LOG.debug("Can I resolve {}", element.getTagName()); |
|
|
|
if (!engineCanResolve(element, baseURI, storage)) { |
|
return null; |
|
} |
|
|
|
try { |
|
KeyInfo referent = resolveReferentKeyInfo(element, baseURI, storage); |
|
if (referent != null) { |
|
return referent.getSecretKey(); |
|
} |
|
} catch (XMLSecurityException e) { |
|
LOG.debug("XMLSecurityException", e); |
|
} |
|
|
|
return null; |
|
} |
|
|
|
|
|
public PrivateKey engineLookupAndResolvePrivateKey(Element element, String baseURI, StorageResolver storage) |
|
throws KeyResolverException { |
|
|
|
LOG.debug("Can I resolve " + element.getTagName()); |
|
|
|
if (!engineCanResolve(element, baseURI, storage)) { |
|
return null; |
|
} |
|
|
|
try { |
|
KeyInfo referent = resolveReferentKeyInfo(element, baseURI, storage); |
|
if (referent != null) { |
|
return referent.getPrivateKey(); |
|
} |
|
} catch (XMLSecurityException e) { |
|
LOG.debug("XMLSecurityException", e); |
|
} |
|
|
|
return null; |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
*/ |
|
private KeyInfo resolveReferentKeyInfo(Element element, String baseURI, StorageResolver storage) throws XMLSecurityException { |
|
KeyInfoReference reference = new KeyInfoReference(element, baseURI); |
|
Attr uriAttr = reference.getURIAttr(); |
|
|
|
XMLSignatureInput resource = resolveInput(uriAttr, baseURI, secureValidation); |
|
|
|
Element referentElement = null; |
|
try { |
|
referentElement = obtainReferenceElement(resource); |
|
} catch (Exception e) { |
|
LOG.debug("XMLSecurityException", e); |
|
return null; |
|
} |
|
|
|
if (referentElement == null) { |
|
LOG.debug("De-reference of KeyInfoReference URI returned null: {}", uriAttr.getValue()); |
|
return null; |
|
} |
|
|
|
validateReference(referentElement); |
|
|
|
KeyInfo referent = new KeyInfo(referentElement, baseURI); |
|
referent.addStorageResolver(storage); |
|
return referent; |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
*/ |
|
private void validateReference(Element referentElement) throws XMLSecurityException { |
|
if (!XMLUtils.elementIsInSignatureSpace(referentElement, Constants._TAG_KEYINFO)) { |
|
Object exArgs[] = { new QName(referentElement.getNamespaceURI(), referentElement.getLocalName()) }; |
|
throw new XMLSecurityException("KeyInfoReferenceResolver.InvalidReferentElement.WrongType", exArgs); |
|
} |
|
|
|
KeyInfo referent = new KeyInfo(referentElement, ""); |
|
if (referent.containsKeyInfoReference()) { |
|
if (secureValidation) { |
|
throw new XMLSecurityException("KeyInfoReferenceResolver.InvalidReferentElement.ReferenceWithSecure"); |
|
} else { |
|
// Don't support chains of references at this time. If do support in the future, this is where the code |
|
// would go to validate that don't have a cycle, resulting in an infinite loop. This may be unrealistic |
|
|
|
throw new XMLSecurityException("KeyInfoReferenceResolver.InvalidReferentElement.ReferenceWithoutSecure"); |
|
} |
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
*/ |
|
private XMLSignatureInput resolveInput(Attr uri, String baseURI, boolean secureValidation) |
|
throws XMLSecurityException { |
|
ResourceResolver resRes = ResourceResolver.getInstance(uri, baseURI, secureValidation); |
|
return resRes.resolve(uri, baseURI, secureValidation); |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
*/ |
|
private Element obtainReferenceElement(XMLSignatureInput resource) |
|
throws CanonicalizationException, ParserConfigurationException, |
|
IOException, SAXException, KeyResolverException { |
|
|
|
Element e; |
|
if (resource.isElement()){ |
|
e = (Element) resource.getSubNode(); |
|
} else if (resource.isNodeSet()) { |
|
LOG.debug("De-reference of KeyInfoReference returned an unsupported NodeSet"); |
|
return null; |
|
} else { |
|
|
|
byte inputBytes[] = resource.getBytes(); |
|
e = getDocFromBytes(inputBytes, this.secureValidation); |
|
} |
|
return e; |
|
} |
|
} |