Skip to content

Commit

Permalink
Merged PR 4084: [3.1.5]
Browse files Browse the repository at this point in the history
  • Loading branch information
David-Engel committed Oct 18, 2023
1 parent 8596bf9 commit 254d76d
Showing 1 changed file with 27 additions and 9 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -161,8 +161,8 @@ private void VerifyAttestationInfo(string attestationUrl, HealthReport healthRep
X509Certificate2Collection signingCerts = GetSigningCertificate(attestationUrl, shouldForceUpdateSigningKeys);

// Verify SQL Health report root chain of trust is the HGS root signing cert
X509ChainStatusFlags chainStatus = VerifyHealthReportAgainstRootCertificate(signingCerts, healthReport.Certificate);
if (chainStatus != X509ChainStatusFlags.NoError)
if (!VerifyHealthReportAgainstRootCertificate(signingCerts, healthReport.Certificate, out X509ChainStatusFlags chainStatus) ||
chainStatus != X509ChainStatusFlags.NoError)
{
// In cases if we fail to validate the health report, it might be possible that we are using old signing keys
// let's re-download the signing keys again and re-validate the health report
Expand Down Expand Up @@ -223,11 +223,20 @@ private bool AnyCertificatesExpired(X509Certificate2Collection certificates)
return certificates.OfType<X509Certificate2>().Any(c => c.NotAfter < DateTime.Now);
}

// Verifies that a chain of trust can be built from the health report provided
// by SQL Server and the attestation service's root signing certificate(s).
private X509ChainStatusFlags VerifyHealthReportAgainstRootCertificate(X509Certificate2Collection signingCerts, X509Certificate2 healthReportCert)
/// <summary>
/// Verifies that a chain of trust can be built from the health report provided
/// by SQL Server and the attestation service's root signing certificate(s).
///
/// If the method returns false, the value of chainStatus doesn't matter. The chain could not be validated.
/// </summary>
/// <param name="signingCerts"></param>
/// <param name="healthReportCert"></param>
/// <param name="chainStatus"></param>
/// <returns>A <see cref="T:System.Boolean" /> that indicates if the certificate was able to be verified.</returns>
private bool VerifyHealthReportAgainstRootCertificate(X509Certificate2Collection signingCerts, X509Certificate2 healthReportCert, out X509ChainStatusFlags chainStatus)
{
var chain = new X509Chain();
chainStatus = X509ChainStatusFlags.NoError;

foreach (var cert in signingCerts)
{
Expand All @@ -249,9 +258,14 @@ private X509ChainStatusFlags VerifyHealthReportAgainstRootCertificate(X509Certif
}
else
{
return status.Status;
chainStatus = status.Status;
return true;
}
}
// The only ways past or out of the loop are:
// 1. untrustedRoot is true, in which case we want to continue to below
// 2. chainStatus is set to the first status in the chain and we return true
// 3. the ChainStatus is empty

// if the chain failed with untrusted root, this could be because the client doesn't have the root cert
// installed. If the chain's untrusted root cert has the same thumbprint as the signing cert, then we
Expand All @@ -268,17 +282,21 @@ private X509ChainStatusFlags VerifyHealthReportAgainstRootCertificate(X509Certif
{
if (element.Certificate.Thumbprint == cert.Thumbprint)
{
return X509ChainStatusFlags.NoError;
return true;
}
}
}

// in the case where we didn't find matching thumbprint
return X509ChainStatusFlags.UntrustedRoot;
chainStatus = X509ChainStatusFlags.UntrustedRoot;
return true;
}

// There was an unknown failure and X509Chain.Build() returned an empty ChainStatus.
return false;
}

return X509ChainStatusFlags.NoError;
return true;
}

// Verifies the enclave report signature using the health report.
Expand Down

0 comments on commit 254d76d

Please sign in to comment.