首页 > 解决方案 > 在android的棉花糖设备中从RSA密钥生成私钥的异常

问题描述

我正在使用此代码从 RSA 密钥生成私钥

public class ReadPKCS8Pem {
    private final static String PRIVATE_KEY = Keys.PRIVATE_KEY;

    public static PrivateKey getKey(){
        try{
            // Read in the key into a String
            StringBuilder pkcs8Lines = new StringBuilder();
            BufferedReader rdr = new BufferedReader(new StringReader(PRIVATE_KEY));
            String line;
            while ((line = rdr.readLine()) != null) {
                pkcs8Lines.append(line);
            }

            // Remove the "BEGIN" and "END" lines, as well as any whitespace

            String pkcs8Pem = pkcs8Lines.toString();
            pkcs8Pem = pkcs8Pem.replace("-----BEGIN RSA PRIVATE KEY-----", "");
            pkcs8Pem = pkcs8Pem.replace("-----END RSA PRIVATE KEY-----", "");
            pkcs8Pem = pkcs8Pem.replaceAll("\\s+","");

            // Base64 decode the result

            byte [] pkcs8EncodedBytes = Base64.decode(pkcs8Pem, Base64.DEFAULT);

            // extract the private key

            PKCS8EncodedKeySpec keySpec = new PKCS8EncodedKeySpec(pkcs8EncodedBytes);
            KeyFactory kf = KeyFactory.getInstance("RSA");
            PrivateKey privKey = kf.generatePrivate(keySpec);
            System.out.println(privKey);

            return privKey;
        }catch (Exception ex){
            ex.printStackTrace();
        }
        return null;
    }
}

在行中:

PrivateKey privKey = kf.generatePrivate(keySpec);

我得到了例外:

java.lang.RuntimeException: error:0c0890ba:ASN.1 encoding routines:asn1_check_tlen:WRONG_TAG

标签: javaandroidrsaandroid-6.0-marshmallowandroid-8.0-oreo

解决方案


PEM 文件中的标头-----BEGIN RSA PRIVATE KEY-----表示它包含 PKCS#1 密钥,但 Java(和 Android)不支持 pkcs1,因此您需要使用 openssl 之类的工具将密钥从 pkcs1 转换为 pcks8

# openssl pkcs8 -topk8 -inform PEM -outform PEM -nocrypt -in pkcs1.key -out pkcs8.key

或者,您可以使用 Bouncycastle 加载密钥。请参阅在 JAVA 中读取格式为 PKCS1 的 RSA 私钥


我不知道为什么它在 Oreo 中有效,即使没有记录https://developer.android.com/reference/java/security/spec/PKCS8EncodedKeySpec


推荐阅读