我是新来的。
我正在学习C#中的数字签名。生成证书,后跟this document。我读过的其他文件:RSACng, X509Certificate2。
我正在使用Windows 10 Pro 1809,.Net Core 2.1,VSCode。
class Program
{
static void Main(string[] args)
{
var passwd = "password";
// Get client certificate.
var clientCertPath = @"./Certificates/test.pfx";
var clientCert = new X509Certificate2(clientCertPath, passwd);
// Get server certificate.
var serverCertPath = @"./Certificates/test.cer";
var serverCert = new X509Certificate2(serverCertPath);
// Generate data.
var translateResultData = BuildData();
var content = String.Join('&', translateResultData.Select(p => String.Join('=', p.Key, p.Value)));
// Sign
var sign = SignatureUtil.Sign(data: content, clientCert: clientCert);
// translateResultData.TryAdd(key: "sign", value : sign);
// Copy content ONLY for test.
var checkSign = sign;
var checkContent = content;
// Verify
var valid = SignatureUtil.Verify(data: checkContent, signature: checkSign, serverCert: serverCert);
System.Console.WriteLine(valid);
}
}
public class SignatureUtil
{
public static string Sign(string data, X509Certificate2 clientCert)
{
using(var privateKey = clientCert.GetRSAPrivateKey())
{
var dataByteArray = Encoding.UTF8.GetBytes(data);
var signatureByteArray = privateKey.SignData(
data: dataByteArray,
hashAlgorithm: HashAlgorithmName.SHA256,
padding: RSASignaturePadding.Pkcs1);
return Convert.ToBase64String(signatureByteArray);
}
}
public static bool Verify(string data, string signature, X509Certificate2 serverCert)
{
try
{
using(var publicKey = serverCert.GetRSAPublicKey())
{
var dataByteArray = Encoding.UTF8.GetBytes(data);
var signatureByteArray = Convert.FromBase64String(signature);
return publicKey.VerifyData(
data: dataByteArray,
signature: signatureByteArray,
hashAlgorithm: HashAlgorithmName.SHA256,
padding: RSASignaturePadding.Pkcs1);
}
}
catch (System.Exception)
{
return false;
}
}
}
预期结果:valid
应该为true
,因为我正在检查原始数据。
事实:即使传递原始数据,Verify
方法也始终返回false
。
你能告诉我我做错了吗?
答案 0 :(得分:1)
由于无法重现代码,因此我无法告诉您代码中出了什么问题。 Here是非常详细的答案,您如何使用RSA和SHA256进行签名。您的方法和此答案中描述的方法在概念上是相同的,但与该答案相比,您的代码可能有所不同。
这是一个示例,在我的公司中如何将与智能卡关联的证书用于签名和验证签名。一个很大的不同是我们不将签名存储为字符串,而是将其保存为字节数组。
public byte[] SignData(byte[] data)
{
using (var sha256 = SHA256.Create())
{
using (var rsa = (RSACryptoServiceProvider) Certificate.PrivateKey)
{
return rsa.SignData(data, sha256);
}
}
}
public bool VerifySignature(byte[] data, byte[] signature)
{
using (var sha256 = SHA256.Create())
{
using (var rsa = (RSACryptoServiceProvider) Certificate.PrivateKey)
{
return rsa.VerifyData(data, sha256, signature);
}
}
}