diff --git a/src/libraries/Common/src/Microsoft/Win32/SafeHandles/SafeCertContextHandleWithKeyContainerDeletion.cs b/src/libraries/Common/src/Microsoft/Win32/SafeHandles/SafeCertContextHandleWithKeyContainerDeletion.cs index 7488f624b90c4..59a84bc923097 100644 --- a/src/libraries/Common/src/Microsoft/Win32/SafeHandles/SafeCertContextHandleWithKeyContainerDeletion.cs +++ b/src/libraries/Common/src/Microsoft/Win32/SafeHandles/SafeCertContextHandleWithKeyContainerDeletion.cs @@ -50,10 +50,16 @@ internal static void DeleteKeyContainer(SafeCertContextHandle pCertContext) string providerName = Marshal.PtrToStringUni((IntPtr)(pProvInfo->pwszProvName))!; string keyContainerName = Marshal.PtrToStringUni((IntPtr)(pProvInfo->pwszContainerName))!; + CngKeyOpenOptions openOpts = CngKeyOpenOptions.None; + + if ((pProvInfo->dwFlags & Interop.Crypt32.CryptAcquireContextFlags.CRYPT_MACHINE_KEYSET) != 0) + { + openOpts = CngKeyOpenOptions.MachineKey; + } try { - using (CngKey cngKey = CngKey.Open(keyContainerName, new CngProvider(providerName))) + using (CngKey cngKey = CngKey.Open(keyContainerName, new CngProvider(providerName), openOpts)) { cngKey.Delete(); } diff --git a/src/libraries/System.Security.Cryptography/tests/System.Security.Cryptography.Tests.csproj b/src/libraries/System.Security.Cryptography/tests/System.Security.Cryptography.Tests.csproj index 89ea24dd50327..688d6b1f8c917 100644 --- a/src/libraries/System.Security.Cryptography/tests/System.Security.Cryptography.Tests.csproj +++ b/src/libraries/System.Security.Cryptography/tests/System.Security.Cryptography.Tests.csproj @@ -495,7 +495,8 @@ Link="Common\Interop\Windows\Crypt32\Interop.MsgEncodingType.cs" /> - + + diff --git a/src/libraries/System.Security.Cryptography/tests/X509Certificates/X509FilesystemTests.Windows.cs b/src/libraries/System.Security.Cryptography/tests/X509Certificates/X509FilesystemTests.Windows.cs new file mode 100644 index 0000000000000..8c8c7a42ccc38 --- /dev/null +++ b/src/libraries/System.Security.Cryptography/tests/X509Certificates/X509FilesystemTests.Windows.cs @@ -0,0 +1,393 @@ +// Licensed to the .NET Foundation under one or more agreements. +// The .NET Foundation licenses this file to you under the MIT license. + +using System.Collections.Generic; +using System.IO; +using System.Linq; +using System.Runtime.CompilerServices; +using System.Security.Cryptography.Pkcs; +using System.Security.Principal; +using System.Threading; +using Test.Cryptography; +using Xunit; + +namespace System.Security.Cryptography.X509Certificates.Tests +{ + [Collection("X509Filesystem")] + public static class X509FilesystemTests + { + // Microsoft Strong Cryptographic Provider + private static readonly AsnEncodedData s_capiCsp = new AsnEncodedData( + new Oid("1.3.6.1.4.1.311.17.1", null), + ( + "1E4E004D006900630072006F0073006F006600740020005300740072006F006E" + + "0067002000430072007900700074006F00670072006100700068006900630020" + + "00500072006F00760069006400650072" + ).HexToByteArray()); + + private static readonly AsnEncodedData s_machineKey = new AsnEncodedData( + new Oid("1.3.6.1.4.1.311.17.2", null), + [0x05, 0x00]); + + // 6 random keys that will used across all of the tests in this file + private const int KeyGenKeySize = 2048; + private static readonly RSA[] s_keys = + { + RSA.Create(KeyGenKeySize), RSA.Create(KeyGenKeySize), RSA.Create(KeyGenKeySize), + RSA.Create(KeyGenKeySize), RSA.Create(KeyGenKeySize), RSA.Create(KeyGenKeySize), + }; + + [Theory] + [InlineData(X509KeyStorageFlags.DefaultKeySet)] + [InlineData(X509KeyStorageFlags.DefaultKeySet, true)] + [InlineData(X509KeyStorageFlags.UserKeySet)] + [InlineData(X509KeyStorageFlags.UserKeySet, true)] + [InlineData(X509KeyStorageFlags.MachineKeySet)] + [InlineData(X509KeyStorageFlags.MachineKeySet, true)] + public static void AllFilesDeleted_MultiplePrivateKey_Ctor(X509KeyStorageFlags storageFlags, bool capi = false) + { + AllFilesDeletedTest( + storageFlags, + capi, + multiPrivate: true, + static (bytes, pwd, flags) => new X509Certificate2(bytes, pwd, flags)); + } + + [Theory] + [InlineData(X509KeyStorageFlags.DefaultKeySet)] + [InlineData(X509KeyStorageFlags.DefaultKeySet, true)] + [InlineData(X509KeyStorageFlags.UserKeySet)] + [InlineData(X509KeyStorageFlags.UserKeySet, true)] + [InlineData(X509KeyStorageFlags.MachineKeySet)] + [InlineData(X509KeyStorageFlags.MachineKeySet, true)] + public static void AllFilesDeleted_SinglePrivateKey_Ctor(X509KeyStorageFlags storageFlags, bool capi = false) + { + AllFilesDeletedTest( + storageFlags, + capi, + multiPrivate: false, + static (bytes, pwd, flags) => new X509Certificate2(bytes, pwd, flags)); + } + + [Theory] + [InlineData(X509KeyStorageFlags.DefaultKeySet)] + [InlineData(X509KeyStorageFlags.DefaultKeySet, true)] + [InlineData(X509KeyStorageFlags.UserKeySet)] + [InlineData(X509KeyStorageFlags.UserKeySet, true)] + [InlineData(X509KeyStorageFlags.MachineKeySet)] + [InlineData(X509KeyStorageFlags.MachineKeySet, true)] + public static void AllFilesDeleted_MultiplePrivateKey_CollImport(X509KeyStorageFlags storageFlags, bool capi = false) + { + AllFilesDeletedTest( + storageFlags, + capi, + multiPrivate: true, + Cert.Import); + } + + [Theory] + [InlineData(X509KeyStorageFlags.DefaultKeySet)] + [InlineData(X509KeyStorageFlags.DefaultKeySet, true)] + [InlineData(X509KeyStorageFlags.UserKeySet)] + [InlineData(X509KeyStorageFlags.UserKeySet, true)] + [InlineData(X509KeyStorageFlags.MachineKeySet)] + [InlineData(X509KeyStorageFlags.MachineKeySet, true)] + public static void AllFilesDeleted_SinglePrivateKey_CollImport(X509KeyStorageFlags storageFlags, bool capi = false) + { + AllFilesDeletedTest( + storageFlags, + capi, + multiPrivate: false, + Cert.Import); + } + + private static void AllFilesDeletedTest( + X509KeyStorageFlags storageFlags, + bool capi, + bool multiPrivate, + Func importer, + [CallerMemberName] string? name = null) + { + const X509KeyStorageFlags NonDefaultKeySet = + X509KeyStorageFlags.UserKeySet | + X509KeyStorageFlags.MachineKeySet; + + bool defaultKeySet = (storageFlags & NonDefaultKeySet) == 0; + int certAndKeyCount = multiPrivate ? s_keys.Length : 1; + + byte[] pfx = MakePfx(certAndKeyCount, capi, name); + + EnsureNoKeysGained( + (Bytes: pfx, Flags: storageFlags, Importer: importer), + static state => state.Importer(state.Bytes, "", state.Flags)); + + // When importing for DefaultKeySet, try both 010101 and 101010 + // intermixing of machine and user keys so that single key import + // gets both a machine key and a user key. + if (defaultKeySet) + { + pfx = MakePfx(certAndKeyCount, capi, name, 1); + + EnsureNoKeysGained( + (Bytes: pfx, Flags: storageFlags, Importer: importer), + static state => state.Importer(state.Bytes, "", state.Flags)); + } + } + + private static byte[] MakePfx( + int certAndKeyCount, + bool capi, + [CallerMemberName] string? name = null, + int machineKeySkew = 0) + { + Pkcs12SafeContents keys = new Pkcs12SafeContents(); + Pkcs12SafeContents certs = new Pkcs12SafeContents(); + DateTimeOffset notBefore = DateTimeOffset.UtcNow.AddMinutes(-5); + DateTimeOffset notAfter = notBefore.AddMinutes(10); + + PbeParameters pbeParams = new PbeParameters( + PbeEncryptionAlgorithm.TripleDes3KeyPkcs12, + HashAlgorithmName.SHA1, + 1); + + Span indices = [0, 1, 2, 3, 4, 5]; + RandomNumberGenerator.Shuffle(indices); + + for (int i = 0; i < s_keys.Length; i++) + { + RSA key = s_keys[indices[i]]; + + CertificateRequest req = new CertificateRequest( + $"CN={name}.{i}", + key, + HashAlgorithmName.SHA256, + RSASignaturePadding.Pkcs1); + + using (X509Certificate2 cert = req.CreateSelfSigned(notBefore, notAfter)) + { + Pkcs12CertBag certBag = certs.AddCertificate(cert); + + if (i < certAndKeyCount) + { + Pkcs12ShroudedKeyBag keyBag = keys.AddShroudedKey(key, "", pbeParams); + + if (capi) + { + keyBag.Attributes.Add(s_capiCsp); + } + + if (int.IsEvenInteger(i + machineKeySkew)) + { + keyBag.Attributes.Add(s_machineKey); + } + + byte keyId = checked((byte)i); + Pkcs9LocalKeyId localKeyId = new Pkcs9LocalKeyId(new ReadOnlySpan(ref keyId)); + keyBag.Attributes.Add(localKeyId); + certBag.Attributes.Add(localKeyId); + } + } + } + + Pkcs12Builder builder = new Pkcs12Builder(); + builder.AddSafeContentsEncrypted(certs, "", pbeParams); + builder.AddSafeContentsUnencrypted(keys); + builder.SealWithMac("", HashAlgorithmName.SHA1, 1); + return builder.Encode(); + } + + private static void EnsureNoKeysGained(TState state, Func importer) + { + const int ERROR_ACCESS_DENIED = (unchecked((int)0x80010005)); + + // In the good old days, before we had threads or parallel processes, these tests would be easy: + // * Read the directory listing(s) + // * Import a thing + // * See what new things were added + // * Dispose the thing + // * See that the new things went away + // + // But, since files can be created by tests on other threads, or even by other processes, + // recheck the directory a few times (MicroRetryCount) after sleeping (SleepMs). + // + // Sadly, that's not sufficient, because an extra file gained during that window could itself + // be leaked, or be intentionally persisted beyond the recheck interval. So, instead of failing, + // try again from the beginning. If we get parallel leaked on MacroRetryCount times in a row + // we'll still false-fail, but unless a majority of the tests in the process are leaking keys, + // it's unlikely. + // + // Before changing these constants to bigger numbers, consider the combinatorics. Failure will + // sleep (MacroRetryCount * (MicroRetryCount - 1) * SleepMs) ms, and also involves non-zero work. + // Failing 29 tests at (3, 5, 1000) adds about 6 minutes to the test run compared to success. + + const int MacroRetryCount = 3; + const int MicroRetryCount = 5; + const int SleepMs = 1000; + + KeyPaths keyPaths = KeyPaths.GetKeyPaths(); + HashSet gainedFiles = null; + + for (int macro = 0; macro < MacroRetryCount; macro++) + { + List keysBefore = new(keyPaths.EnumerateAllKeys()); + + IDisposable imported = null; + + try + { + imported = importer(state); + } + catch (CryptographicException ex) when (ex.HResult == ERROR_ACCESS_DENIED) + { + } + + imported?.Dispose(); + + gainedFiles = new HashSet(keyPaths.EnumerateAllKeys()); + gainedFiles.ExceptWith(keysBefore); + + for (int micro = 0; micro < MicroRetryCount; micro++) + { + if (gainedFiles.Count == 0) + { + return; + } + + HashSet thisTry = new(keyPaths.EnumerateAllKeys()); + gainedFiles.IntersectWith(thisTry); + + if (gainedFiles.Count != 0 && micro < MicroRetryCount - 1) + { + Thread.Sleep(SleepMs); + } + } + } + + Assert.Empty(keyPaths.MapPaths(gainedFiles)); + } + + private sealed class KeyPaths + { + private static volatile KeyPaths s_instance; + + private string _capiUserDsa; + private string _capiUserRsa; + private string _capiMachineDsa; + private string _capiMachineRsa; + private string _cngUser; + private string _cngMachine; + + private KeyPaths() + { + } + + internal IEnumerable MapPaths(IEnumerable paths) + { + foreach (string path in paths) + { + yield return + Replace(path, _cngUser, "CNG-USER") ?? + Replace(path, _capiUserRsa, "CAPI-USER-RSA") ?? + Replace(path, _cngMachine, "CNG-MACH") ?? + Replace(path, _capiMachineRsa, "CAPI-MACH-RSA") ?? + Replace(path, _capiUserDsa, "CAPI-USER-DSS") ?? + Replace(path, _capiMachineDsa, "CAPI-MACH-DSS") ?? + path; + } + + static string Replace(string path, string prefix, string ifMatched) + { + if (path.StartsWith(prefix)) + { + return path.Replace(prefix, ifMatched); + } + + return null; + } + } + + internal IEnumerable EnumerateCapiUserKeys() + { + return EnumerateFiles(_capiUserRsa).Concat(EnumerateFiles(_capiUserDsa)); + } + + internal IEnumerable EnumerateCapiMachineKeys() + { + return EnumerateFiles(_capiMachineRsa).Concat(EnumerateFiles(_capiMachineDsa)); + } + + internal IEnumerable EnumerateCngUserKeys() + { + return EnumerateFiles(_cngUser); + } + + internal IEnumerable EnumerateCngMachineKeys() + { + return EnumerateFiles(_cngMachine); + } + + internal IEnumerable EnumerateUserKeys() + { + return EnumerateCapiUserKeys().Concat(EnumerateCngUserKeys()); + } + + internal IEnumerable EnumerateMachineKeys() + { + return EnumerateCapiMachineKeys().Concat(EnumerateCngMachineKeys()); + } + + internal IEnumerable EnumerateAllKeys() + { + return EnumerateUserKeys().Concat(EnumerateMachineKeys()); + } + + private static IEnumerable EnumerateFiles(string directory) + { + try + { + return Directory.EnumerateFiles(directory); + } + catch (DirectoryNotFoundException) + { + } + + return []; + } + + internal static KeyPaths GetKeyPaths() + { + if (s_instance is not null) + { + return s_instance; + } + + // https://learn.microsoft.com/en-us/windows/win32/seccng/key-storage-and-retrieval + WindowsIdentity identity = WindowsIdentity.GetCurrent(); + string userSid = identity.User!.ToString(); + + string userKeyBase = Path.Join( + Environment.GetFolderPath(Environment.SpecialFolder.ApplicationData), + "Microsoft", + "Crypto"); + + string machineKeyBase = Path.Join( + Environment.GetFolderPath(Environment.SpecialFolder.CommonApplicationData), + "Microsoft", + "Crypto"); + + KeyPaths paths = new() + { + _capiUserDsa = Path.Join(userKeyBase, "DSS", userSid), + _capiUserRsa = Path.Join(userKeyBase, "RSA", userSid), + _capiMachineDsa = Path.Join(machineKeyBase, "DSS", "MachineKeys"), + _capiMachineRsa = Path.Join(machineKeyBase, "RSA", "MachineKeys"), + _cngUser = Path.Join(userKeyBase, "Keys"), + _cngMachine = Path.Join(machineKeyBase, "Keys"), + }; + + s_instance = paths; + return s_instance; + } + } + } +}