java - 固定长度 64 字节 EC P-256 带有 JCE 的签名

标签 java cryptography jce elliptic-curve ecdsa

我需要一个固定长度的 64 字节 ECDSA 签名和 NIST P-256 曲线。

实现必须使用JCE。

下面的代码示例可以生成签名并验证它。

Provider provSign = new SunEC();
Provider provVerify = new SunEC();


    // generate EC key
    KeyPairGenerator kg = KeyPairGenerator.getInstance("EC", provSign);
    ECGenParameterSpec ecParam = new ECGenParameterSpec("secp256r1");
    kg.initialize(ecParam);      
    KeyPair keyPair = kg.generateKeyPair(); 
    PrivateKey privateKey = keyPair.getPrivate();      
    PublicKey publicKey = keyPair.getPublic();

    try
    {
      // export public key                  
      KeyFactory kf = KeyFactory.getInstance("EC", provSign);
      ECPublicKeySpec publicKeySpec = kf.getKeySpec(keyPair.getPublic(), ECPublicKeySpec.class);

      // import public key into other provider
      kf = KeyFactory.getInstance("EC", provVerify);
      publicKey = (PublicKey)kf.generatePublic(publicKeySpec);      
    }
    catch (InvalidKeySpecException ex)
    {                       
      ex.printStackTrace();
    }


      // do test        
      Signature sig = Signature.getInstance("SHA256withECDSA", provSign);
      Signature ver = Signature.getInstance("SHA256withECDSA", provVerify);

      byte[] data = new byte[64];

      // sign
      sig.initSign(privateKey);
      sig.update(data);
      byte [] sign = sig.sign();

      // Working Signature verification
      ver.initVerify(publicKey);
      ver.update(data);
      if (ver.verify(sign) == false)
      {
        throw new Exception("Signature Verification failed");
      }

问题是符号以某种方式编码(我认为是 DER 格式)并且长度在 70 到 72 字节之间,但我需要 64 字节(未编码/原始)签名。

我尝试过的: 转换为固定长度的64字节签名

      DerInputStream derInputStream = new DerInputStream(sign);
      DerValue[] values = derInputStream.getSequence(2);
      byte[] random = values[0].getPositiveBigInteger().toByteArray();
      byte[] signature = values[1].getPositiveBigInteger().toByteArray();


      // r and s each occupy half the array
      // Remove padding bytes
      byte[] tokenSignature = new byte[64];
      System.arraycopy(random, random.length > 32 ? 1 : 0, tokenSignature, random.length < 32 ? 1 : 0,
              random.length > 32 ? 32 : random.length);
      System.arraycopy(signature, signature.length > 32 ? 1 : 0, tokenSignature, signature.length < 32 ? 33 : 32,
              signature.length > 32 ? 32 : signature.length);

      System.out.println("Full Signature length: "+tokenSignature.length+" r length: "+random.length+" s length"+signature.length);

现在如何检查 64 Bytes tokenSignature??? 我不知道如何将 64 字节的 tokenSignature 转换回正确的格式

      ver.initVerify(publicKey);
      ver.update(data);
      if (ver.verify(???) == false)
      {
        throw new Exception("Signature Verification failed");
      }

我已经使用 BouncyCaSTLe ECDSASigner 实现了 64 字节的签名验证。但我不能使用 ECDSASigner,因为它没有扩展 SignatureSpi,因此不能与 JCE 兼容的加密服务器一起工作。

最佳答案

感谢@MaartenBodewes,我现在可以正常工作了

//How to Check Signature
byte[] r = Arrays.copyOfRange(tokenSignature, 0,tokenSignature.length/2);
byte[] s = Arrays.copyOfRange(tokenSignature, tokenSignature.length/2,tokenSignature.length);

ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();
DEROutputStream derOutputStream = new DEROutputStream(byteArrayOutputStream);
ASN1EncodableVector v=new ASN1EncodableVector();
v.add(new ASN1Integer(new BigInteger(1,r)));
v.add(new ASN1Integer(new BigInteger(1,s)));
derOutputStream.writeObject(new DERSequence(v));
byte[] derSignature = byteArrayOutputStream.toByteArray();

ver.update(data);
if (ver.verify(derSignature) == false)
{
  throw new Exception("Signature Verification failed");
}

关于java - 固定长度 64 字节 EC P-256 带有 JCE 的签名,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34063694/

相关文章:

java - 添加 HTML 后防止 WebView 滚动到页面顶部

.net - 使用 System.Text.UnicodeEncoding.Unicode.GetString(byte[]) 反向编码字节数组时间歇性失败

testing - 如何测试加密算法的质量?

java - PKIXCertPathBuilder 使用 Bouncy CaSTLe 提供程序失败但使用默认 (SUN) 提供程序

Java zip util 添加额外的文件夹

java - 枚举、单例和反序列化

java - OneToMany 和 JoinColumn 注释是分开的

java - 安全非重复随机字母数字 URL slug

java - 启用 Java 安全性(Java 加密扩展 - JCE)后 Eclipse 崩溃

java - 如何使用 RSA 和 AES 算法加密和解密文件