AspNetCore.Docs/aspnetcore/security/data-protection/implementation/key-encryption-at-rest.md

129 lines
6.5 KiB
Markdown
Raw Normal View History

---
title: Key encryption at rest in Windows and Azure using ASP.NET Core
2016-10-29 01:35:15 +08:00
author: rick-anderson
description: Learn implementation details of ASP.NET Core Data Protection key encryption at rest.
2018-01-29 23:21:31 +08:00
ms.author: riande
2018-07-24 00:54:35 +08:00
ms.date: 07/16/2018
no-loc: [Home, Privacy, Kestrel, appsettings.json, "ASP.NET Core Identity", cookie, Cookie, Blazor, "Blazor Server", "Blazor WebAssembly", "Identity", "Let's Encrypt", Razor, SignalR]
uid: security/data-protection/implementation/key-encryption-at-rest
2016-10-29 01:35:15 +08:00
---
# Key encryption at rest in Windows and Azure using ASP.NET Core
2016-10-29 01:35:15 +08:00
2018-07-24 00:54:35 +08:00
The data protection system [employs a discovery mechanism by default](xref:security/data-protection/configuration/default-settings) to determine how cryptographic keys should be encrypted at rest. The developer can override the discovery mechanism and manually specify how keys should be encrypted at rest.
2016-10-29 01:35:15 +08:00
2018-07-24 00:54:35 +08:00
> [!WARNING]
> If you specify an explicit [key persistence location](xref:security/data-protection/implementation/key-storage-providers), the data protection system deregisters the default key encryption at rest mechanism. Consequently, keys are no longer encrypted at rest. We recommend that you [specify an explicit key encryption mechanism](xref:security/data-protection/implementation/key-encryption-at-rest) for production deployments. The encryption-at-rest mechanism options are described in this topic.
2016-10-29 01:35:15 +08:00
2018-07-24 00:54:35 +08:00
::: moniker range=">= aspnetcore-2.1"
2016-10-29 01:35:15 +08:00
2018-07-24 00:54:35 +08:00
## Azure Key Vault
2016-10-29 01:35:15 +08:00
2018-07-24 00:54:35 +08:00
To store keys in [Azure Key Vault](https://azure.microsoft.com/services/key-vault/), configure the system with [ProtectKeysWithAzureKeyVault](/dotnet/api/microsoft.aspnetcore.dataprotection.azuredataprotectionbuilderextensions.protectkeyswithazurekeyvault) in the `Startup` class:
```csharp
public void ConfigureServices(IServiceCollection services)
{
services.AddDataProtection()
.PersistKeysToAzureBlobStorage(new Uri("<blobUriWithSasToken>"))
.ProtectKeysWithAzureKeyVault("<keyIdentifier>", "<clientId>", "<clientSecret>");
}
```
For more information, see [Configure ASP.NET Core Data Protection: ProtectKeysWithAzureKeyVault](xref:security/data-protection/configuration/overview#protectkeyswithazurekeyvault).
::: moniker-end
2016-10-29 01:35:15 +08:00
## Windows DPAPI
2018-07-24 00:54:35 +08:00
**Only applies to Windows deployments.**
2016-10-29 01:35:15 +08:00
2018-07-24 00:54:35 +08:00
When Windows DPAPI is used, key material is encrypted with [CryptProtectData](/windows/desktop/api/dpapi/nf-dpapi-cryptprotectdata) before being persisted to storage. DPAPI is an appropriate encryption mechanism for data that's never read outside of the current machine (though it's possible to back these keys up to Active Directory; see [DPAPI and Roaming Profiles](https://support.microsoft.com/kb/309408/#6)). To configure DPAPI key-at-rest encryption, call one of the [ProtectKeysWithDpapi](/dotnet/api/microsoft.aspnetcore.dataprotection.dataprotectionbuilderextensions.protectkeyswithdpapi) extension methods:
2016-10-29 01:35:15 +08:00
2016-11-18 13:03:07 +08:00
```csharp
2018-07-24 00:54:35 +08:00
public void ConfigureServices(IServiceCollection services)
{
// Only the local user account can decrypt the keys
services.AddDataProtection()
.ProtectKeysWithDpapi();
}
```
2016-10-29 01:35:15 +08:00
2018-07-24 00:54:35 +08:00
If `ProtectKeysWithDpapi` is called with no parameters, only the current Windows user account can decipher the persisted key ring. You can optionally specify that any user account on the machine (not just the current user account) be able to decipher the key ring:
2016-10-29 01:35:15 +08:00
2016-11-18 13:03:07 +08:00
```csharp
2018-07-24 00:54:35 +08:00
public void ConfigureServices(IServiceCollection services)
{
// All user accounts on the machine can decrypt the keys
services.AddDataProtection()
.ProtectKeysWithDpapi(protectToLocalMachine: true);
}
```
2016-10-29 01:35:15 +08:00
2018-07-24 00:54:35 +08:00
::: moniker range=">= aspnetcore-2.0"
2016-10-29 01:35:15 +08:00
2018-07-24 00:54:35 +08:00
## X.509 certificate
2016-10-29 01:35:15 +08:00
2018-07-24 00:54:35 +08:00
If the app is spread across multiple machines, it may be convenient to distribute a shared X.509 certificate across the machines and configure the hosted apps to use the certificate for encryption of keys at rest:
2016-10-29 01:35:15 +08:00
2016-11-18 13:03:07 +08:00
```csharp
2018-07-24 00:54:35 +08:00
public void ConfigureServices(IServiceCollection services)
{
services.AddDataProtection()
.ProtectKeysWithCertificate("3BCE558E2AD3E0E34A7743EAB5AEA2A9BD2575A0");
}
```
2016-10-29 01:35:15 +08:00
2018-07-24 00:54:35 +08:00
Due to .NET Framework limitations, only certificates with CAPI private keys are supported. See the content below for possible workarounds to these limitations.
2016-10-29 01:35:15 +08:00
2018-07-24 00:54:35 +08:00
::: moniker-end
2016-10-29 01:35:15 +08:00
## Windows DPAPI-NG
2018-07-24 00:54:35 +08:00
**This mechanism is available only on Windows 8/Windows Server 2012 or later.**
2016-10-29 01:35:15 +08:00
2018-07-24 00:54:35 +08:00
Beginning with Windows 8, Windows OS supports DPAPI-NG (also called CNG DPAPI). For more information, see [About CNG DPAPI](/windows/desktop/SecCNG/cng-dpapi).
2016-10-29 01:35:15 +08:00
2018-07-24 00:54:35 +08:00
The principal is encoded as a protection descriptor rule. In the following example that calls [ProtectKeysWithDpapiNG](/dotnet/api/microsoft.aspnetcore.dataprotection.dataprotectionbuilderextensions.protectkeyswithdpaping), only the domain-joined user with the specified SID can decrypt the key ring:
2016-10-29 01:35:15 +08:00
2016-11-18 13:03:07 +08:00
```csharp
2018-07-24 00:54:35 +08:00
public void ConfigureServices(IServiceCollection services)
{
// Uses the descriptor rule "SID=S-1-5-21-..."
services.AddDataProtection()
.ProtectKeysWithDpapiNG("SID=S-1-5-21-...",
flags: DpapiNGProtectionDescriptorFlags.None);
}
```
2016-10-29 01:35:15 +08:00
2018-07-24 00:54:35 +08:00
There's also a parameterless overload of `ProtectKeysWithDpapiNG`. Use this convenience method to specify the rule "SID={CURRENT_ACCOUNT_SID}", where *CURRENT_ACCOUNT_SID* is the SID of the current Windows user account:
2016-10-29 01:35:15 +08:00
2016-11-18 13:03:07 +08:00
```csharp
2018-07-24 00:54:35 +08:00
public void ConfigureServices(IServiceCollection services)
{
// Use the descriptor rule "SID={current account SID}"
services.AddDataProtection()
.ProtectKeysWithDpapiNG();
}
```
2016-10-29 01:35:15 +08:00
2018-07-24 00:54:35 +08:00
In this scenario, the AD domain controller is responsible for distributing the encryption keys used by the DPAPI-NG operations. The target user can decipher the encrypted payload from any domain-joined machine (provided that the process is running under their identity).
2016-10-29 01:35:15 +08:00
## Certificate-based encryption with Windows DPAPI-NG
2018-07-24 00:54:35 +08:00
If the app is running on Windows 8.1/Windows Server 2012 R2 or later, you can use Windows DPAPI-NG to perform certificate-based encryption. Use the rule descriptor string "CERTIFICATE=HashId:THUMBPRINT", where *THUMBPRINT* is the hex-encoded SHA1 thumbprint of the certificate:
2016-10-29 01:35:15 +08:00
2016-11-18 13:03:07 +08:00
```csharp
2018-07-24 00:54:35 +08:00
public void ConfigureServices(IServiceCollection services)
{
services.AddDataProtection()
.ProtectKeysWithDpapiNG("CERTIFICATE=HashId:3BCE558E2...B5AEA2A9BD2575A0",
flags: DpapiNGProtectionDescriptorFlags.None);
}
```
2016-10-29 01:35:15 +08:00
2018-07-24 00:54:35 +08:00
Any app pointed at this repository must be running on Windows 8.1/Windows Server 2012 R2 or later to decipher the keys.
2016-10-29 01:35:15 +08:00
## Custom key encryption
2018-07-24 00:54:35 +08:00
If the in-box mechanisms aren't appropriate, the developer can specify their own key encryption mechanism by providing a custom [IXmlEncryptor](/dotnet/api/microsoft.aspnetcore.dataprotection.xmlencryption.ixmlencryptor).