Файл подписи ECDSA с ключом из хранилища C#.Net CNG
Я пытаюсь подписать файл с ECDSA, используя API CNG и сертификат из хранилища сертификатов Microsoft. Я прочитал много документации и почти готов, но я зациклен на импорте закрытого ключа из сертификата. Я сделал то же самое с RSA, но, похоже, все по-другому. Вот код, который у меня есть:
static void signFile()
{
X509Certificate2 myCert =
selectCert(StoreName.My,
StoreLocation.CurrentUser,
"Select a Certificate",
"Please select a certificate from the list below:");
Console.Write("Path for file to sign: ");
string path = Console.ReadLine();
TextReader file = null;
try
{
file = new StreamReader(path);
}
catch (Exception e)
{
Console.WriteLine(e.Message);
Console.Write("\nPress any key to return to the main menu: ");
Console.ReadKey();
}
UnicodeEncoding encoding = new UnicodeEncoding();
byte[] data = encoding.GetBytes(file.ReadToEnd());
ECDsaCng dsa = new ECDsaCng(
CngKey.Import(StringToByteArray(myCert.PrivateKey.ToString()),
CngKeyBlobFormat.EccPrivateBlob,
CngProvider.MicrosoftSoftwareKeyStorageProvider));
dsa.HashAlgorithm = CngAlgorithm.Sha384;
byte[] sig = dsa.SignData(data);
TextWriter signatureFile = new StreamWriter("signature.txt");
signatureFile.WriteLine("-----BEGIN SHA384 SIGNATURE-----" +
ByteArrayToString(sig) +
"-----END SHA384 SIGNATURE-----");
signatureFile.Close();
}
И я получаю ошибку
System.NotSupportedException: алгоритм ключа сертификата не поддерживается.
Мой сертификат ECDSA_P256 sha384ECDSA со следующими расширениями:
Digital Signature, Non-repudiation, independent signing revocation list (CRL), CRL Signing (CRL) (c2)
Server Authentication (1.3.6.1.5.5.7.3.1)
Client Authentication (1.3.6.1.5.5.7.3.2)
Code Signing (1.3.6.1.5.5.7.3.3)
Unknown Key Usage (1.3.6.1.4.1.311.2.1.22)
Unknown Key Usage (1.3.6.1.4.1.311.2.1.21)
IKE-intermediary IP-security (1.3.6.1.5.5.8.2.2)
Казалось бы, если сертификат был проблемой, но я не уверен, что это может быть код или нет.
Вот мой сертификат с открытым ключом:
Certificate:
Data:
Version: 3 (0x2)
Serial Number: 2 (0x2)
Signature Algorithm: ecdsa-with-SHA384
Issuer: C=##, O=#######, OU=#####, OU=#####, CN=###########
Validity
Not Before: Apr 27 16:35:51 2012 GMT
Not After : Apr 26 16:35:51 2017 GMT
Subject: C=##, O=###########, OU=#####, CN=#############
Subject Public Key Info:
Public Key Algorithm: id-ecPublicKey
Public-Key: (256 bit)
pub:
04:fc:d5:ce:ad:1f:0c:19:b9:3d:2b:bd:7d:f0:8c:
44:46:db:e3:42:14:b1:1a:9f:7c:ab:e1:be:ad:a5:
0c:03:2d:0f:ff:3f:10:d4:69:eb:4c:82:a1:2a:61:
56:45:03:04:a6:49:f7:16:6e:dd:60:22:c6:20:c5:
4d:44:49:21:41
ASN1 OID: prime256v1
X509v3 extensions:
X509v3 Key Usage: critical
Digital Signature, Non Repudiation, CRL Sign
X509v3 Extended Key Usage: critical
TLS Web Server Authentication, TLS Web Client Authentication, Co
de Signing, Microsoft Commercial Code Signing, Microsoft Individual Code Signing
, 1.3.6.1.5.5.8.2.2
X509v3 Authority Key Identifier:
DirName:/C=##/O=#######/OU=#####/OU=#####/CN=######
serial:01
X509v3 Subject Key Identifier:
B7:A8:F9:55:9A:43:9E:BE:1C:4B:62:52:91:C2:F1:39:72:E1:CE:1B
X509v3 Basic Constraints: critical
CA:FALSE
Signature Algorithm: ecdsa-with-SHA384
30:81:88:02:42:01:75:55:f3:64:f9:aa:2a:66:55:b1:ca:dc:
86:ac:1f:7d:2a:ec:10:87:db:74:88:0e:77:e3:18:82:15:a7:
32:91:1a:2d:ea:07:2e:78:8d:dc:8a:18:3c:2b:5a:9b:6a:0f:
97:f6:f8:8d:c5:fc:0e:9f:20:e9:b0:16:90:1a:c4:58:ac:02:
42:01:dc:b3:88:ae:44:54:c4:e0:b7:c2:37:88:0b:19:6b:96:
99:f7:21:12:45:12:21:e5:ab:83:39:a6:47:3a:08:87:b0:fa:
0e:31:1b:97:83:8d:65:30:a1:43:c1:82:27:77:6e:93:89:1b:
bd:57:b1:7a:54:9e:cc:e1:44:cc:74:16:c5
3 ответа
.NET 4.6.1 решил основные потребности этой проблемы. Новый код будет
...
byte[] sig;
using (ECDsa ecdsa = cert.GetECDsaPrivateKey())
{
if (ecdsa == null) throw new Exception("Not an ECDSA cert, or has no private key");
sig = ecdsa.SignData(data, HashAlgorithmName.SHA384);
}
В.NET 4.6.1 также исправлена ошибка, из-за которой некоторые ключи сертификатов возвращались как ECDH и, таким образом, не выполнялись. (Что ж, это не решило проблему некоторых закрытых ключей, считающихся ECDH - что не имело ничего общего с EKU аутентификации сервера, но было хорошим предположением - но теперь считает эти ключи действительными).
Я долгое время боролся с ECDsa и CngKey с помощью сертификатов X509, и у меня были точно такие же проблемы. Мы закончили тем, что создали наши собственные CngKeys с ECDsa_P256 SHA256, но я верю, что я узнал кое-что, копаясь в CryptoApi:
Если у вас есть сертификат, помеченный как "Аутентификация сервера (1.3.6.1.5.5.7.3.1)" (используйте в качестве SSL-сертификата), ваш сертификат будет содержать алгоритм обмена ключами ECDH. И каким-то образом это "имеет приоритет" над группой алгоритмов ECDsa. Таким образом, вы получите страшное "Алгоритм ключа сертификата не поддерживается".
Я провел более часа с Symantec, оглядываясь через плечо, и они не могли решить головоломку, поэтому они отказались от "Извините, мы не поддерживаем использование SSL-сертификатов ни для чего, кроме SSL".
Вы можете получить свой личный CngKey из сертификата с CLRSecurity от Codeplex ( http://clrsecurity.codeplex.com/). Это дает вашему x509Certificate2 расширение, которое учитывает этот код:
X509Certificate cer = <getyourcertcode>;
CngKey k = cer.GetCngPrivateKey();
Изучите "k" и убедитесь, что ваша группа алгоритмов, вероятно, отличается от ожидаемой. Мой был ECDH...
Решение, которое я пытаюсь сейчас, - это настроить новый сервер CA, заставить его делать именно то, что я хочу. По сути, это сертификат X509, который используется ТОЛЬКО для подписи кода...
"Использование ключа X509v3: критически важная цифровая подпись" может быть ЕДИНСТВЕННЫМ разрешенным использованием...
Надеюсь, это поможет кому-то еще:-)
Если вы работаете в Windows Vista или Windows 2008, CngKeyBlobFormat.EccPrivateBlob не поддерживается. Какую ОС вы используете? CngKey.Import генерирует CryptographicException только на некоторых машинах