Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

AE with enclaves - multi-platform and .NET Standard 2.1 support #676

Merged
merged 48 commits into from
Aug 21, 2020
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
48 commits
Select commit Hold shift + click to select a range
13857f2
Merge pull request #2 from dotnet/master
May 29, 2020
b807d0b
Merge remote-tracking branch 'upstream/master'
johnnypham Jun 4, 2020
26b0f35
Set up CI with Azure Pipelines
Jun 8, 2020
3da84f8
Delete azure-pipelines.yml
Jun 8, 2020
9b49213
Merge remote-tracking branch 'upstream/master'
johnnypham Jun 14, 2020
d02b69c
Set up CI with Azure Pipelines
Jun 8, 2020
896caa1
Delete azure-pipelines.yml
Jun 8, 2020
0c04741
Merge branch 'master' of https://github.com/johnnypham/SqlClient
johnnypham Jun 18, 2020
035ef3d
Merge remote-tracking branch 'upstream/master'
johnnypham Jun 21, 2020
d1dc8bb
Merge remote-tracking branch 'upstream/master'
johnnypham Jul 3, 2020
acbb0a5
Merge remote-tracking branch 'upstream/master'
johnnypham Jul 10, 2020
0d8b82f
aev2
johnnypham Jul 21, 2020
a98f2a0
azure attestation, vbs
johnnypham Jul 22, 2020
0d5c11e
Update EnclaveProviderBase.NetCoreApp.cs
johnnypham Jul 24, 2020
6eeaf13
Update EnclaveProviderBase.NetCoreApp.cs
johnnypham Jul 24, 2020
7bec0ea
replace tobytearray
johnnypham Jul 24, 2020
1831ef0
move new methods to utils
johnnypham Jul 24, 2020
b79a97e
minor changes
johnnypham Jul 27, 2020
0c898c3
ae apishould tests for linux
johnnypham Jul 29, 2020
361505a
test akv fixture
johnnypham Jul 29, 2020
c9785ea
Update SQLSetupStrategy.cs
johnnypham Jul 29, 2020
50c5b12
bulkcopyae akv test
johnnypham Jul 29, 2020
d1da600
apishould cross platform
johnnypham Jul 30, 2020
2a93a2e
test fixture changes
johnnypham Jul 30, 2020
743d6ce
Update BulkCopyAE.cs
johnnypham Jul 30, 2020
81654cc
Update BulkCopyAE.cs
johnnypham Jul 30, 2020
9ab0eac
test platform specific fixtures
johnnypham Jul 31, 2020
170ddc9
test platform specific fixtures
johnnypham Jul 31, 2020
ba1b31a
update ae tests for linux
johnnypham Jul 31, 2020
1cad3ff
minor changes
johnnypham Jul 31, 2020
daa2b13
netstandard
johnnypham Aug 4, 2020
c12121f
netstandard changes
johnnypham Aug 5, 2020
f264806
change targetframework property to targetframeworks
johnnypham Aug 5, 2020
0448362
Update SqlColumnEncryptionCertificateStoreProvider.Windows.cs
johnnypham Aug 7, 2020
9ba0fc9
Address feedback
johnnypham Aug 7, 2020
bcc7e6c
Merge branch 'aev2' of https://github.com/johnnypham/SqlClient into aev2
johnnypham Aug 7, 2020
4780d48
Merge remote-tracking branch 'upstream/master' into aev2
johnnypham Aug 7, 2020
104e0d6
fix culture exceptions in functional tests for ubuntu 18.04
johnnypham Aug 11, 2020
5575f1c
revert test changes
johnnypham Aug 11, 2020
7a1e9fb
test against .net standard 2.1
johnnypham Aug 13, 2020
5285c26
add dispose to new fixture class
johnnypham Aug 14, 2020
ec12750
Update AlwaysEncryptedEnclaveProviderUtils.cs
johnnypham Aug 18, 2020
dca6da8
Update AlwaysEncryptedEnclaveProviderUtils.cs
johnnypham Aug 18, 2020
e575519
Update AlwaysEncryptedEnclaveProviderUtils.cs
johnnypham Aug 19, 2020
50e2ee3
Merge remote-tracking branch 'upstream/master' into aev2
johnnypham Aug 20, 2020
58235f7
casing
johnnypham Aug 20, 2020
5d0fde8
Address comments
johnnypham Aug 21, 2020
a552100
Address comments
johnnypham Aug 21, 2020
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Prev Previous commit
Next Next commit
azure attestation, vbs
  • Loading branch information
