Java Key Store 总是以空别名结束

Val*_*inZ 3 java bouncycastle keystore public-key private-key

我已经尝试了几天,我无可救药地被困住了。为了完全理解 java 密钥库的工作原理,我一直在尝试创建自己的密钥库,在其中放入一些东西,然后从另一个程序中检索它们。

这是我的密钥库生成器:

{

    //generate a X509 certificate
    Security.addProvider(new BouncyCastleProvider());
    CertificateFactory certificateFactory = CertificateFactory.getInstance("X.509", "BC");
    X509Certificate certificate = (X509Certificate) certificateFactory.generateCertificate(new FileInputStream("certificate.cer"));

    LOGGER.debug("BouncyCastle provider & X509 certificate added.");

    //generate a private & a public key
    KeyPair keyPair = generateRSAKeyPair();
    RSAPrivateKey priv = (RSAPrivateKey) keyPair.getPrivate();
    RSAPublicKey pub = (RSAPublicKey) keyPair.getPublic();

    //generate a keystore
    KeyStore ks = KeyStore.getInstance("PKCS12");
    char[] keyStorePassword = "keystore_password".toCharArray();
    ks.load(null, keyStorePassword);
    try (FileOutputStream fos = new FileOutputStream("TestKeyStore.jks")) {
        ks.store(fos, keyStorePassword);
    }

    ks.load(new FileInputStream("TestKeyStore.jks"), keyStorePassword);

    //Symmetric key
    SecretKey secretKey = KeyGenerator.getInstance("AES").generateKey();
    KeyStore.SecretKeyEntry secretKeyEntry = new KeyStore.SecretKeyEntry((secretKey));
    KeyStore.ProtectionParameter protectionParameter = new KeyStore.PasswordProtection(keyStorePassword);
    ks.setEntry("symmetric_key", secretKeyEntry, protectionParameter);

    //Asymmetric key
    X509Certificate[] x509Certificates = new X509Certificate[1];
    x509Certificates[0] = certificate;
    ks.setKeyEntry("asymmetric key", priv, keyStorePassword, x509Certificates);

    //certificate
    ks.setCertificateEntry("test_certif", certificate);

    Key key = ks.getKey("symmetric_key", keyStorePassword);
    System.out.println("I have this symmetric key : " + key);
    X509Certificate certificate1 = (X509Certificate) ks.getCertificate("test_certif");
    System.out.println("I have this certificate : " + certificate1);

    System.out.println(ks.aliases().nextElement());


    LOGGER.debug("all went well");
}
Run Code Online (Sandbox Code Playgroud)

正如您可能已经注意到的那样,它还没有完全完善:我现在的目标只是在密钥库中放入一些东西。但这里的重点,从最后一个System.out.println(ks.aliases().nextElement());,只是为了看到密钥库中确实有一些东西。这工作得很好,它回馈了symmetric_key.

现在,来自同一个文件夹的是另一个应该从该密钥库中读取的类。

注意:该文件没有问题(我已经通过移动其本地化进行了测试),因此不可能是这样。

KeyStore ks = KeyStore.getInstance("PKCS12");
        char[] keyStorePassword = "keystore_password".toCharArray();
        ks.load(new FileInputStream("TestKeyStore.jks"), keyStorePassword);
        System.out.println(ks.containsAlias("symmetric_key"));
Run Code Online (Sandbox Code Playgroud)

这总是让我: false

如果我尝试这个:System.out.println(ks.aliases());,它总是null

如果我试试这个:

if (!keystore.aliases().hasMoreElements()) {
    System.out.println("nothing inside the keystore");
}
Run Code Online (Sandbox Code Playgroud)

它让我回来nothing inside the keystore

即使在生成器中并非如此。

有什么线索吗?谢谢

fjs*_*jsv 8

问题是您在写入密钥库后设置条目。

如果你搬家:

try (FileOutputStream fos = new FileOutputStream("TestKeyStore.jks")) {
    ks.store(fos, keyStorePassword);
}
Run Code Online (Sandbox Code Playgroud)

直到这一行之后:

ks.setCertificateEntry("test_certif", certificate);
Run Code Online (Sandbox Code Playgroud)

一切都应该正常工作。

波纹管你有一个工作示例,为了清楚起见,删除了一些代码:

public static void main(String[] args) throws Exception{

    final KeyPairGenerator keyGen = KeyPairGenerator.getInstance("RSA");
    keyGen.initialize(2048);
    final KeyPair keyPair = keyGen.genKeyPair();

    RSAPrivateKey priv = (RSAPrivateKey) keyPair.getPrivate();

    //generate a keystore
    KeyStore ks = KeyStore.getInstance("PKCS12");
    char[] keyStorePassword = PASSWORD;
    ks.load(null, keyStorePassword);

    X509Certificate[] chain = {generateCertificate("cn=Unknown", keyPair, 365, "SHA256withRSA")};

    // saving one keypair in keystore object
    ks.setKeyEntry("asymmetric key", keyPair.getPrivate(), keyStorePassword, chain);

    //Symmetric key
    SecretKey secretKey = KeyGenerator.getInstance("AES").generateKey();
    KeyStore.SecretKeyEntry secretKeyEntry = new KeyStore.SecretKeyEntry((secretKey));
    KeyStore.ProtectionParameter protectionParameter = new KeyStore.PasswordProtection(keyStorePassword);
    // saving symmetric key in keystore object
    ks.setEntry("symmetric_key", secretKeyEntry, protectionParameter);

    // Saving our keystore object into the filesystem
    try (FileOutputStream fos = new FileOutputStream("TestKeyStore.p12")) {
        ks.store(fos, keyStorePassword);
    }
Run Code Online (Sandbox Code Playgroud)

// 其余代码 }

之所以在保存keystore的方法中起作用,是因为更改后的keystore还在内存中,而不是在文件系统上。另外,由于您正在创建PKCS12密钥库,我会避免使用.jks扩展名并使用.pkcs12.