diff --git a/doc/samples/AzureKeyVaultProviderWithEnclaveProviderExample.cs b/doc/samples/AzureKeyVaultProviderLegacyWithEnclaveProviderExample.cs
similarity index 100%
rename from doc/samples/AzureKeyVaultProviderWithEnclaveProviderExample.cs
rename to doc/samples/AzureKeyVaultProviderLegacyWithEnclaveProviderExample.cs
diff --git a/doc/samples/AzureKeyVaultProviderWithEnclaveProviderExample_2_0.cs b/doc/samples/AzureKeyVaultProviderWithEnclaveProviderExample_2_0.cs
new file mode 100644
index 0000000000..a4e288a1d9
--- /dev/null
+++ b/doc/samples/AzureKeyVaultProviderWithEnclaveProviderExample_2_0.cs
@@ -0,0 +1,253 @@
+using System;
+//
+using System.Collections.Generic;
+using System.Security.Cryptography;
+using System.Threading.Tasks;
+using Azure.Identity;
+using Microsoft.Data.SqlClient;
+using Microsoft.Data.SqlClient.AlwaysEncrypted.AzureKeyVaultProvider;
+
+namespace AKVEnclaveExample
+{
+ class Program
+ {
+ static readonly string s_algorithm = "RSA_OAEP";
+
+ // ********* Provide details here ***********
+ static readonly string s_akvUrl = "https://{KeyVaultName}.vault.azure.net/keys/{Key}/{KeyIdentifier}";
+ static readonly string s_clientId = "{Application_Client_ID}";
+ static readonly string s_clientSecret = "{Application_Client_Secret}";
+ static readonly string s_tenantId = "{Azure_Key_Vault_Active_Directory_Tenant_Id}";
+ static readonly string s_connectionString = "Server={Server}; Database={database}; Integrated Security=true; Column Encryption Setting=Enabled; Attestation Protocol=HGS; Enclave Attestation Url = {attestation_url_for_HGS};";
+ // ******************************************
+
+ static void Main(string[] args)
+ {
+ // Initialize AKV provider
+ ClientSecretCredential clientSecretCredential = new ClientSecretCredential(s_tenantId, s_clientId, s_clientSecret);
+ SqlColumnEncryptionAzureKeyVaultProvider akvProvider = new SqlColumnEncryptionAzureKeyVaultProvider(clientSecretCredential);
+
+ // Register AKV provider
+ SqlConnection.RegisterColumnEncryptionKeyStoreProviders(customProviders: new Dictionary(capacity: 1, comparer: StringComparer.OrdinalIgnoreCase)
+ {
+ { SqlColumnEncryptionAzureKeyVaultProvider.ProviderName, akvProvider}
+ });
+ Console.WriteLine("AKV provider Registered");
+
+ // Create connection to database
+ using (SqlConnection sqlConnection = new SqlConnection(s_connectionString))
+ {
+ string cmkName = "CMK_WITH_AKV";
+ string cekName = "CEK_WITH_AKV";
+ string tblName = "AKV_TEST_TABLE";
+
+ CustomerRecord customer = new CustomerRecord(1, @"Microsoft", @"Corporation");
+
+ try
+ {
+ sqlConnection.Open();
+
+ // Drop Objects if exists
+ dropObjects(sqlConnection, cmkName, cekName, tblName);
+
+ // Create Column Master Key with AKV Url
+ createCMK(sqlConnection, cmkName, akvProvider);
+ Console.WriteLine("Column Master Key created.");
+
+ // Create Column Encryption Key
+ createCEK(sqlConnection, cmkName, cekName, akvProvider);
+ Console.WriteLine("Column Encryption Key created.");
+
+ // Create Table with Encrypted Columns
+ createTbl(sqlConnection, cekName, tblName);
+ Console.WriteLine("Table created with Encrypted columns.");
+
+ // Insert Customer Record in table
+ insertData(sqlConnection, tblName, customer);
+ Console.WriteLine("Encryted data inserted.");
+
+ // Read data from table
+ verifyData(sqlConnection, tblName, customer);
+ Console.WriteLine("Data validated successfully.");
+ }
+ finally
+ {
+ // Drop table and keys
+ dropObjects(sqlConnection, cmkName, cekName, tblName);
+ Console.WriteLine("Dropped Table, CEK and CMK");
+ }
+
+ Console.WriteLine("Completed AKV provider Sample.");
+
+ Console.ReadKey();
+ }
+ }
+
+ private static void createCMK(SqlConnection sqlConnection, string cmkName, SqlColumnEncryptionAzureKeyVaultProvider sqlColumnEncryptionAzureKeyVaultProvider)
+ {
+ string KeyStoreProviderName = SqlColumnEncryptionAzureKeyVaultProvider.ProviderName;
+
+ byte[] cmkSign = sqlColumnEncryptionAzureKeyVaultProvider.SignColumnMasterKeyMetadata(s_akvUrl, true);
+ string cmkSignStr = string.Concat("0x", BitConverter.ToString(cmkSign).Replace("-", string.Empty));
+
+ string sql =
+ $@"CREATE COLUMN MASTER KEY [{cmkName}]
+ WITH (
+ KEY_STORE_PROVIDER_NAME = N'{KeyStoreProviderName}',
+ KEY_PATH = N'{s_akvUrl}',
+ ENCLAVE_COMPUTATIONS (SIGNATURE = {cmkSignStr})
+ );";
+
+ using (SqlCommand command = sqlConnection.CreateCommand())
+ {
+ command.CommandText = sql;
+ command.ExecuteNonQuery();
+ }
+ }
+
+ private static void createCEK(SqlConnection sqlConnection, string cmkName, string cekName, SqlColumnEncryptionAzureKeyVaultProvider sqlColumnEncryptionAzureKeyVaultProvider)
+ {
+ string sql =
+ $@"CREATE COLUMN ENCRYPTION KEY [{cekName}]
+ WITH VALUES (
+ COLUMN_MASTER_KEY = [{cmkName}],
+ ALGORITHM = '{s_algorithm}',
+ ENCRYPTED_VALUE = {GetEncryptedValue(sqlColumnEncryptionAzureKeyVaultProvider)}
+ )";
+
+ using (SqlCommand command = sqlConnection.CreateCommand())
+ {
+ command.CommandText = sql;
+ command.ExecuteNonQuery();
+ }
+ }
+
+ private static string GetEncryptedValue(SqlColumnEncryptionAzureKeyVaultProvider sqlColumnEncryptionAzureKeyVaultProvider)
+ {
+ byte[] plainTextColumnEncryptionKey = new byte[32];
+ RNGCryptoServiceProvider rngCsp = new RNGCryptoServiceProvider();
+ rngCsp.GetBytes(plainTextColumnEncryptionKey);
+
+ byte[] encryptedColumnEncryptionKey = sqlColumnEncryptionAzureKeyVaultProvider.EncryptColumnEncryptionKey(s_akvUrl, s_algorithm, plainTextColumnEncryptionKey);
+ string EncryptedValue = string.Concat("0x", BitConverter.ToString(encryptedColumnEncryptionKey).Replace("-", string.Empty));
+ return EncryptedValue;
+ }
+
+ private static void createTbl(SqlConnection sqlConnection, string cekName, string tblName)
+ {
+ string ColumnEncryptionAlgorithmName = @"AEAD_AES_256_CBC_HMAC_SHA_256";
+
+ string sql =
+ $@"CREATE TABLE [dbo].[{tblName}]
+ (
+ [CustomerId] [int] ENCRYPTED WITH (COLUMN_ENCRYPTION_KEY = [{cekName}], ENCRYPTION_TYPE = RANDOMIZED, ALGORITHM = '{ColumnEncryptionAlgorithmName}'),
+ [FirstName] [nvarchar](50) COLLATE Latin1_General_BIN2 ENCRYPTED WITH (COLUMN_ENCRYPTION_KEY = [{cekName}], ENCRYPTION_TYPE = RANDOMIZED, ALGORITHM = '{ColumnEncryptionAlgorithmName}'),
+ [LastName] [nvarchar](50) COLLATE Latin1_General_BIN2 ENCRYPTED WITH (COLUMN_ENCRYPTION_KEY = [{cekName}], ENCRYPTION_TYPE = RANDOMIZED, ALGORITHM = '{ColumnEncryptionAlgorithmName}')
+ )";
+
+ using (SqlCommand command = sqlConnection.CreateCommand())
+ {
+ command.CommandText = sql;
+ command.ExecuteNonQuery();
+ }
+ }
+
+ private static void insertData(SqlConnection sqlConnection, string tblName, CustomerRecord customer)
+ {
+ string insertSql = $"INSERT INTO [{tblName}] (CustomerId, FirstName, LastName) VALUES (@CustomerId, @FirstName, @LastName);";
+
+ using (SqlTransaction sqlTransaction = sqlConnection.BeginTransaction())
+ using (SqlCommand sqlCommand = new SqlCommand(insertSql,
+ connection: sqlConnection, transaction: sqlTransaction,
+ columnEncryptionSetting: SqlCommandColumnEncryptionSetting.Enabled))
+ {
+ sqlCommand.Parameters.AddWithValue(@"CustomerId", customer.Id);
+ sqlCommand.Parameters.AddWithValue(@"FirstName", customer.FirstName);
+ sqlCommand.Parameters.AddWithValue(@"LastName", customer.LastName);
+
+ sqlCommand.ExecuteNonQuery();
+ sqlTransaction.Commit();
+ }
+ }
+
+ private static void verifyData(SqlConnection sqlConnection, string tblName, CustomerRecord customer)
+ {
+ // Test INPUT parameter on an encrypted parameter
+ using (SqlCommand sqlCommand = new SqlCommand($"SELECT CustomerId, FirstName, LastName FROM [{tblName}] WHERE FirstName = @firstName",
+ sqlConnection))
+ {
+ SqlParameter customerFirstParam = sqlCommand.Parameters.AddWithValue(@"firstName", @"Microsoft");
+ customerFirstParam.Direction = System.Data.ParameterDirection.Input;
+ customerFirstParam.ForceColumnEncryption = true;
+
+ using (SqlDataReader sqlDataReader = sqlCommand.ExecuteReader())
+ {
+ ValidateResultSet(sqlDataReader);
+ }
+ }
+ }
+
+ private static void ValidateResultSet(SqlDataReader sqlDataReader)
+ {
+ Console.WriteLine(" * Row available: " + sqlDataReader.HasRows);
+
+ while (sqlDataReader.Read())
+ {
+ if (sqlDataReader.GetInt32(0) == 1)
+ {
+ Console.WriteLine(" * Employee Id received as sent: " + sqlDataReader.GetInt32(0));
+ }
+ else
+ {
+ Console.WriteLine("Employee Id didn't match");
+ }
+
+ if (sqlDataReader.GetString(1) == @"Microsoft")
+ {
+ Console.WriteLine(" * Employee Firstname received as sent: " + sqlDataReader.GetString(1));
+ }
+ else
+ {
+ Console.WriteLine("Employee FirstName didn't match.");
+ }
+
+ if (sqlDataReader.GetString(2) == @"Corporation")
+ {
+ Console.WriteLine(" * Employee LastName received as sent: " + sqlDataReader.GetString(2));
+ }
+ else
+ {
+ Console.WriteLine("Employee LastName didn't match.");
+ }
+ }
+ }
+
+ private static void dropObjects(SqlConnection sqlConnection, string cmkName, string cekName, string tblName)
+ {
+ using (SqlCommand cmd = sqlConnection.CreateCommand())
+ {
+ cmd.CommandText = $@"IF EXISTS (select * from sys.objects where name = '{tblName}') BEGIN DROP TABLE [{tblName}] END";
+ cmd.ExecuteNonQuery();
+ cmd.CommandText = $@"IF EXISTS (select * from sys.column_encryption_keys where name = '{cekName}') BEGIN DROP COLUMN ENCRYPTION KEY [{cekName}] END";
+ cmd.ExecuteNonQuery();
+ cmd.CommandText = $@"IF EXISTS (select * from sys.column_master_keys where name = '{cmkName}') BEGIN DROP COLUMN MASTER KEY [{cmkName}] END";
+ cmd.ExecuteNonQuery();
+ }
+ }
+
+ private class CustomerRecord
+ {
+ internal int Id { get; set; }
+ internal string FirstName { get; set; }
+ internal string LastName { get; set; }
+
+ public CustomerRecord(int id, string fName, string lName)
+ {
+ Id = id;
+ FirstName = fName;
+ LastName = lName;
+ }
+ }
+ }
+}
+//
diff --git a/src/Microsoft.Data.SqlClient/add-ons/AzureKeyVaultProvider/AzureSqlKeyCryptographer.cs b/src/Microsoft.Data.SqlClient/add-ons/AzureKeyVaultProvider/AzureSqlKeyCryptographer.cs
index fa99c44350..a10e65c1cd 100644
--- a/src/Microsoft.Data.SqlClient/add-ons/AzureKeyVaultProvider/AzureSqlKeyCryptographer.cs
+++ b/src/Microsoft.Data.SqlClient/add-ons/AzureKeyVaultProvider/AzureSqlKeyCryptographer.cs
@@ -61,9 +61,9 @@ internal void AddKey(string keyIdentifierUri)
{
if (TheKeyHasNotBeenCached(keyIdentifierUri))
{
- ParseAKVPath(keyIdentifierUri, out Uri vaultUri, out string keyName);
+ ParseAKVPath(keyIdentifierUri, out Uri vaultUri, out string keyName, out string keyVersion);
CreateKeyClient(vaultUri);
- FetchKey(vaultUri, keyName, keyIdentifierUri);
+ FetchKey(vaultUri, keyName, keyVersion, keyIdentifierUri);
}
bool TheKeyHasNotBeenCached(string k) => !_keyDictionary.ContainsKey(k) && !_keyFetchTaskDictionary.ContainsKey(k);
@@ -151,10 +151,11 @@ private CryptographyClient GetCryptographyClient(string keyIdentifierUri)
///
/// The Azure Key Vault URI
/// The name of the Azure Key Vault key
+ /// The version of the Azure Key Vault key
/// The Azure Key Vault key identifier
- private void FetchKey(Uri vaultUri, string keyName, string keyResourceUri)
+ private void FetchKey(Uri vaultUri, string keyName, string keyVersion, string keyResourceUri)
{
- Task> fetchKeyTask = FetchKeyFromKeyVault(vaultUri, keyName);
+ Task> fetchKeyTask = FetchKeyFromKeyVault(vaultUri, keyName, keyVersion);
_keyFetchTaskDictionary.AddOrUpdate(keyResourceUri, fetchKeyTask, (k, v) => fetchKeyTask);
fetchKeyTask
@@ -169,11 +170,12 @@ private void FetchKey(Uri vaultUri, string keyName, string keyResourceUri)
///
/// The Azure Key Vault URI
/// Then name of the key
+ /// Then version of the key
///
- private Task> FetchKeyFromKeyVault(Uri vaultUri, string keyName)
+ private Task> FetchKeyFromKeyVault(Uri vaultUri, string keyName, string keyVersion)
{
_keyClientDictionary.TryGetValue(vaultUri, out KeyClient keyClient);
- return keyClient.GetKeyAsync(keyName);
+ return keyClient.GetKeyAsync(keyName, keyVersion);
}
///
@@ -209,11 +211,13 @@ private void CreateKeyClient(Uri vaultUri)
/// The Azure Key Vault key identifier
/// The Azure Key Vault URI
/// The name of the key
- private void ParseAKVPath(string masterKeyPath, out Uri vaultUri, out string masterKeyName)
+ /// The version of the key
+ private void ParseAKVPath(string masterKeyPath, out Uri vaultUri, out string masterKeyName, out string masterKeyVersion)
{
Uri masterKeyPathUri = new Uri(masterKeyPath);
vaultUri = new Uri(masterKeyPathUri.GetLeftPart(UriPartial.Authority));
masterKeyName = masterKeyPathUri.Segments[2];
+ masterKeyVersion = masterKeyPathUri.Segments.Length > 3 ? masterKeyPathUri.Segments[3] : null;
}
}
}
diff --git a/src/Microsoft.Data.SqlClient/add-ons/AzureKeyVaultProvider/SqlColumnEncryptionAzureKeyVaultProvider.cs b/src/Microsoft.Data.SqlClient/add-ons/AzureKeyVaultProvider/SqlColumnEncryptionAzureKeyVaultProvider.cs
index 013c8e9cb0..4c3c56cee2 100644
--- a/src/Microsoft.Data.SqlClient/add-ons/AzureKeyVaultProvider/SqlColumnEncryptionAzureKeyVaultProvider.cs
+++ b/src/Microsoft.Data.SqlClient/add-ons/AzureKeyVaultProvider/SqlColumnEncryptionAzureKeyVaultProvider.cs
@@ -35,7 +35,7 @@ namespace Microsoft.Data.SqlClient.AlwaysEncrypted.AzureKeyVaultProvider
/// API only once in the lifetime of driver to register this custom provider by implementing a custom Authentication Callback mechanism.
///
@@ -117,10 +117,10 @@ public SqlColumnEncryptionAzureKeyVaultProvider(TokenCredential tokenCredential,
#region Public methods
///
- /// Uses an asymmetric key identified by the key path to sign the masterkey metadata consisting of (masterKeyPath, allowEnclaveComputations bit, providerName).
+ /// Uses an asymmetric key identified by the key path to sign the master key metadata consisting of (masterKeyPath, allowEnclaveComputations bit, providerName).
///
/// Complete path of an asymmetric key. Path format is specific to a key store provider.
- /// Boolean indicating whether this key can be sent to trusted enclave
+ /// Boolean indicating whether this key can be sent to a trusted enclave
/// Encrypted column encryption key
public override byte[] SignColumnMasterKeyMetadata(string masterKeyPath, bool allowEnclaveComputations)
{
@@ -133,7 +133,7 @@ public override byte[] SignColumnMasterKeyMetadata(string masterKeyPath, bool al
}
///
- /// Uses an asymmetric key identified by the key path to verify the masterkey metadata consisting of (masterKeyPath, allowEnclaveComputations bit, providerName).
+ /// Uses an asymmetric key identified by the key path to verify the master key metadata consisting of (masterKeyPath, allowEnclaveComputations bit, providerName).
///
/// Complete path of an asymmetric key. Path format is specific to a key store provider.
/// Boolean indicating whether this key can be sent to trusted enclave
@@ -153,7 +153,7 @@ public override bool VerifyColumnMasterKeyMetadata(string masterKeyPath, bool al
/// This function uses the asymmetric key specified by the key path
/// and decrypts an encrypted CEK with RSA encryption algorithm.
///
- /// Complete path of an asymmetric key in AKV
+ /// Complete path of an asymmetric key in Azure Key Vault
/// Asymmetric Key Encryption Algorithm
/// Encrypted Column Encryption Key
/// Plain text column encryption key
@@ -234,7 +234,7 @@ public override byte[] DecryptColumnEncryptionKey(string masterKeyPath, string e
/// This function uses the asymmetric key specified by the key path
/// and encrypts CEK with RSA encryption algorithm.
///
- /// Complete path of an asymmetric key in AKV
+ /// Complete path of an asymmetric key in Azure Key Vault
/// Asymmetric Key Encryption Algorithm
/// Plain text column encryption key
/// Encrypted column encryption key
@@ -253,7 +253,7 @@ public override byte[] EncryptColumnEncryptionKey(string masterKeyPath, string e
// Construct the encryptedColumnEncryptionKey
// Format is
- // s_firstVersion + keyPathLength + ciphertextLength + ciphertext + keyPath + signature
+ // s_firstVersion + keyPathLength + ciphertextLength + keyPath + ciphertext + signature
// Get the Unicode encoded bytes of cultureinvariant lower case masterKeyPath
byte[] masterKeyPathBytes = Encoding.Unicode.GetBytes(masterKeyPath.ToLowerInvariant());
@@ -310,8 +310,7 @@ internal void ValidateNonEmptyAKVPath(string masterKeyPath, bool isSystemOp)
throw new ArgumentException(errorMessage, Constants.AeParamMasterKeyPath);
}
-
- if (!Uri.TryCreate(masterKeyPath, UriKind.Absolute, out Uri parsedUri))
+ if (!Uri.TryCreate(masterKeyPath, UriKind.Absolute, out Uri parsedUri) || parsedUri.Segments.Length < 3)
{
// Return an error indicating that the AKV url is invalid.
throw new ArgumentException(string.Format(CultureInfo.InvariantCulture, Strings.InvalidAkvUrlTemplate, masterKeyPath), Constants.AeParamMasterKeyPath);
diff --git a/src/Microsoft.Data.SqlClient/tests/ManualTests/AlwaysEncrypted/AKVUnitTests.cs b/src/Microsoft.Data.SqlClient/tests/ManualTests/AlwaysEncrypted/AKVUnitTests.cs
index d58ee69281..c042339dc2 100644
--- a/src/Microsoft.Data.SqlClient/tests/ManualTests/AlwaysEncrypted/AKVUnitTests.cs
+++ b/src/Microsoft.Data.SqlClient/tests/ManualTests/AlwaysEncrypted/AKVUnitTests.cs
@@ -5,6 +5,10 @@
using Microsoft.Data.SqlClient.AlwaysEncrypted.AzureKeyVaultProvider;
using Azure.Identity;
using Xunit;
+using Azure.Security.KeyVault.Keys;
+using Azure.Core;
+using System.Reflection;
+using System;
namespace Microsoft.Data.SqlClient.ManualTesting.Tests.AlwaysEncrypted
{
@@ -52,5 +56,52 @@ public static void TokenCredentialRotationTest()
byte[] decryptedCekWithNewProvider = newAkvProvider.DecryptColumnEncryptionKey(DataTestUtility.AKVUrl, EncryptionAlgorithm, encryptedCekWithOldProvider);
Assert.Equal(s_columnEncryptionKey, decryptedCekWithNewProvider);
}
+
+ [ConditionalFact(typeof(DataTestUtility), nameof(DataTestUtility.IsAKVSetupAvailable))]
+ public static void ReturnSpecifiedVersionOfKeyWhenItIsNotTheMostRecentVersion()
+ {
+ Uri keyPathUri = new Uri(DataTestUtility.AKVOriginalUrl);
+ Uri vaultUri = new Uri(keyPathUri.GetLeftPart(UriPartial.Authority));
+
+ //If key version is not specified then we cannot test.
+ if (KeyIsVersioned(keyPathUri))
+ {
+ string keyName = keyPathUri.Segments[2];
+ string keyVersion = keyPathUri.Segments[3];
+ ClientSecretCredential clientSecretCredential = new ClientSecretCredential(DataTestUtility.AKVTenantId, DataTestUtility.AKVClientId, DataTestUtility.AKVClientSecret);
+ KeyClient keyClient = new KeyClient(vaultUri, clientSecretCredential);
+ KeyVaultKey currentVersionKey = keyClient.GetKey(keyName);
+ KeyVaultKey specifiedVersionKey = keyClient.GetKey(keyName, keyVersion);
+
+ //If specified versioned key is the most recent version of the key then we cannot test.
+ if (!KeyIsLatestVersion(specifiedVersionKey, currentVersionKey))
+ {
+ SqlColumnEncryptionAzureKeyVaultProvider azureKeyProvider = new SqlColumnEncryptionAzureKeyVaultProvider(clientSecretCredential);
+ // Perform an operation to initialize the internal caches
+ azureKeyProvider.EncryptColumnEncryptionKey(DataTestUtility.AKVOriginalUrl, EncryptionAlgorithm, s_columnEncryptionKey);
+
+ PropertyInfo keyCryptographerProperty = azureKeyProvider.GetType().GetProperty("KeyCryptographer", BindingFlags.NonPublic | BindingFlags.Instance);
+ var keyCryptographer = keyCryptographerProperty.GetValue(azureKeyProvider);
+ MethodInfo getKeyMethod = keyCryptographer.GetType().GetMethod("GetKey", BindingFlags.NonPublic | BindingFlags.Instance);
+ KeyVaultKey key = (KeyVaultKey)getKeyMethod.Invoke(keyCryptographer, new[] { DataTestUtility.AKVOriginalUrl });
+
+ Assert.Equal(keyVersion, key.Properties.Version);
+ }
+ }
+ }
+
+ static bool KeyIsVersioned(Uri keyPath) => keyPath.Segments.Length > 3;
+ static bool KeyIsLatestVersion(KeyVaultKey specifiedVersionKey, KeyVaultKey currentVersionKey) => currentVersionKey.Properties.Version == specifiedVersionKey.Properties.Version;
+
+ [ConditionalFact(typeof(DataTestUtility), nameof(DataTestUtility.IsAKVSetupAvailable))]
+ public static void ThrowWhenUrlHasLessThanThreeSegments()
+ {
+ SqlColumnEncryptionAzureKeyVaultProvider azureKeyProvider = new SqlColumnEncryptionAzureKeyVaultProvider(new SqlClientCustomTokenCredential());
+ string invalidKeyPath = "https://my-key-vault.vault.azure.net/keys";
+ Exception ex1 = Assert.Throws(() => azureKeyProvider.EncryptColumnEncryptionKey(invalidKeyPath, EncryptionAlgorithm, s_columnEncryptionKey));
+ Assert.Contains($"Invalid url specified: '{invalidKeyPath}'", ex1.Message);
+ Exception ex2 = Assert.Throws(() => azureKeyProvider.DecryptColumnEncryptionKey(invalidKeyPath, EncryptionAlgorithm, s_columnEncryptionKey));
+ Assert.Contains($"Invalid url specified: '{invalidKeyPath}'", ex2.Message);
+ }
}
}
diff --git a/src/Microsoft.Data.SqlClient/tests/ManualTests/DataCommon/DataTestUtility.cs b/src/Microsoft.Data.SqlClient/tests/ManualTests/DataCommon/DataTestUtility.cs
index 0f9211ee3a..617f905ff7 100644
--- a/src/Microsoft.Data.SqlClient/tests/ManualTests/DataCommon/DataTestUtility.cs
+++ b/src/Microsoft.Data.SqlClient/tests/ManualTests/DataCommon/DataTestUtility.cs
@@ -15,6 +15,8 @@
using Microsoft.Identity.Client;
using Microsoft.Data.SqlClient.TestUtilities;
using Xunit;
+using Azure.Security.KeyVault.Keys;
+using Azure.Identity;
namespace Microsoft.Data.SqlClient.ManualTesting.Tests
{
@@ -32,6 +34,7 @@ public static class DataTestUtility
public static readonly string AADAccessToken = null;
public static readonly string AKVBaseUrl = null;
public static readonly string AKVUrl = null;
+ public static readonly string AKVOriginalUrl = null;
public static readonly string AKVTenantId = null;
public static readonly string AKVClientId = null;
public static readonly string AKVClientSecret = null;
@@ -103,14 +106,14 @@ static DataTestUtility()
AADAccessToken = GenerateAccessToken(AADAuthorityURL, username, password);
}
- string url = c.AzureKeyVaultURL;
- if (!string.IsNullOrEmpty(url) && Uri.TryCreate(url, UriKind.Absolute, out Uri AKVBaseUri))
+ AKVOriginalUrl = c.AzureKeyVaultURL;
+ if (!string.IsNullOrEmpty(AKVOriginalUrl) && Uri.TryCreate(AKVOriginalUrl, UriKind.Absolute, out Uri AKVBaseUri))
{
AKVBaseUri = new Uri(AKVBaseUri, "/");
AKVBaseUrl = AKVBaseUri.AbsoluteUri;
AKVUrl = (new Uri(AKVBaseUri, $"/keys/{AKVKeyName}")).AbsoluteUri;
}
-
+
AKVTenantId = c.AzureKeyVaultTenantId;
AKVClientId = c.AzureKeyVaultClientId;
AKVClientSecret = c.AzureKeyVaultClientSecret;