johnnypham committed Jul 22, 2020
commit a98f2a04ffaef90bf7609e3f057e4db6646f89b1
Original file line number Diff line number Diff line change
Expand Up @@ -73,15 +73,14 @@ internal override void GetEnclaveSession(string servername, string attestationUr
// Gets the information that SqlClient subsequently uses to initiate the process of attesting the enclave and to establish a secure session with the enclave.
internal override SqlEnclaveAttestationParameters GetAttestationParameters(string attestationUrl, byte[] customData, int customDataLength)
{
ECDiffieHellmanCng clientDHKey = new ECDiffieHellmanCng(DiffieHellmanKeySize);
clientDHKey.KeyDerivationFunction = ECDiffieHellmanKeyDerivationFunction.Hash;
clientDHKey.HashAlgorithm = CngAlgorithm.Sha256;
ECDiffieHellman clientDHKey = ECDiffieHellman.Create();
clientDHKey.KeySize = DiffieHellmanKeySize;
byte[] attestationParam = PrepareAttestationParameters(attestationUrl, customData, customDataLength);
return new SqlEnclaveAttestationParameters(AzureBasedAttestationProtocolId, attestationParam, clientDHKey);
}

// When overridden in a derived class, performs enclave attestation, generates a symmetric key for the session, creates a an enclave session and stores the session information in the cache.
internal override void CreateEnclaveSession(byte[] attestationInfo, ECDiffieHellmanCng clientDHKey, string attestationUrl, string servername, byte[] customData, int customDataLength, out SqlEnclaveSession sqlEnclaveSession, out long counter)
internal override void CreateEnclaveSession(byte[] attestationInfo, ECDiffieHellman clientDHKey, string attestationUrl, string servername, byte[] customData, int customDataLength, out SqlEnclaveSession sqlEnclaveSession, out long counter)
{
sqlEnclaveSession = null;
counter = 0;
Expand Down Expand Up @@ -514,7 +513,7 @@ private void ValidateClaim(Dictionary<string, string> claims, string claimName,
}

// Derives the shared secret between the client and enclave.
private byte[] GetSharedSecret(EnclavePublicKey enclavePublicKey, byte[] nonce, EnclaveType enclaveType, EnclaveDiffieHellmanInfo enclaveDHInfo, ECDiffieHellmanCng clientDHKey)
private byte[] GetSharedSecret(EnclavePublicKey enclavePublicKey, byte[] nonce, EnclaveType enclaveType, EnclaveDiffieHellmanInfo enclaveDHInfo, ECDiffieHellman clientDHKey)
{
byte[] enclaveRsaPublicKey = enclavePublicKey.PublicKey;

Expand All @@ -529,17 +528,18 @@ private byte[] GetSharedSecret(EnclavePublicKey enclavePublicKey, byte[] nonce,
}

// Perform signature verification. The enclave's DiffieHellman public key was signed by the enclave's RSA public key.
CngKey cngkey = CngKey.Import(enclaveRsaPublicKey, CngKeyBlobFormat.GenericPublicBlob);
using (RSACng rsacng = new RSACng(cngkey))
RSAParameters rsaParams = RSAKeyBlobToParams(enclaveRsaPublicKey);
using (RSA rsa = RSA.Create(rsaParams))
{
if (!rsacng.VerifyData(enclaveDHInfo.PublicKey, enclaveDHInfo.PublicKeySignature, HashAlgorithmName.SHA256, RSASignaturePadding.Pkcs1))
if (!rsa.VerifyData(enclaveDHInfo.PublicKey, enclaveDHInfo.PublicKeySignature, HashAlgorithmName.SHA256, RSASignaturePadding.Pkcs1))
{
throw new ArgumentException(SR.GetSharedSecretFailed);
}
}

CngKey key = CngKey.Import(enclaveDHInfo.PublicKey, CngKeyBlobFormat.GenericPublicBlob);
return clientDHKey.DeriveKeyMaterial(key);
ECParameters ecParams = ECCKeyBlobToParams(enclaveDHInfo.PublicKey);
ECDiffieHellman enclaveDHKey = ECDiffieHellman.Create(ecParams);
return clientDHKey.DeriveKeyFromHash(enclaveDHKey.PublicKey, HashAlgorithmName.SHA256);
}
#endregion
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@
// See the LICENSE file in the project root for more information.

using System;
using System.Linq;
using System.Runtime.Caching;
using System.Security.Cryptography;
using System.Threading;
Expand Down Expand Up @@ -210,6 +211,53 @@ protected SqlEnclaveSession AddEnclaveSessionToCache(string attestationUrl, stri
{
return SessionCache.CreateSession(attestationUrl, servername, sharedSecret, sessionId, out counter);
}

// Extracts the public key's modulus and exponent from the key blob
protected RSAParameters RSAKeyBlobToParams(byte[] keyBlob)
{
// The RSA public key blob is structured as follows:
// BCRYPT_RSAKEY_BLOB header
// byte[cbPublicExp] publicExponent - Exponent
// byte[cbModulus] modulus - Modulus

// The exponent is the final 3 bytes in the header
// The modulus is the final 512 bytes in the key blob
const int modulusSize = 512;
const int exponentSize = 3;
int BcryptRsaKeyBlobHeaderSize = keyBlob.Length - modulusSize;
int exponentOffset = BcryptRsaKeyBlobHeaderSize - exponentSize;
int modulusOffset = exponentOffset + exponentSize;

return new RSAParameters()
{
Exponent = keyBlob.Skip(exponentOffset).Take(exponentSize).ToArray(),
Modulus = keyBlob.Skip(modulusOffset).Take(modulusSize).ToArray()
};
}

// Extracts the public key's X and Y coordinate from the key blob
protected ECParameters ECCKeyBlobToParams(byte[] keyBlob)
{
// The ECC public key blob is structured as follows:
// BCRYPT_ECCKEY_BLOB header
// byte[cbKey] X - X coordinate
// byte[cbKey] Y - Y coordinate

// The size of each key is found after the first 4 byes (magic number)
const int keySizeOffset = 4;
int keySize = BitConverter.ToInt32(keyBlob, keySizeOffset);
int keyOffset = keySizeOffset + sizeof(int);

return new ECParameters
{
Curve = ECCurve.NamedCurves.nistP384,
Q = new ECPoint
{
X = keyBlob.Skip(keyOffset).Take(keySize).ToArray(),
Y = keyBlob.Skip(keyOffset + keySize).Take(keySize).ToArray()
},
};
}
}
#endregion
}
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,6 @@
using System;
using System.Linq;
using System.Runtime.Caching;
using System.Runtime.InteropServices;
using System.Security.Cryptography;
using System.Security.Cryptography.X509Certificates;
using System.Threading;
Expand Down Expand Up @@ -296,15 +295,13 @@ private void VerifyEnclaveReportSignature(EnclaveReportPackage enclaveReportPack
}

// IDK_S is contained in healthReport cert public key
RSA rsacsp = healthReportCert.GetRSAPublicKey();
RSAParameters rsaparams = rsacsp.ExportParameters(includePrivateParameters: false);
RSACng rsacng = new RSACng();
rsacng.ImportParameters(rsaparams);

if (!rsacng.VerifyData(enclaveReportPackage.ReportAsBytes, enclaveReportPackage.SignatureBlob, HashAlgorithmName.SHA256, RSASignaturePadding.Pss))
using (RSA rsa = healthReportCert.GetRSAPublicKey())
{
throw new ArgumentException(SR.VerifyEnclaveReportFailed);

RSAParameters rsaparams = rsa.ExportParameters(includePrivateParameters: false);
if (!rsa.VerifyData(enclaveReportPackage.ReportAsBytes, enclaveReportPackage.SignatureBlob, HashAlgorithmName.SHA256, RSASignaturePadding.Pss))
{
throw new ArgumentException(SR.VerifyEnclaveReportFailed);
}
}
}

