繁体   English   中英

如何在使用GO语言加密时在c#中解密RSA加密字符串。 解码OAEP填充时发生错误

[英]How to decrypt RSA encrypted string in c# while it was encrypted in GO language. Error occurred while decoding OAEP padding

我有一个在c#上运行的应用程序和另一个服务器应用程序。 我需要使用rsa实现安全通信。
我正在做的是我已经在我的C#应用​​程序中初始化了RSA提供程序并生成了公钥来提取模数和指数。 然后连接模数(十六进制)和指数(int)并将此字符串转换为Base64字符串,将其发送到GO端点。

这是C#代码片段

public string ConvertToPublicKey()
    {
        CspParameters rsaParameters = GetCspParameters();
        RSACryptoServiceProvider provider = newRSACryptoServiceProvider(rsaParameters);
        string paramsXml = RsaProvider.ToXmlString(false);

        XDocument xDocument = XDocument.Parse(paramsXml);
        string modulus = xDocument.Descendants().FirstOrDefault(x => x.Name == "Modulus")?.Value ?? string.Empty;
        string exponent = xDocument.Descendants().FirstOrDefault(x => x.Name == "Exponent")?.Value ?? string.Empty;
        byte[] base64BytesOfModulus = Convert.FromBase64String(modulus);
        string hexaDecimalofModulus = BitConverter.ToString(base64BytesOfModulus).Replace("-", string.Empty);
        byte[] base64BytesOfExponent = Convert.FromBase64String(exponent);
        string hexadecimalOfExponent = BitConverter.ToString(base64BytesOfExponent).Replace("-", string.Empty);
        int intOfExponent = Convert.ToInt32(hexadecimalOfExponent, 16);
        byte[] publicKey = Encoding.UTF8.GetBytes($"{hexaDecimalofModulus};{intOfExponent}");
        return Convert.ToBase64String(publicKey);
    }

    private static CspParameters GetCspParameters()
    {
        const string containerName = "KeyContainer";
        return new CspParameters
        {
            KeyContainerName = containerName,
            Flags = CspProviderFlags.UseMachineKeyStore
        };
    }

在Go端点,我正确地收到了公钥和模数。 然后我使用公钥对消息进行加密,并在将加密消息byte []转换为base64后将其发送回c#应用程序作为响应。

这是GO代码片段

func GetLicenseInfo(responseWriter http.ResponseWriter,request*http.Request) 
    {
    encryptionKey := request.Header.Get("Authorization")
    var decodedStringBytes, errors = b64.StdEncoding.DecodeString(encryptionKey)
    if errors == nil {
        var decodedString = string(decodedStringBytes)
        result := strings.Split(decodedString, ";")
        modulus := new(big.Int)
        modulus.SetString(result[0], 16)
        exponent, exponentErrors := strconv.Atoi(result[1])
        if exponentErrors == nil {
            var someInfo = utils.GetInfo()
            var InfoInJson = ToJson(someInfo)
            publicKey := &rsa.PublicKey{N: modulus, E: exponent}
            var encryptedMessage, err = rsa.EncryptOAEP(sha256.New(),rand.Reader, publicKey,[]byte(InfoInJson), []byte(""))
            var response = b64.StdEncoding.EncodeToString(encryptedMessage)
            if err == nil {
                json.NewEncoder(responseWriter).Encode(response)
            }
        }

    }
}

func ToJson(model InfoModel) string {
    InfoInJson, errors := json.Marshal(model)
    if errors != nil {
        panic("An error occurred while serializing the response")
    }
    return string(InfoInJson)
}

当我收到Base64字符串中的响应时,我将其转换为Byte[]并尝试使用相同的RSACryptoServiceProvider实例解密,然后抛出以下异常

解码OAEP填充时发生错误。

任何帮助?

例如, UPDATE是我在GO中收到的base64字符串

QUQ2NDlFRTlCQTA3Q0IxNEI1MTNDMzczQzBBMjNBOEQyMDI5MkVGQTBFMjgyNUIyMEEyMzM1MEE3OTUyNjgyQ0Y3MEFBQjJBMTZGMzQyNTM4MkU2RDZBRjU5M0IxRTI2MTE0OEIyQkFFRTY3MUVDMTQ1NDk1NjBDRkNEQUNCQzI3RUUxNDRFODZDQUI4RDBDOUY2OENBNTUwNUMxQjZGQkVBQjQ0MTlBMjg3RDhBRjgxRDUyREY3MEM0RDZDQTA5MkREMzk5Q0NEODU5Q0FGQzAzQ0JEQ0JBQzgwOTg3NDY0NThBMkY4NEREOTc1QjU5QTJBMUNBNzQxQTBDNkQ2RDs2NTUzNw ==

这是我的GO应用程序发回的内容

QuWpWdEPSJR + l9UJTkh + heJJ / NpPwhz / hVVu1VdKYdz37YGWWdKTj7Fc5lZ3A8p1WjtC4F + yieZCz0tEatCqTpRmm9g6Oioyjbtr9qGTxO / PE + GA33YyBe6nmMRe674SPePx / fg6l3nnfSZ4 / + iLCV4bNgyNqFHCaXc7H4Snms8 =

更新2我已经更新了代码片段并包含了数据类型,这里是从GO端点接收内容的部分

public byte[] Decrypt(byte[] encryptedData, RSAParameters rsaParameters)
    {
        RsaProvider.ImportParameters(rsaParameters);
        return RsaProvider.Decrypt(encryptedData, true);
    }

我收到一个base64字符串,然后我使用它转换为byte []

byte[] b = Convert.FromBase64String(responseString); byte[] decryptedBytes=crypto.Decrypt(b, crypto.RsaProvider.ExportParameters(false));

crypto是包含解密逻辑的类的实例, RSACryptoServiceProvider instacne和上面给出的返回公钥的方法(ConvertToPublicKey)

你去了

var encryptedMessage, err = rsa.EncryptOAEP(sha256.New(), ...

我将继续并假设是使用SHA-2-256的OAEP。

在C#中你有

return RsaProvider.Decrypt(encryptedData, true);

哪个是带有SHA-1的OAEP。

您需要放弃RSACryptoServiceProvider。 如果你切换到RSACng,你可以用它解密

using (RSA rsa = new RSACng())
{
    rsa.ImportParameters(rsaParameters);
    return rsa.Decrypt(encryptedData, RSAEncryptionPadding.OaepSHA256);
}

而我在这里:

ConvertToPublicKey您导出XML,解析XML,并将其转换为字节数组。 为什么不直接调用ExportParameters(false)并提取模数和指数字节数组?

在解决@bartonjs的评论之后,我也改变了回复的响应

var encryptedMessage, err = rsa.EncryptOAEP(sha256.New(), rand.Reader, publicKey, []byte(licenseInformationJson), []byte(""))
if err == nil {
    responseWriter.Write([]byte(encryptedMessage))
}

注意,从go c#发送的[]byte(encryptedMessage)字节流略有改变,因为encryptedMessage[]unint8 因此,您需要将encryptedMessage类型转换为[]byte以便在c#中正确映射值。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM