c# - cryptico.js 加密消息,必须在 C# 中解密

标签 c# javascript public-key-encryption cryptico

我需要能够使用 C# 在服务器上解密字符串,但该字符串是在客户端使用公钥加密和 cryptico.js 加密的。有关详细信息,请参阅最后的上下文。

Cryptico 给了我一个像这样的 RSA 私钥(注意 - “像”这个 - 我为这个问题创建了一个新 key ):

Array ( [n] => 8029845567507477803775928519657066509146751167600087041355508603090505634905205233922950527978886894355290423984597739819216469551137046641801207199138209 [e] => 3 [d] => 5353230378338318535850619013104711006097834111733391360903672402060337089936682996269976597251251223844095913209399106464214877696419418951728015128013411 [p] => 102067954277225510613941189336789903269738979633396754230261162567549753196947 [q] => 78671563708406591396117399809764267229341143260756252277657051641634753921147 [dmp1] => 68045302851483673742627459557859935513159319755597836153507441711699835464631 [dmq1] => 52447709138937727597411599873176178152894095507170834851771367761089835947431 [coeff] => 26458340158787140383846156526777567128582042036682248240414722856369310516021 

...加上一堆方法。

我正在尝试这样解密:

                RSAParameters parameters = new RSAParameters();

            System.Text.ASCIIEncoding encoding = new System.Text.ASCIIEncoding();

            parameters.Exponent = encoding.GetBytes("3");

            //dmp1
            parameters.DP =
                encoding.GetBytes("68045302851483673742627459557859935513159319755597836153507441711699835464631");

            //dmq1
            parameters.DQ =
                encoding.GetBytes("52447709138937727597411599873176178152894095507170834851771367761089835947431");

            //d
            parameters.D =
                encoding.GetBytes(
                    "5353230378338318535850619013104711006097834111733391360903672402060337089936682996269976597251251223844095913209399106464214877696419418951728015128013411");

            //p
            parameters.P =
                encoding.GetBytes("102067954277225510613941189336789903269738979633396754230261162567549753196947");

            //q
            parameters.Q =
                encoding.GetBytes("78671563708406591396117399809764267229341143260756252277657051641634753921147");

            //n
            parameters.InverseQ =
                encoding.GetBytes(
                    "8029845567507477803775928519657066509146751167600087041355508603090505634905205233922950527978886894355290423984597739819216469551137046641801207199138209");

            //coeff
            parameters.Modulus =
                encoding.GetBytes("26458340158787140383846156526777567128582042036682248240414722856369310516021");

            RSA rsa = new RSACryptoServiceProvider();
            rsa.ImportParameters(parameters);

            var decryptThis = encoding.GetBytes(ciphertext);

            var result = rsa.DecryptValue(decryptThis);

            resultString = encoding.GetString(result);

但这会抛出“错误数据”异常。

有没有对 C# 更有经验的人知道我哪里出错了?

谢谢,

G


上下文详细信息:我正在尝试在应用程序的客户端和服务器端实现密码强度检查功能,但仅使用服务器端的代码。为了在客户端实现这一点,我想将假定的密码发送到服务器,判断其强度,然后返回一个显示在客户端的分数。这意味着我只需要在服务器上维护密码强度检查代码。作为一项额外的安全措施,我使用 cryptico.js 库对假定密码进行加密,然后再将其发送到服务器进行判断。

最佳答案

BigInteger 和 RSAParameter 类以不同的格式存储数字。

BigInteger 将数字存储为小端 - 这意味着如果您创建一个 BigInteger 保存十六进制数 0xABCD (43981) - ToByteArray 将返回字节 {0xCD, 0xAB, 0x00}(稍后会详细介绍额外的零)。

RSAParameter 旨在以大端格式存储数字。 这意味着字节 {0xCD, 0xAB, 0x00} 写入,例如,Exponent 属性,将被解释 通过 RSA 实现作为编号 0xCDAB00 (13478656)。

让事情变得更复杂——RSAParameters 总是正数,而 BigInteger 支持符号。 和相同的数字 0xFF - 将被不同地解释。对于 RSA,它是十进制的 255。对于 BigInteger,它意味着 -1, 因为它将最高有效位解释为一个符号。 这就是添加零的原因——在构造时,BigInteger 知道我们正在传递正值 43981。 但是将此值存储为两个字节会产生负数 (-21555),因为最高位设置为字节 0xCD(二进制为 11001101)。 所以 BigInteger 加零表示正数。自己试试:

Console.WriteLine(new BigInteger(new byte[]{0xCD,0xAB}));
Console.WriteLine(BitConverter.ToString(new BigInteger(0xABCD).ToByteArray()));

因此,我们将大整数作为十进制字符串“1234567...”传递给我们,我们希望将其转换为 RSA 使用的表示形式。 为此,我们需要:

  1. 将字符串解析为 BigInteger。
  2. 提取 BigInteger 字节。
  3. 确保 BigInteger 没有添加任何前导零来补偿符号。
  4. 将数组反转回大端格式,并提供给 RSA。

代码(使用您的参数):

using System;
using System.Numerics;
using System.Security.Cryptography;
using System.Text;

class App
{
    static void Main()
    {
      var parameters = new RSAParameters();
      parameters.Exponent=B("3"); 
      parameters.Modulus=B("8029845567507477803775928519657066509146751167600087041355508603090505634905205233922950527978886894355290423984597739819216469551137046641801207199138209"); 
      parameters.D=B("5353230378338318535850619013104711006097834111733391360903672402060337089936682996269976597251251223844095913209399106464214877696419418951728015128013411");
      parameters.P=B("102067954277225510613941189336789903269738979633396754230261162567549753196947");
      parameters.Q=B("78671563708406591396117399809764267229341143260756252277657051641634753921147");
      parameters.DP=B("68045302851483673742627459557859935513159319755597836153507441711699835464631");
      parameters.DQ=B("52447709138937727597411599873176178152894095507170834851771367761089835947431");
      parameters.InverseQ=B("26458340158787140383846156526777567128582042036682248240414722856369310516021");

      var rsa = new RSACryptoServiceProvider();
      rsa.ImportParameters(parameters);
      var ciphertext = rsa.Encrypt(Encoding.ASCII.GetBytes("Hello"), false);
      Console.WriteLine(Encoding.ASCII.GetString(rsa.Decrypt(ciphertext, false)));
    }

    static byte[] B(string s)
    {
      var b = BigInteger.Parse(s);
      var ret = b.ToByteArray();
      if (ret[ret.Length - 1] == 0) 
      {
        Array.Resize(ref ret, ret.Length - 1);
      }
      Array.Reverse(ret);
      return ret;
    }
}

关于c# - cryptico.js 加密消息,必须在 C# 中解密,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/13644525/

相关文章:

javascript - 需要从 firebase html 调用数据的帮助

javascript - 如何在 json 中创建以下内容并在 javascript 中作为数组获取并搜索特定值?

带有字符串公钥的PHP OpenSSL公钥加密

c# - 使用 Postsharp 在 VS2012 中编译 c# 项目时出错

c# - 如何增加分配的内存? {由于内存不足异常,函数评估被禁用。}

javascript - 按升序对周数和年份进行排序

c++ - OpenSSL:我使用存储在外部文件中的 RSA key 对文件进行加密和解密,但有时我的程序会崩溃

c# - 您将如何在单个 ASP.Net MVC Web 应用程序中构建访问者和管理功能?

c# - 如何让表格布局适合 100% 的容器

java - 在java中将字符串转换为java.Security.Publickey