Expand Down Expand Up @@ -354,72 +351,18 @@ private byte[] GetSharedSecret(EnclavePublicKey enclavePublicKey, EnclaveDiffieH
{
// Perform signature verification. The enclave's DiffieHellman public key was signed by the enclave's RSA public key.
RSAParameters rsaParams = RSAKeyBlobToParams(enclavePublicKey.PublicKey);
RSA rsa = RSA.Create(rsaParams);

if (!rsa.VerifyData(enclaveDHInfo.PublicKey, enclaveDHInfo.PublicKeySignature, HashAlgorithmName.SHA256, RSASignaturePadding.Pkcs1))
using (RSA rsa = RSA.Create(rsaParams))
{
throw new ArgumentException(SR.GetSharedSecretFailed);
if (!rsa.VerifyData(enclaveDHInfo.PublicKey, enclaveDHInfo.PublicKeySignature, HashAlgorithmName.SHA256, RSASignaturePadding.Pkcs1))
{
throw new ArgumentException(SR.GetSharedSecretFailed);
}
}

ECParameters ecParams = ECCKeyBlobToParams(enclaveDHInfo.PublicKey);
ECDiffieHellman enclaveDHKey = ECDiffieHellman.Create(ecParams);

return clientDHKey.DeriveKeyMaterial(enclaveDHKey.PublicKey);
return clientDHKey.DeriveKeyFromHash(enclaveDHKey.PublicKey, HashAlgorithmName.SHA256);
}

// Extracts the public key's modulus and exponent from the key blob

// this works on windows because the win32 api uses the bcrypt structures
// will it work on linux?
// according to odbc docs, the modulus is always 512 bytes for a VBS-HGS enclave response
// so it should work for any vbs-hgs enclave provider...
private RSAParameters RSAKeyBlobToParams(byte[] keyBlob)
{
// The RSA public key blob is structured as follows:
// BCRYPT_RSAKEY_BLOB header
// byte[cbPublicExp] publicExponent - Exponent
// byte[cbModulus] modulus - Modulus

// The exponent is the final 3 bytes in the header
// The modulus is the final 512 bytes in the key blob
const int modulusSize = 512;
const int exponentSize = 3;
int BcryptRsaKeyBlobHeaderSize = keyBlob.Length - modulusSize;
int exponentOffset = BcryptRsaKeyBlobHeaderSize - exponentSize;
int modulusOffset = exponentOffset + exponentSize;

return new RSAParameters()
{
Exponent = keyBlob.Skip(exponentOffset).Take(exponentSize).ToArray(),
Modulus = keyBlob.Skip(modulusOffset).Take(modulusSize).ToArray()
};
}

// Extracts the public key's X and Y coordinate
private ECParameters ECCKeyBlobToParams(byte[] keyBlob)
{
// The ECC public key blob is structured as follows:
// BCRYPT_ECCKEY_BLOB header
// byte[cbKey] X - X coordinate
// byte[cbKey] Y - Y coordinate

// The size of each key is found after the first 4 byes (magic number)
const int keySizeOffset = 4;
int keySize = BitConverter.ToInt32(keyBlob, keySizeOffset);
int keyOffset = keySizeOffset + sizeof(int);

return new ECParameters
{
// should we read the magic bytes and set it based on that?
//Curve = ECCurve.NamedCurves.nistP384,
Q = new ECPoint
{
X = keyBlob.Skip(keyOffset).Take(keySize).ToArray(),
Y = keyBlob.Skip(keyOffset + keySize).Take(keySize).ToArray()
},
};
}

#endregion
}
}