【问题标题】:Outputting encrypted PK8 private key from Java BouncyCastle从 Java BouncyCastle 输出加密的 PK8 私钥
【发布时间】:2016-06-15 21:11:53
【问题描述】:

我正在尝试在 Matlab 中使用 Java 生成加密的私钥和 CSR。 Matlab 增加了一些小的复杂性,但这主要是一个 Java 问题。我从私钥开始:

java.security.Security.addProvider(org.bouncycastle.jce.provider.BouncyCastleProvider());
keyGen = java.security.KeyPairGenerator.getInstance('RSA', 'BC');
keyGen.initialize(2048,  java.security.SecureRandom());
keypair = keyGen.generateKeyPair();
privateKey = keypair.getPrivate();

如果我加密密钥并将其输出为 PEM:

m=org.bouncycastle.openssl.PKCS8Generator.PBE_SHA1_3DES;
encryptorBuilder = org.bouncycastle.openssl.jcajce.JceOpenSSLPKCS8EncryptorBuilder(m);
encryptorBuilder.setRandom(java.security.SecureRandom());
encryptorBuilder.setPasssword(password);
oe = encryptorBuilder.build();
gen = org.bouncycastle.openssl.jcajce.JcaPKCS8Generator(privateKey,oe);
privKeyObj = gen.generate();
fos = java.io.FileWriter('private.pem');
pem = org.bouncycastle.openssl.jcajce.JcaPEMWriter(fos);
pem.writeObject(privKeyObj);
pem.flush();
fos.close();

我得到了一把非常好的钥匙。问题是我想将密钥与 jdbc 一起使用,所以我需要一个 DER 格式的 pk8 密钥。我不知道如何从 BouncyCastle 中得到这个。一个成功的解决方法:

textWriter = java.io.StringWriter();
pem = org.bouncycastle.openssl.jcajce.JcaPEMWriter(textWriter);
pem.writeObject(privateKey);
pem.flush();
thekey = char(textWriter.toString());
cmd = ['echo "' thekey '"|openssl pkcs8 -topk8 -out private.pk8 -inform PEM -outform DER -passout pass:' password];
system(cmd);

现在,显然这会暴露未加密的私钥和密码。我已经尝试了各种方法来将 privKeyObj 强制转换为 DER,但它们通常会给我留下:

$openssl pkcs8 -inform DER -outform PEM -in private.pk8 -out private.pem
Error decrypting key
140735211835472:error:0D0680A8:asn1 encoding routines:ASN1_CHECK_TLEN:wrong tag:tasn_dec.c:1201:
140735211835472:error:0D06C03A:asn1 encoding routines:ASN1_D2I_EX_PRIMITIVE:nested asn1 error:tasn_dec.c:765:
140735211835472:error:0D08303A:asn1 encoding routines:ASN1_TEMPLATE_NOEXP_D2I:nested asn1 error:tasn_dec.c:697:Field=version, Type=PKCS8_PRIV_KEY_INFO

此代码的目的是在最终用户的机器上生成一个 CSR,然后我对其进行签名,并使用机器的 MAC 地址(和盐)加密,以便程序只能在授权的机器上运行,并且只有授权的机器才能访问我的 PostgreSql 数据库。

建议?

【问题讨论】:

  • 请注意,我完全错过了这篇文章,因为它缺少 Java 和密码学标签。始终使用顶级访问量大的标签并指明您的语言。这比使用更具体的问题更重要:它们基本上是为了让您的问题被其他人注意到。

标签: java cryptography bouncycastle private-key pkcs#8


【解决方案1】:

我想通了。在我的原始代码中,我使用了 BcPKCS12PBEOutputEncryptorBuilder。错误的!正确的调用是 JcePKCSPBEOutputEncryptorBuilder。正确的代码(在 MATLAB 中,但转换为 Java 很简单)是:

    java.security.Security.addProvider(org.bouncycastle.jce.provider.BouncyCastleProvider());
    keyGen = java.security.KeyPairGenerator.getInstance('RSA', 'BC');
    keyGen.initialize(2048,  java.security.SecureRandom());
    keypair = keyGen.generateKeyPair();
    privateKey = keypair.getPrivate();
    builder=org.bouncycastle.pkcs.jcajce.JcaPKCS8EncryptedPrivateKeyInfoBuilder(privateKey);

    m=org.bouncycastle.asn1.pkcs.PKCSObjectIdentifiers.pbeWithSHAAnd3_KeyTripleDES_CBC;
    encryptorBuilder = org.bouncycastle.pkcs.jcajce.JcePKCSPBEOutputEncryptorBuilder(m);
    password = 'test';
    outputBuilder = encryptorBuilder.build(password);
    privKeyObj = builder.build(outputBuilder);
    fos = java.io.FileOutputStream('testkey.pk8');
    fos.write(privKeyObj.getEncoded());
    fos.flush();
    fos.close();

这会生成一个 DER 格式的 PCS#8 文件。

    openssl pkcs8 -inform DER -outform PEM -in testkey.pk8 -out testkey.pem

现在返回 PEM 私钥。读取密钥:

    myPath = java.nio.file.Paths.get(pwd,'testkey.pk8');
    encodedKey = java.nio.file.Files.readAllBytes(myPath);
    privKeyObj =org.bouncycastle.pkcs.PKCS8EncryptedPrivateKeyInfo(encodedKey);
    cp=org.bouncycastle.openssl.jcajce.JcaPEMKeyConverter();
    cp.setProvider('BC');

    decryptorBuilder = org.bouncycastle.pkcs.jcajce.JcePKCSPBEInputDecryptorProviderBuilder();
    inputBuilder = decryptorBuilder.build(password);
    info = privKeyObj.decryptPrivateKeyInfo(inputBuilder);
    decodedKey=cp.getPrivateKey(info);

请注意,在 MATLAB 中,您不需要声明返回对象的类型,也不需要在构造函数前面加上“new”。

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 2015-03-21
    • 2013-01-13
    • 2014-05-20
    • 2021-09-22
    • 2015-07-29
    • 2017-03-17
    • 1970-01-01
    相关资源
    最近更新 更多