java - 充气城堡 : Signed Certificate with an existing CA

标签 java bouncycastle signature ca

我正在尝试创建一个证书 (A),该证书是为存储在 p12 keystore 中的其他证书 (B) 签名的。此存储的证书 (B) 已添加到我的本地计算机的受信任证书存储中。

证书A用于使用bouncy caSTLe 1.52库签署pdf文档,但我在签名文档中获得的数字签名无效。

如果有人可以帮助我,我将解释所完成的步骤。

首先,我从 p12 keystore 创建一个 CSR(B):

    private static PKCS10CertificationRequest generateCSR() {
    PKCS10CertificationRequest csr = null;
    try {
        initCACert();
        PKCS10CertificationRequestBuilder p10Builder = new JcaPKCS10CertificationRequestBuilder(
                new X500Principal("CN=Requested Test Certificate"), CAcert.getPublicKey());
        JcaContentSignerBuilder csBuilder = new JcaContentSignerBuilder("SHA256withRSA");
        ContentSigner signer = csBuilder.build(CApk);
        csr = p10Builder.build(signer);
    } catch (Exception e) {
        log.error(e);
    }
    return csr;
}

然后,使用此 CSR 生成证书 (A)。

    private static Certificate signCSR() throws Exception { 
    PKCS10CertificationRequest csr = generateCSR();

    AlgorithmIdentifier sigAlgId = new DefaultSignatureAlgorithmIdentifierFinder().find("SHA256withRSA");
    AlgorithmIdentifier digAlgId = new DefaultDigestAlgorithmIdentifierFinder().find(sigAlgId);

    X500Name issuer = X500Name.getInstance(CAcert.getSubjectX500Principal().getEncoded());

    BigInteger serial = new BigInteger(32, new SecureRandom());
    Calendar c = Calendar.getInstance();
    c.add(Calendar.SECOND, -1);
    Date from = c.getTime();
    c.add(Calendar.YEAR, 5);
    Date to = c.getTime();

    X509v1CertificateBuilder certBuilder = new X509v1CertificateBuilder(issuer, serial, from, to, csr.getSubject(),
            csr.getSubjectPublicKeyInfo());

    ContentSigner signer = new BcRSAContentSignerBuilder(sigAlgId, digAlgId)
            .build(PrivateKeyFactory.createKey(CApk.getEncoded()));
    X509CertificateHolder holder = certBuilder.build(signer);

    CertificateFactory certFactory = CertificateFactory.getInstance("X.509");
    InputStream in = new ByteArrayInputStream(holder.getEncoded());
    Certificate cert = certFactory.generateCertificate(in);

    return cert;
}

最后,我使用生成的证书 (A) 来签署我的 pdf。

        Certificate cert = signCSR();

        SignerInfoGeneratorBuilder signerInfoBuilder = new SignerInfoGeneratorBuilder(
                    new JcaDigestCalculatorProviderBuilder().setProvider("BC").build()
                    );
        signerInfoBuilder.setSignedAttributeGenerator( signedAttributeGenerator );

        JcaContentSignerBuilder contentSignerBuilder = new JcaContentSignerBuilder( "SHA1WITHRSA" );
        contentSignerBuilder.setProvider("BC");


        X509CertificateHolder certificateHolder = new X509CertificateHolder( cert.getEncoded( ) );

        generator.addSignerInfoGenerator(
                signerInfoBuilder.build( contentSignerBuilder.build( CApk ),
                        certificateHolder ) 
                );

        ArrayList<X509CertificateHolder> signingChainHolder = new ArrayList<X509CertificateHolder>( );
        certificateHolder = new X509CertificateHolder( cert.getEncoded() );
        certificateHolder = new X509CertificateHolder( CAcert.getEncoded() );

        signingChainHolder.add( certificateHolder );

        Store certs = new JcaCertStore( signingChainHolder );
        generator.addCertificates( certs );

        CMSTypedData content = new CMSProcessableByteArray(datos);

        CMSSignedData signedData = generator.generate( content, true );
        ByteArrayOutputStream baos = new ByteArrayOutputStream();
        new DEROutputStream(baos).writeObject(signedData.toASN1Structure());
        result = baos.toByteArray();

所有过程显然都正确执行,但是当我打开 pdf 时,签名无效:

enter image description here

enter image description here

enter image description here

编辑:我已导出生成的证书。这是得到的结果。

enter image description here

如果有任何评论或信息可以帮助我解决此问题,我将不胜感激。

提前致谢。

最佳答案

    generator.addSignerInfoGenerator(
            signerInfoBuilder.build( contentSignerBuilder.build( CApk ),
                    certificateHolder ) 
            );

如果我没看错的话,您正在使用 CA 的私钥来签署数据。应该是证书的。因此私钥和公钥不匹配,因此签名验证检查失败。

关于java - 充气城堡 : Signed Certificate with an existing CA,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/51750354/

相关文章:

php - 什么是 PHP 中的参数签名?

java - 程序读取七个整数值并打印出每个值出现的次数

java - 从十六进制字符串创建 EC 私钥

java - 如何通过反射找到所有具有特定参数的方法?

java - 如何使用 spongy caSTLe 为比特币曲线 (secp256k1) 创建 ECDSA key 对(256 位)?

java - 使用椭圆曲线 Diffie-Hellman 和辅因子 key 生成对称 key

python - 从哪里可以获得所有已知病毒特征的列表?

java - 当线程正在执行某种操作时暂停其他线程?

java - 通过循环从套接字读取数据时获取 "java.io.EOFException"

java - 升级 Java 后,Mockito 验证方法因 Object...args 而失败