1
我一直在努力研究C#的RSACryptoServiceProvider如何簽署數據,或找到任何有關如何簽署數據的文檔。手動簽名RSASSA-PKCS1-V1_5-SIGN與C#RSACryptoServiceProvider?
我試過按照指定的簽名方案here,但我似乎無法生成符合.NET實現的簽名。
RSACryptoServiceProvider如何生成一個簽名,或者爲什麼我的任何manualSignedDatas都不匹配下面示例代碼中的RSACryptoServiceProviderSignedData?
public void ManualSignatureTest(byte[] data, X509Certificate2 cert)
{
var sha256 = new SHA256CryptoServiceProvider();
var rsa = (RSACryptoServiceProvider)cert.PrivateKey;
var RSACryptoServiceProviderSignedData = rsa.SignData(data, "SHA256");
var manualSignedDataLittle = ManualSignDataSha256(data, rsa, true, false);
var manualSignedDataBig = ManualSignDataSha256(data, rsa, false, false);
//https://msdn.microsoft.com/en-us/library/system.security.cryptography.rsacryptoserviceprovider(v=vs.110).aspx
//the RSACryptoServiceProvider class reverses the order of an encrypted array of bytes after encryption and before decryption
var manualSignedDataReversedLittle = ManualSignDataSha256(data, rsa, true, true);
var manualSignedDataReversedBig = ManualSignDataSha256(data, rsa, false, true);
}
public byte[] ManualSignDataSha256(byte[] data, RSACryptoServiceProvider rsa, bool littleEndian, bool reverseArray)
{
//https://tools.ietf.org/html/rfc8017#section-8.2.1
//EM = EMSA-PKCS1-V1_5-ENCODE (M, k)
var emsaEncodedMessage = EmsaPkcs1V1_5Encode_SHA256(data, 245);
if (reverseArray) Array.Reverse(emsaEncodedMessage);
//m = OS2IP (EM)
var intMessageRepresentative = OS2IP(emsaEncodedMessage, littleEndian);
//s = RSASP1 (K, m)
var intSignatureRepresentative = RSASP1(rsa, intMessageRepresentative);
//S = I2OSP (s, k)
var signature = I2OSP(intSignatureRepresentative, 256, littleEndian);
return signature;
}
public byte[] HexToByte(string hex)
{
return Enumerable.Range(0, hex.Length/2).Select(x => Convert.ToByte(hex.Substring(x * 2, 2), 16)).ToArray();
}
public byte[] EmsaPkcs1V1_5Encode_SHA256(byte[] message, int emLen)
{
var digest = new SHA256CryptoServiceProvider().ComputeHash(message);
var digestAlgorithm = HexToByte("3031300d060960864801650304020105000420");
var digestInfo = digest.Concat(digestAlgorithm).ToArray();
var paddingLength = emLen - digestInfo.Length - 3;
var paddingHexString = "";
for (int i = 0; i < paddingLength; i++)
{
paddingHexString += "FF";
}
return HexToByte("0001" + paddingHexString + "00").Concat(digestInfo).ToArray();
}
public BigInteger OS2IP(byte[] data, bool isLittleEndian)
{
BigInteger bi = 0;
if (isLittleEndian)
{
for (int i = 0; i < data.Length; i++)
{
bi += BigInteger.Pow(256, i) * data[i];
}
}
else
{
for (int i = 1; i <= data.Length; i++)
{
bi += BigInteger.Pow(256, i - 1) * data[data.Length - i];
}
}
return bi;
}
public BigInteger RSASP1(RSACryptoServiceProvider rsa, BigInteger message)
{
var keyParameters = rsa.ExportParameters(true);
var n = new BigInteger(keyParameters.Modulus);
var d = new BigInteger(keyParameters.D);
return BigInteger.ModPow(message, d, n);
return default(BigInteger);
}
public byte[] I2OSP(BigInteger x, int xLen, bool makeLittleEndian)
{
byte[] result = new byte[xLen];
int index = 0;
while ((x > 0) && (index < result.Length))
{
result[index++] = (byte)(x % 256);
x /= 256;
}
if (!makeLittleEndian)
Array.Reverse(result);
return result;
}
C#調用在哪裏?你沒有把它與PSS錯誤地進行比較?我沒有看到任何示例輸入,輸出或您嘗試過的東西(例如以十六進制打印中間值並驗證它們)。 –