Add validation for support RIM hash and base RIM signature.

This commit is contained in:
chubtub 2020-11-04 15:21:48 -05:00
parent 1b3abe465a
commit 24cf71642d
5 changed files with 355 additions and 4 deletions

View File

@ -4,6 +4,9 @@ import hirs.data.persist.BaseReferenceManifest;
import hirs.data.persist.ReferenceManifest; import hirs.data.persist.ReferenceManifest;
import hirs.data.persist.SupportReferenceManifest; import hirs.data.persist.SupportReferenceManifest;
import hirs.data.persist.SwidResource; import hirs.data.persist.SwidResource;
import hirs.data.persist.certificate.Certificate;
import hirs.data.persist.certificate.CertificateAuthorityCredential;
import hirs.persist.CertificateManager;
import hirs.persist.DBManagerException; import hirs.persist.DBManagerException;
import hirs.persist.ReferenceManifestManager; import hirs.persist.ReferenceManifestManager;
import hirs.tpm.eventlog.TCGEventLog; import hirs.tpm.eventlog.TCGEventLog;
@ -12,6 +15,7 @@ import hirs.attestationca.portal.page.PageController;
import hirs.attestationca.portal.page.PageMessages; import hirs.attestationca.portal.page.PageMessages;
import hirs.attestationca.portal.page.params.ReferenceManifestDetailsPageParams; import hirs.attestationca.portal.page.params.ReferenceManifestDetailsPageParams;
import java.io.ByteArrayInputStream;
import java.io.IOException; import java.io.IOException;
import java.security.NoSuchAlgorithmException; import java.security.NoSuchAlgorithmException;
import java.security.cert.CertificateException; import java.security.cert.CertificateException;
@ -21,6 +25,7 @@ import java.util.List;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.UUID; import java.util.UUID;
import hirs.utils.ReferenceManifestValidator;
import org.apache.logging.log4j.LogManager; import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger; import org.apache.logging.log4j.Logger;
import org.springframework.beans.factory.annotation.Autowired; import org.springframework.beans.factory.annotation.Autowired;
@ -38,6 +43,7 @@ public class ReferenceManifestDetailsPageController
extends PageController<ReferenceManifestDetailsPageParams> { extends PageController<ReferenceManifestDetailsPageParams> {
private final ReferenceManifestManager referenceManifestManager; private final ReferenceManifestManager referenceManifestManager;
private final CertificateManager certificateManager;
private static final Logger LOGGER private static final Logger LOGGER
= LogManager.getLogger(ReferenceManifestDetailsPageController.class); = LogManager.getLogger(ReferenceManifestDetailsPageController.class);
@ -48,9 +54,11 @@ public class ReferenceManifestDetailsPageController
*/ */
@Autowired @Autowired
public ReferenceManifestDetailsPageController( public ReferenceManifestDetailsPageController(
final ReferenceManifestManager referenceManifestManager) { final ReferenceManifestManager referenceManifestManager,
final CertificateManager certificateManager) {
super(Page.RIM_DETAILS); super(Page.RIM_DETAILS);
this.referenceManifestManager = referenceManifestManager; this.referenceManifestManager = referenceManifestManager;
this.certificateManager = certificateManager;
} }
/** /**
@ -80,7 +88,7 @@ public class ReferenceManifestDetailsPageController
} else { } else {
try { try {
UUID uuid = UUID.fromString(params.getId()); UUID uuid = UUID.fromString(params.getId());
data.putAll(getRimDetailInfo(uuid, referenceManifestManager)); data.putAll(getRimDetailInfo(uuid, referenceManifestManager, certificateManager));
} catch (IllegalArgumentException iaEx) { } catch (IllegalArgumentException iaEx) {
String uuidError = "Failed to parse ID from: " + params.getId(); String uuidError = "Failed to parse ID from: " + params.getId();
messages.addError(uuidError); messages.addError(uuidError);
@ -114,7 +122,8 @@ public class ReferenceManifestDetailsPageController
* @throws CertificateException if a certificate doesn't parse. * @throws CertificateException if a certificate doesn't parse.
*/ */
public static HashMap<String, Object> getRimDetailInfo(final UUID uuid, public static HashMap<String, Object> getRimDetailInfo(final UUID uuid,
final ReferenceManifestManager referenceManifestManager) throws IOException, final ReferenceManifestManager referenceManifestManager,
final CertificateManager certificateManager) throws IOException,
CertificateException, NoSuchAlgorithmException { CertificateException, NoSuchAlgorithmException {
HashMap<String, Object> data = new HashMap<>(); HashMap<String, Object> data = new HashMap<>();
@ -183,11 +192,22 @@ public class ReferenceManifestDetailsPageController
logProcessor = new TCGEventLog(support.getRimBytes()); logProcessor = new TCGEventLog(support.getRimBytes());
} }
} }
ReferenceManifestValidator rimValidator = new ReferenceManifestValidator(
new ByteArrayInputStream(bRim.getRimBytes()));
// going to have to pull the filename and grab that from the DB // going to have to pull the filename and grab that from the DB
// to get the id to make the link // to get the id to make the link
for (SwidResource swidRes : resources) { for (SwidResource swidRes : resources) {
if (support != null && swidRes.getName() if (support != null && swidRes.getName()
.equals(support.getFileName())) { .equals(support.getFileName())) {
rimValidator.validateSupportRimHash(support.getRimBytes(),
rimValidator.getDocument());
if (rimValidator.isSupportRimValid()) {
data.put("supportRimHashValid", true);
} else {
data.put("supportRimHashValid", false);
}
swidRes.setPcrValues(Arrays.asList( swidRes.setPcrValues(Arrays.asList(
logProcessor.getExpectedPCRValues())); logProcessor.getExpectedPCRValues()));
break; break;
@ -198,6 +218,20 @@ public class ReferenceManifestDetailsPageController
data.put("associatedRim", bRim.getAssociatedRim()); data.put("associatedRim", bRim.getAssociatedRim());
data.put("swidFiles", resources); data.put("swidFiles", resources);
rimValidator.validateXmlSignature(rimValidator.getDocument());
data.put("signatureValid", rimValidator.isSignatureValid());
if (rimValidator.isSignatureValid()) {
try {
Certificate certificate =
CertificateAuthorityCredential.select(certificateManager)
.byEncodedPublicKey(rimValidator.getPublicKey().getEncoded())
.getCertificate();
data.put("issuerID", certificate.getId().toString());
} catch (NullPointerException e) {
LOGGER.info("Unable to get signing certificate link: " + e.getMessage());
}
}
} else { } else {
SupportReferenceManifest sRim = SupportReferenceManifest SupportReferenceManifest sRim = SupportReferenceManifest
.select(referenceManifestManager) .select(referenceManifestManager)

View File

@ -17,7 +17,13 @@
</a> </a>
</jsp:attribute> </jsp:attribute>
<jsp:body> <jsp:body>
<div id="certificate-details-page" class="container-fluid"> <c:set var="passIcon" value="${icons}/ic_checkbox_marked_circle_black_green_24dp.png"/>
<c:set var="failIcon" value="${icons}/ic_error_red_24dp.png"/>
<c:set var="signatureValidText" value="Signature valid!"/>
<c:set var="signatureInvalidText" value="Signature not valid!"/>
<c:set var="supportRimHashValidText" value="Support RIM hash valid!"/>
<c:set var="supportRimHashInvalidText" value="Support RIM hash not valid!"/>
<div id="certificate-details-page" class="container-fluid">
<c:choose> <c:choose>
<c:when test="${initialData.rimType=='Support'}"> <c:when test="${initialData.rimType=='Support'}">
<div class="row"> <div class="row">
@ -171,6 +177,14 @@
<c:choose> <c:choose>
<c:when test="${not empty initialData.associatedRim}"> <c:when test="${not empty initialData.associatedRim}">
<a href="${portal}/rim-details?id=${initialData.associatedRim}">${resource.getName()}</a> <a href="${portal}/rim-details?id=${initialData.associatedRim}">${resource.getName()}</a>
<c:choose>
<c:when test="${not empty initialData.supportRimHashValid}">
<img src="${passIcon}" title="${supportRimHashValidText}"/>
</c:when>
<c:otherwise>
<img src="${failIcon}" title="${supportRimHashInvalidText}"/>
</c:otherwise>
</c:choose>
</c:when> </c:when>
<c:otherwise> <c:otherwise>
${resource.getName()} ${resource.getName()}
@ -237,6 +251,24 @@
</div> </div>
</div> </div>
</div> </div>
<div class="row">
<div class="col-md-1 col-md-offset-1"><span class="colHeader">Signature</span></div>
<div id="signature" class="col col-md-8">
<div>Validity:&nbsp;<span>
<c:choose>
<c:when test="${initialData.signatureValid}">
<img src="${passIcon}" title="${signatureValidText}"/>
<c:if test="${not empty initialData.issuerID}">
<div><a href="${portal}/certificate-details?id=${initialData.issuerID}&type=certificateauthority">Signing certificate</a></div>
</c:if>
</c:when>
<c:otherwise>
<img src="${failIcon}" title="${signatureInvalidText}"/>
</c:otherwise>
</c:choose>
</span></div>
</div>
</div>
</c:otherwise> </c:otherwise>
</c:choose> </c:choose>
</div> </div>

View File

@ -151,6 +151,7 @@ public class BaseReferenceManifest extends ReferenceManifest {
public BaseReferenceManifest(final byte[] rimBytes) throws IOException { public BaseReferenceManifest(final byte[] rimBytes) throws IOException {
super(rimBytes); super(rimBytes);
this.setRimType(BASE_RIM); this.setRimType(BASE_RIM);
this.setFileName("");
SoftwareIdentity si = validateSwidTag(new ByteArrayInputStream(rimBytes)); SoftwareIdentity si = validateSwidTag(new ByteArrayInputStream(rimBytes));
// begin parsing valid swid tag // begin parsing valid swid tag

View File

@ -0,0 +1,274 @@
package hirs.utils;
import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;
import org.w3c.dom.Document;
import org.w3c.dom.Element;
import org.w3c.dom.NodeList;
import org.xml.sax.SAXException;
import javax.xml.XMLConstants;
import javax.xml.bind.JAXBContext;
import javax.xml.bind.JAXBException;
import javax.xml.bind.UnmarshalException;
import javax.xml.bind.Unmarshaller;
import javax.xml.crypto.*;
import javax.xml.crypto.dsig.XMLSignature;
import javax.xml.crypto.dsig.XMLSignatureException;
import javax.xml.crypto.dsig.XMLSignatureFactory;
import javax.xml.crypto.dsig.dom.DOMValidateContext;
import javax.xml.crypto.dsig.keyinfo.KeyInfo;
import javax.xml.crypto.dsig.keyinfo.X509Data;
import javax.xml.transform.*;
import javax.xml.transform.dom.DOMResult;
import javax.xml.transform.stream.StreamSource;
import javax.xml.validation.Schema;
import javax.xml.validation.SchemaFactory;
import java.io.IOException;
import java.io.InputStream;
import java.io.UnsupportedEncodingException;
import java.nio.file.Files;
import java.nio.file.Paths;
import java.security.Key;
import java.security.MessageDigest;
import java.security.NoSuchAlgorithmException;
import java.security.PublicKey;
import java.security.cert.X509Certificate;
import java.util.Iterator;
public class ReferenceManifestValidator {
private static final String SIGNATURE_ALGORITHM_RSA_SHA256 =
"http://www.w3.org/2001/04/xmldsig-more#rsa-sha256";
private static final String SCHEMA_PACKAGE = "hirs.utils.xjc";
private static final String SCHEMA_URL = "swid_schema.xsd";
private static final String SCHEMA_LANGUAGE = XMLConstants.W3C_XML_SCHEMA_NS_URI;
private static final String IDENTITY_TRANSFORM = "identity_transform.xslt";
private static final String SHA256 = "SHA-256";
private static final Logger LOGGER = LogManager.getLogger(ReferenceManifestValidator.class);
private Unmarshaller unmarshaller;
private PublicKey publicKey;
private Document document;
private boolean signatureValid, supportRimValid;
public boolean isSignatureValid() {
return signatureValid;
}
public boolean isSupportRimValid() {
return supportRimValid;
}
public PublicKey getPublicKey() { return publicKey; }
public Document getDocument() { return document; }
public ReferenceManifestValidator() {}
public ReferenceManifestValidator(InputStream input) {
try {
JAXBContext jaxbContext = JAXBContext.newInstance(SCHEMA_PACKAGE);
unmarshaller = jaxbContext.createUnmarshaller();
signatureValid = false;
supportRimValid = false;
publicKey = null;
document = unmarshallSwidTag(removeXMLWhitespace(new StreamSource(input)));
} catch (JAXBException e) {
LOGGER.warn("Error initializing JAXBContext: " + e.getMessage());
} catch (IOException e) {
LOGGER.warn("Error during unmarshalling: " + e.getMessage());
}
}
public void validateSupportRimHash(byte[] input, Document doc) {
String calculatedHash = getHashValue(input, SHA256);
Element file = (Element) doc.getElementsByTagName("File").item(0);
LOGGER.info("Calculated hash: " + calculatedHash
+ ", actual: " + file.getAttribute("SHA256:hash"));
if (file.getAttribute("SHA256:hash").equals(calculatedHash)) {
supportRimValid = true;
} else {
supportRimValid = false;
}
}
/**
* This method validates the signature block in the Document and stores the
* result for public access.
*
* @param doc the xml data in Document form.
*/
public void validateXmlSignature(Document doc) throws IOException {
signatureValid = validateSignedXMLDocument(doc);
}
private String getHashValue(byte[] input, String sha) {
String resultString = null;
try {
MessageDigest md = MessageDigest.getInstance(sha);
byte[] bytes = md.digest(input);
StringBuilder sb = new StringBuilder();
for (int i = 0; i < bytes.length; i++) {
sb.append(Integer.toString((bytes[i] & 0xff) + 0x100, 16).substring(1));
}
resultString = sb.toString();
} catch (NoSuchAlgorithmException grex) {
LOGGER.warn(grex.getMessage());
}
return resultString;
}
/**
* This method validates a Document with a signature element.
*
* @param doc
*/
private boolean validateSignedXMLDocument(Document doc) {
DOMValidateContext context;
boolean isValid = false;
try {
NodeList nodes = doc.getElementsByTagNameNS(XMLSignature.XMLNS, "Signature");
if (nodes.getLength() == 0) {
throw new Exception("Signature element not found!");
}
NodeList embeddedCert = doc.getElementsByTagName("X509Data");
if (embeddedCert.getLength() > 0) {
X509KeySelector keySelector = new ReferenceManifestValidator.X509KeySelector();
context = new DOMValidateContext(keySelector, nodes.item(0));
XMLSignatureFactory sigFactory = XMLSignatureFactory.getInstance("DOM");
XMLSignature signature = sigFactory.unmarshalXMLSignature(context);
isValid = signature.validate(context);
publicKey = keySelector.getPublicKey();
} else {
LOGGER.info("Signing certificate not found for validation!");
}
} catch (MarshalException | XMLSignatureException e) {
LOGGER.warn(e.getMessage());
} catch (Exception e) {
LOGGER.warn(e.getMessage());
LOGGER.info(e.getMessage());
}
return isValid;
}
public class X509KeySelector extends KeySelector {
private PublicKey publicKey;
public KeySelectorResult select(KeyInfo keyinfo,
KeySelector.Purpose purpose,
AlgorithmMethod algorithm,
XMLCryptoContext context) throws KeySelectorException {
Iterator keyinfoItr = keyinfo.getContent().iterator();
while(keyinfoItr.hasNext()) {
XMLStructure element = (XMLStructure) keyinfoItr.next();
if (element instanceof X509Data) {
X509Data data = (X509Data) element;
Iterator dataItr = data.getContent().iterator();
while (dataItr.hasNext()) {
Object object = dataItr.next();
if (object instanceof X509Certificate) {
publicKey = ((X509Certificate) object).getPublicKey();
if (areAlgorithmsEqual(algorithm.getAlgorithm(), publicKey.getAlgorithm())) {
return new ReferenceManifestValidator.X509KeySelector.RIMKeySelectorResult(publicKey);
}
}
}
}
}
throw new KeySelectorException("No key found!");
}
public boolean areAlgorithmsEqual(String uri, String name) {
if (uri.equals(SIGNATURE_ALGORITHM_RSA_SHA256) && name.equalsIgnoreCase("RSA")) {
return true;
} else {
return false;
}
}
public PublicKey getPublicKey() {
return publicKey;
}
private class RIMKeySelectorResult implements KeySelectorResult {
private Key key;
public RIMKeySelectorResult(Key key) {
this.key = key;
}
public Key getKey() {
return key;
}
}
}
/**
* This method unmarshalls the swidtag found at [path] into a Document object
* and validates it according to the schema.
*
* @param doc of the input swidtag.
* @return document validated against the schema.
*/
private Document unmarshallSwidTag(Document doc) {
InputStream is = null;
try {
is = ReferenceManifestValidator.class
.getClassLoader().getResourceAsStream(SCHEMA_URL);
SchemaFactory schemaFactory = SchemaFactory.newInstance(SCHEMA_LANGUAGE);
Schema schema = schemaFactory.newSchema(new StreamSource(is));
unmarshaller.setSchema(schema);
unmarshaller.unmarshal(doc);
} catch (SAXException e) {
LOGGER.warn("Error setting schema for validation!");
} catch (UnmarshalException e) {
LOGGER.warn("Error validating swidtag file!");
} catch (IllegalArgumentException e) {
LOGGER.warn("Input file empty.");
} catch (JAXBException e) {
e.printStackTrace();
} finally {
if (is != null) {
try {
is.close();
} catch (IOException e) {
LOGGER.warn("Error closing input stream");
}
}
}
return doc;
}
/**
* This method strips all whitespace from an xml file, including indents and spaces
* added for human-readability.
* @param source of the input xml.
* @return Document representation of the xml.
*/
private Document removeXMLWhitespace(StreamSource source) throws IOException {
TransformerFactory tf = TransformerFactory.newInstance();
Source identitySource = new StreamSource(
ReferenceManifestValidator.class.getClassLoader()
.getResourceAsStream(IDENTITY_TRANSFORM));
Document doc = null;
try {
Transformer transformer = tf.newTransformer(identitySource);
DOMResult result = new DOMResult();
transformer.transform(source, result);
doc = (Document) result.getNode();
} catch (TransformerConfigurationException e) {
LOGGER.warn("Error configuring transformer!");
e.printStackTrace();
} catch (TransformerException e) {
LOGGER.warn("Error transforming input!");
e.printStackTrace();
}
return doc;
}
}

View File

@ -0,0 +1,10 @@
<?xml version="1.0"?>
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output indent="no" />
<xsl:strip-space elements="*"/>
<xsl:template match="@*|node()">
<xsl:copy>
<xsl:apply-templates select="@*|node()"/>
</xsl:copy>
</xsl:template>
</xsl:stylesheet>