diff --git a/sdk/keyvault/Azure.Security.KeyVault.Secrets/MigrationGuide.md b/sdk/keyvault/Azure.Security.KeyVault.Secrets/MigrationGuide.md new file mode 100644 index 0000000000000..fc7685e4504b4 --- /dev/null +++ b/sdk/keyvault/Azure.Security.KeyVault.Secrets/MigrationGuide.md @@ -0,0 +1,251 @@ +# Migrate from Microsoft.Azure.KeyVault to Azure.Security.KeyVault.Secrets + +This guide is intended to assist in the migration to version 4 of the Key Vault client library [`Azure.Security.KeyVault.Secrets`](https://www.nuget.org/packages/Azure.Security.KeyVault.Secrets) from version 3 of [`Microsoft.Azure.KeyVault`](https://www.nuget.org/packages/Microsoft.Azure.KeyVault). It will focus on side-by-side comparisons for similar operations between the two packages. + +Familiarity with the `Microsoft.Azure.KeyVault` library is assumed. For those new to the Key Vault client library for .NET, please refer to the [README](https://github.com/Azure/azure-sdk-for-net/blob/master/sdk/keyvault/Azure.Security.KeyVault.Secrets/README.md) and [Key Vault secrets samples](https://github.com/Azure/azure-sdk-for-net/tree/master/sdk/keyvault/Azure.Security.KeyVault.Secrets/samples) for the `Azure.Security.KeyVault.Secrets` library rather than this guide. + +## Table of contents + +- [Migration benefits](#migration-benefits) +- [General changes](#general-changes) + - [Package and namespaces](#package-and-namespaces) + - [Separate clients](#separate-clients) + - [Client constructors](#client-constructors) + - [Setting secrets](#setting-secrets) + - [Getting secrets](#getting-secrets) + - [Listing secrets](#listing-secrets) + - [Deleting secrets](#deleting-secrets) +- [Additional samples](#additional-samples) + +## Migration benefits + +A natural question to ask when considering whether or not to adopt a new version or library is what the benefits of doing so would be. As Azure has matured and been embraced by a more diverse group of developers, we have been focused on learning the patterns and practices to best support developer productivity and to understand the gaps that the .NET client libraries have. + +There were several areas of consistent feedback expressed across the Azure client library ecosystem. One of the most important is that the client libraries for different Azure services have not had a consistent approach to organization, naming, and API structure. Additionally, many developers have felt that the learning curve was difficult, and the APIs did not offer a good, approachable, and consistent onboarding story for those learning Azure or exploring a specific Azure service. + +To try and improve the development experience across Azure services, including Key Vault, a set of uniform [design guidelines](https://azure.github.io/azure-sdk/general_introduction.html) was created for all languages to drive a consistent experience with established API patterns for all services. A set of [.NET-specific guidelines](https://azure.github.io/azure-sdk/dotnet_introduction.html) was also introduced to ensure that .NET clients have a natural and idiomatic feel that mirrors that of the .NET base class libraries. Further details are available in the guidelines for those interested. + +The new Key Vault secrets library `Azure.Security.KeyVault.Secrets` provides the ability to share in some of the cross-service improvements made to the Azure development experience, such as using the new `Azure.Identity` library to share a single authentication between clients and a unified diagnostics pipeline offering a common view of the activities across each of the client libraries. + +While we believe that there is significant benefit to adopting the new Key Vault secrets library `Azure.Security.KeyVault.Secrets`, it is important to be aware that the previous version `Microsoft.Azure.KeyVault` has not been officially deprecated. It will continue to be supported with security and critical bug fixes. However, it is not under active development and will eventually be deprecated. + +## General changes + +### Package and namespaces + +Package names and the namespace root for the modern Azure client libraries for .NET have changed. Each will follow the pattern `Azure.[Area].[Services]` where the legacy clients followed the pattern `Microsoft.Azure.[Service]`. This provides a quick and accessible means to help understand, at a glance, whether you are using the modern or legacy clients. + +In the case of Key Vault, the modern client libraries have packages and namespaces that begin with `Azure.Security.KeyVault` and were released beginning with version 4. The legacy client libraries have packages and namespaces that begin with `Microsoft.Azure.KeyVault` and a version of 3.x.x or below. + +### Separate clients + +In the interest of simplifying the API we've split `KeyVaultClient` into separate packages and clients: + +- `Azure.Security.KeyVault.Certificates` contains `CertificateClient` for certificate management operations. +- `Azure.Security.KeyVault.Keys` contains `KeyClient` for key management operations and `CryptographyClient` for cryptographic operations. +- `Azure.Security.KeyVault.Secrets` contains `SecretClient` for secret management operations. + +Because [Role-Based Access Control (RBAC)](https://docs.microsoft.com/azure/role-based-access-control/overview) is recommended, we did not implement APIs for Key Vault-managed storage accounts. See [our sample](https://docs.microsoft.com/samples/azure/azure-sdk-for-net/share-link/) for source you can copy into your project if you require managing [Shared Access Signature (SAS)](https://docs.microsoft.com/azure/storage/common/storage-sas-overview) tokens in your application and cannot use RBAC. + +These clients also share a single connection pool by default despite being separated, resolving an issue with the old `KeyVaultClient` that created a new connection pool with each new instance and could exhaust socket connections. + +### Client constructors + +Across all new Azure client libraries, clients consistently take an endpoint or connection string along with token credentials. This differs from `KeyVaultClient` that took an authentication delegate and could be used for multiple Key Vault endpoints. + +#### Authenticating + +Previously in `Microsoft.Azure.KeyVault`, you could create a `KeyVaultClient` along with the `AzureServiceTokenProvider` from the package `Microsoft.Azure.Services.AppAuthentication`: + +```C# Snippet:Microsoft_Azure_KeyVault_Snippets_MigrationGuide_Create +AzureServiceTokenProvider provider = new AzureServiceTokenProvider(); +KeyVaultClient client = new KeyVaultClient( + new KeyVaultClient.AuthenticationCallback(provider.KeyVaultTokenCallback)); +``` + +Now in `Azure.Security.KeyVault.Secrets`, you create a `SecretClient` along with the `DefaultAzureCredential` from the package `Azure.Identity`: + +```C# Snippet:Azure_Security_KeyVault_Secrets_Snippets_MigrationGuide_Create +SecretClient client = new SecretClient( + new Uri("https://myvault.vault.azure.net"), + new DefaultAzureCredential()); +``` + +[`DefaultAzureCredential`](https://github.com/Azure/azure-sdk-for-net/blob/master/sdk/identity/Azure.Identity/README.md) is optimized for both production and development environments without having to change your source code. + +#### Sharing an HttpClient + +By default, all client libraries built on `Azure.Core` that communicate over HTTP share a single `HttpClient`. In `Microsoft.Azure.KeyVault` with `KeyVaultClient`, a new `HttpClient` was created with each instance but could be shared to prevent connection starvation: + +```C# Snippet:Microsoft_Azure_KeyVault_Snippets_MigrationGuide_CreateWithOptions +using (HttpClient httpClient = new HttpClient()) +{ + AzureServiceTokenProvider provider = new AzureServiceTokenProvider(); + KeyVaultClient client = new KeyVaultClient( + new KeyVaultClient.AuthenticationCallback(provider.KeyVaultTokenCallback), + httpClient); +} +``` + +In `Azure.Security.KeyVault.Secrets`, if you want to share an `HttpClient` with Azure client libraries and other clients you use or implement in your projects, you can pass it via `SecretClientOptions`: + +```C# Snippet:Azure_Security_KeyVault_Secrets_Snippets_MigrationGuide_CreateWithOptions +using (HttpClient httpClient = new HttpClient()) +{ + SecretClientOptions options = new SecretClientOptions + { + Transport = new HttpClientTransport(httpClient) + }; + + SecretClient client = new SecretClient( + new Uri("https://myvault.vault.azure.net"), + new DefaultAzureCredential(), + options); +} +``` + +[`ClientOptions`](https://azure.github.io/azure-sdk/dotnet_introduction.html#dotnet-usage-options) classes are another common feature of Azure client libraries to configure clients, including diagnostics, retry options, and transport options including your pipeline policies. + +### Setting secrets + +Previously in `Microsoft.Azure.KeyVault`, you could set a secret value using the `KeyVaultClient` and a specific Key Vault endpoint: + +```C# Snippet:Microsoft_Azure_KeyVault_Snippets_MigrationGuide_SetSecret +SecretBundle secret = await client.SetSecretAsync("https://myvault.vault.azure.net", "secret-name", "secret-value"); +``` + +Now in `Azure.Security.KeyVault.Secrets`, you set a secret value in the Key Vault you specified when constructing the `SecretClient`: + +```C# Snippet:Azure_Security_KeyVault_Secrets_Snippets_MigrationGuide_SetSecret +KeyVaultSecret secret = await client.SetSecretAsync("secret-name", "secret-value"); +``` + +Setting an existing secret in both cases will create a new version of the secret. + +Synchronous methods are also available on `SecretClient`, though we recommend you use asynchronous methods throughout your projects when possible for better performing applications. + +### Getting secrets + +Previously in `Microsoft.Azure.KeyVault`, you could get a secret value using the `KeyVaultClient` and a specific Key Vault endpoint: + +```C# Snippet:Microsoft_Azure_KeyVault_Snippets_MigrationGuide_GetSecret +// Get the latest secret value. +SecretBundle secret = await client.GetSecretAsync("https://myvault.vault.azure.net", "secret-name", null); + +// Get a specific secret value. +SecretBundle secretVersion = await client.GetSecretAsync("https://myvault.vault.azure.net", "secret-name", "e43af03a7cbc47d4a4e9f11540186048"); +``` + +Now in `Azure.Security.KeyVault.Secrets`, you get a secret value in the Key Vault you specified when constructing the `SecretClient`: + +```C# Snippet:Azure_Security_KeyVault_Secrets_Snippets_MigrationGuide_GetSecret +// Get the latest secret value. +KeyVaultSecret secret = await client.GetSecretAsync("secret-name"); + +// Get a specific secret value. +KeyVaultSecret secretVersion = await client.GetSecretAsync("secret-name", "e43af03a7cbc47d4a4e9f11540186048"); +``` + +Synchronous methods are also available on `SecretClient`, though we recommend you use asynchronous methods throughout your projects when possible for better performing applications. + +### Listing secrets + +Previously in `Microsoft.Azure.KeyVault`, you could list secrets' properties using the `KeyVaultClient` and a specific Key Vault endpoint: + +```C# Snippet:Microsoft_Azure_KeyVault_Snippets_MigrationGuide_ListSecrets +IPage page = await client.GetSecretsAsync("https://myvault.vault.azure.net"); +foreach (SecretItem item in page) +{ + SecretIdentifier secretId = item.Identifier; + SecretBundle secret = await client.GetSecretAsync(secretId.Vault, secretId.Name); +} + +while (page.NextPageLink != null) +{ + page = await client.GetSecretsNextAsync(page.NextPageLink); + foreach (SecretItem item in page) + { + SecretIdentifier secretId = item.Identifier; + SecretBundle secret = await client.GetSecretAsync(secretId.Vault, secretId.Name); + } +} +``` + +Now in `Azure.Security.KeyVault.Secrets`, you list secrets' properties in the Key Vault you specified when constructing the `SecretClient`. This returns an enumerable that enumerates all secrets across any number of pages. If you want to enumerate pages, call the `AsPages` method on the returned enumerable. + +```C# Snippet:Azure_Security_KeyVault_Secrets_Snippets_MigrationGuide_ListSecrets +// List all secrets asynchronously. +await foreach (SecretProperties item in client.GetPropertiesOfSecretsAsync()) +{ + KeyVaultSecret secret = await client.GetSecretAsync(item.Name); +} + +// List all secrets synchronously. +foreach (SecretProperties item in client.GetPropertiesOfSecrets()) +{ + KeyVaultSecret secret = client.GetSecret(item.Name); +} +``` + +### Deleting secrets + +Previously in `Microsoft.Azure.KeyVault`, you could delete a secret using the `KeyVaultClient` and a specific Key Vault endpoint: + +```C# Snippet:Microsoft_Azure_KeyVault_Snippets_MigrationGuide_DeleteSecret +// Delete the secret. +DeletedSecretBundle deletedSecret = await client.DeleteSecretAsync("https://myvault.vault.azure.net", "secret-name"); + +// Purge or recover the deleted secret if soft delete is enabled. +if (deletedSecret.RecoveryId != null) +{ + DeletedSecretIdentifier deletedSecretId = deletedSecret.RecoveryIdentifier; + + // Deleting a secret does not happen immediately. Wait a while and check if the deleted secret exists. + while (true) + { + try + { + await client.GetDeletedSecretAsync(deletedSecretId.Vault, deletedSecretId.Name); + + // Finally deleted. + break; + } + catch (KeyVaultErrorException ex) when (ex.Response.StatusCode == HttpStatusCode.NotFound) + { + // Not yet deleted... + } + } + + // Purge the deleted secret. + await client.PurgeDeletedSecretAsync(deletedSecretId.Vault, deletedSecretId.Name); + + // You can also recover the deleted secret using RecoverDeletedSecretAsync. +} +``` + +Now in `Azure.Security.KeyVault.Secrets`, you delete a secret in the Key Vault you specified when constructing the `SecretClient`: + +```C# Snippet:Azure_Security_KeyVault_Secrets_Snippets_MigrationGuide_DeleteSecret +// Delete the secret. +DeleteSecretOperation deleteOperation = await client.StartDeleteSecretAsync("secret-name"); + +// Purge or recover the deleted secret if soft delete is enabled. +if (deleteOperation.Value.RecoveryId != null) +{ + // Deleting a secret does not happen immediately. Wait for the secret to be deleted. + DeletedSecret deletedSecret = await deleteOperation.WaitForCompletionAsync(); + + // Purge the deleted secret. + await client.PurgeDeletedSecretAsync(deletedSecret.Name); + + // You can also recover the deleted secret using StartRecoverDeletedSecretAsync, + // which returns RecoverDeletedSecretOperation you can await like DeleteSecretOperation above. +} +``` + +Synchronous methods are also available on `SecretClient`, though we recommend you use asynchronous methods throughout your projects when possible for better performing applications. + +## Additional samples + +- [Key Vault secrets samples for .NET](https://docs.microsoft.com/samples/azure/azure-sdk-for-net/azuresecuritykeyvaultsecrets-samples/) +- [All Key Vault samples for .NET](https://docs.microsoft.com/samples/browse/?products=azure-key-vault&languages=csharp) diff --git a/sdk/keyvault/Azure.Security.KeyVault.Secrets/tests/samples/SampleSnippets.cs b/sdk/keyvault/Azure.Security.KeyVault.Secrets/tests/samples/SampleSnippets.cs index 7cdc14dd38471..ccb5cdd5114fd 100644 --- a/sdk/keyvault/Azure.Security.KeyVault.Secrets/tests/samples/SampleSnippets.cs +++ b/sdk/keyvault/Azure.Security.KeyVault.Secrets/tests/samples/SampleSnippets.cs @@ -1,12 +1,13 @@ // Copyright (c) Microsoft Corporation. All rights reserved. // Licensed under the MIT License. +using Azure.Core.Pipeline; using Azure.Identity; using NUnit.Framework; using System; using System.Threading; using System.Threading.Tasks; -using Azure.Security.KeyVault.Tests; +using System.Net.Http; namespace Azure.Security.KeyVault.Secrets.Samples { @@ -187,5 +188,83 @@ public void DeleteAndPurgeSecret() client.PurgeDeletedSecret(secret.Name); #endregion } + + [Ignore("Used only for the migration guide")] + private async Task MigrationGuide() + { + #region Snippet:Azure_Security_KeyVault_Secrets_Snippets_MigrationGuide_Create + SecretClient client = new SecretClient( + new Uri("https://myvault.vault.azure.net"), + new DefaultAzureCredential()); + #endregion Snippet:Azure_Security_KeyVault_Secrets_Snippets_MigrationGuide_Create + + #region Snippet:Azure_Security_KeyVault_Secrets_Snippets_MigrationGuide_CreateWithOptions + using (HttpClient httpClient = new HttpClient()) + { + SecretClientOptions options = new SecretClientOptions + { + Transport = new HttpClientTransport(httpClient) + }; + + //@@SecretClient client = new SecretClient( + /*@@*/ SecretClient _ = new SecretClient( + new Uri("https://myvault.vault.azure.net"), + new DefaultAzureCredential(), + options); + } + #endregion Snippet:Azure_Security_KeyVault_Secrets_Snippets_MigrationGuide_CreateWithOptions + + { + #region Snippet:Azure_Security_KeyVault_Secrets_Snippets_MigrationGuide_SetSecret + KeyVaultSecret secret = await client.SetSecretAsync("secret-name", "secret-value"); + #endregion Snippet:Azure_Security_KeyVault_Secrets_Snippets_MigrationGuide_SetSecret + } + + { + #region Snippet:Azure_Security_KeyVault_Secrets_Snippets_MigrationGuide_GetSecret + // Get the latest secret value. + KeyVaultSecret secret = await client.GetSecretAsync("secret-name"); + + // Get a specific secret value. + KeyVaultSecret secretVersion = await client.GetSecretAsync("secret-name", "e43af03a7cbc47d4a4e9f11540186048"); + #endregion Snippet:Azure_Security_KeyVault_Secrets_Snippets_MigrationGuide_GetSecret + } + + { + #region Snippet:Azure_Security_KeyVault_Secrets_Snippets_MigrationGuide_ListSecrets + // List all secrets asynchronously. + await foreach (SecretProperties item in client.GetPropertiesOfSecretsAsync()) + { + KeyVaultSecret secret = await client.GetSecretAsync(item.Name); + } + + // List all secrets synchronously. + foreach (SecretProperties item in client.GetPropertiesOfSecrets()) + { + KeyVaultSecret secret = client.GetSecret(item.Name); + } + #endregion Snippet:Azure_Security_KeyVault_Secrets_Snippets_MigrationGuide_ListSecrets + } + + { + #region Snippet:Azure_Security_KeyVault_Secrets_Snippets_MigrationGuide_DeleteSecret + // Delete the secret. + DeleteSecretOperation deleteOperation = await client.StartDeleteSecretAsync("secret-name"); + + // Purge or recover the deleted secret if soft delete is enabled. + if (deleteOperation.Value.RecoveryId != null) + { + // Deleting a secret does not happen immediately. Wait for the secret to be deleted. + DeletedSecret deletedSecret = await deleteOperation.WaitForCompletionAsync(); + + // Purge the deleted secret. + await client.PurgeDeletedSecretAsync(deletedSecret.Name); + + // You can also recover the deleted secret using StartRecoverDeletedSecretAsync, + // which returns RecoverDeletedSecretOperation you can await like DeleteSecretOperation above. + } + #endregion Snippet:Azure_Security_KeyVault_Secrets_Snippets_MigrationGuide_DeleteSecret + } + } } } diff --git a/sdk/keyvault/Microsoft.Azure.KeyVault/tests/Microsoft.Azure.KeyVault.Tests.csproj b/sdk/keyvault/Microsoft.Azure.KeyVault/tests/Microsoft.Azure.KeyVault.Tests.csproj index 1a5177d492f30..c364e694c065d 100644 --- a/sdk/keyvault/Microsoft.Azure.KeyVault/tests/Microsoft.Azure.KeyVault.Tests.csproj +++ b/sdk/keyvault/Microsoft.Azure.KeyVault/tests/Microsoft.Azure.KeyVault.Tests.csproj @@ -27,6 +27,7 @@ + diff --git a/sdk/keyvault/Microsoft.Azure.KeyVault/tests/SampleSnippets.cs b/sdk/keyvault/Microsoft.Azure.KeyVault/tests/SampleSnippets.cs new file mode 100644 index 0000000000000..c4b79bcf4c7b9 --- /dev/null +++ b/sdk/keyvault/Microsoft.Azure.KeyVault/tests/SampleSnippets.cs @@ -0,0 +1,111 @@ +// Copyright (c) Microsoft Corporation. All rights reserved. +// Licensed under the MIT License. See License.txt in the project root for +// license information. + +namespace Microsoft.Azure.KeyVault.Tests +{ + using System; + using System.Net; + using System.Net.Http; + using System.Threading.Tasks; + using Microsoft.Rest.Azure; + using Microsoft.Azure.KeyVault; + using Microsoft.Azure.KeyVault.Models; + using Microsoft.Azure.Services.AppAuthentication; + + // Used to compile sample snippets used in Azure.Security.KeyVault.Secrets' MigrationGuide.md. + internal class SampleSnippets + { + private async Task MigrationGuide() + { + #region Snippet:Microsoft_Azure_KeyVault_Snippets_MigrationGuide_Create + AzureServiceTokenProvider provider = new AzureServiceTokenProvider(); + KeyVaultClient client = new KeyVaultClient( + new KeyVaultClient.AuthenticationCallback(provider.KeyVaultTokenCallback)); + #endregion Snippet:Microsoft_Azure_KeyVault_Snippets_MigrationGuide_Create + + #region Snippet:Microsoft_Azure_KeyVault_Snippets_MigrationGuide_CreateWithOptions + using (HttpClient httpClient = new HttpClient()) + { + //@@AzureServiceTokenProvider provider = new AzureServiceTokenProvider(); + /*@@*/ provider = new AzureServiceTokenProvider(); + //@@KeyVaultClient client = new KeyVaultClient( + /*@@*/ client = new KeyVaultClient( + new KeyVaultClient.AuthenticationCallback(provider.KeyVaultTokenCallback), + httpClient); + } + #endregion Snippet:Microsoft_Azure_KeyVault_Snippets_MigrationGuide_CreateWithOptions + + { + #region Snippet:Microsoft_Azure_KeyVault_Snippets_MigrationGuide_SetSecret + SecretBundle secret = await client.SetSecretAsync("https://myvault.vault.azure.net", "secret-name", "secret-value"); + #endregion Snippet:Microsoft_Azure_KeyVault_Snippets_MigrationGuide_SetSecret + } + + { + #region Snippet:Microsoft_Azure_KeyVault_Snippets_MigrationGuide_GetSecret + // Get the latest secret value. + SecretBundle secret = await client.GetSecretAsync("https://myvault.vault.azure.net", "secret-name", null); + + // Get a specific secret value. + SecretBundle secretVersion = await client.GetSecretAsync("https://myvault.vault.azure.net", "secret-name", "e43af03a7cbc47d4a4e9f11540186048"); + #endregion Snippet:Microsoft_Azure_KeyVault_Snippets_MigrationGuide_GetSecret + } + + { + #region Snippet:Microsoft_Azure_KeyVault_Snippets_MigrationGuide_ListSecrets + IPage page = await client.GetSecretsAsync("https://myvault.vault.azure.net"); + foreach (SecretItem item in page) + { + SecretIdentifier secretId = item.Identifier; + SecretBundle secret = await client.GetSecretAsync(secretId.Vault, secretId.Name); + } + + while (page.NextPageLink != null) + { + page = await client.GetSecretsNextAsync(page.NextPageLink); + foreach (SecretItem item in page) + { + SecretIdentifier secretId = item.Identifier; + SecretBundle secret = await client.GetSecretAsync(secretId.Vault, secretId.Name); + } + } + #endregion Snippet:Microsoft_Azure_KeyVault_Snippets_MigrationGuide_ListSecrets + } + + { + #region Snippet:Microsoft_Azure_KeyVault_Snippets_MigrationGuide_DeleteSecret + // Delete the secret. + DeletedSecretBundle deletedSecret = await client.DeleteSecretAsync("https://myvault.vault.azure.net", "secret-name"); + + // Purge or recover the deleted secret if soft delete is enabled. + if (deletedSecret.RecoveryId != null) + { + DeletedSecretIdentifier deletedSecretId = deletedSecret.RecoveryIdentifier; + + // Deleting a secret does not happen immediately. Wait a while and check if the deleted secret exists. + while (true) + { + try + { + await client.GetDeletedSecretAsync(deletedSecretId.Vault, deletedSecretId.Name); + + // Finally deleted. + break; + } + catch (KeyVaultErrorException ex) when (ex.Response.StatusCode == HttpStatusCode.NotFound) + { + // Not yet deleted... + } + } + + // Purge the deleted secret. + await client.PurgeDeletedSecretAsync(deletedSecretId.Vault, deletedSecretId.Name); + + // You can also recover the deleted secret using RecoverDeletedSecretAsync. + } + #endregion Snippet:Microsoft_Azure_KeyVault_Snippets_MigrationGuide_DeleteSecret + } + } + } +}