2012-03-18 26 views
19

Tôi hiện đang làm việc trên một cách để chuyển đổi khóa thành chuỗi và ngược lại. Nó hoạt động cho các khóa chuyển đổi công khai, và chuyển đổi một khóa riêng thành một String. Vì một lý do nào đó, cùng một mã sẽ không chuyển đổi một String trở lại thành khóa riêng tư mà tôi không thể tìm ra.Chuyển đổi chuỗi thành khóa mã hóa và ngược lại java

Mã chuyển đổi là:

import java.security.KeyFactory; 
import java.security.PrivateKey; 
import java.security.PublicKey; 
import java.security.spec.X509EncodedKeySpec; 
import sun.misc.BASE64Decoder; 
import sun.misc.BASE64Encoder; 

@SuppressWarnings("restriction") 
public class KeyConvert { 


public static PublicKey stringToPublicKey(String s) { 

    BASE64Decoder decoder = new BASE64Decoder(); 

    byte[] c = null; 
    KeyFactory keyFact = null; 
    PublicKey returnKey = null; 

    try { 
     c = decoder.decodeBuffer(s); 
     keyFact = KeyFactory.getInstance("DSA", "SUN"); 
    } catch (Exception e) { 
     System.out.println("Error in Keygen"); 
     e.printStackTrace(); 
    } 


    X509EncodedKeySpec x509KeySpec = new X509EncodedKeySpec(c); 
    try { 
     returnKey = keyFact.generatePublic(x509KeySpec); 
    } catch (Exception e) { 

     System.out.println("Error in Keygen2"); 
     e.printStackTrace(); 

    } 

    return returnKey; 

} 
public static PrivateKey stringToPrivateKey(String s) { 

    BASE64Decoder decoder = new BASE64Decoder(); 
    byte[] c = null; 
    KeyFactory keyFact = null; 
    PrivateKey returnKey = null; 

    try { 

        c = decoder.decodeBuffer(s); 
     keyFact = KeyFactory.getInstance("DSA", "SUN"); 
    } catch (Exception e) { 

     System.out.println("Error in first try catch of stringToPrivateKey"); 
     e.printStackTrace(); 
    } 


    X509EncodedKeySpec x509KeySpec = new X509EncodedKeySpec(c); 
    try { //the next line causes the crash 
     returnKey = keyFact.generatePrivate(x509KeySpec); 
    } catch (Exception e) { 

     System.out.println("Error in stringToPrivateKey"); 
     e.printStackTrace(); 
    } 

    return returnKey; 

} 

public static String publicKeyToString(PublicKey p) { 

    byte[] publicKeyBytes = p.getEncoded(); 
    BASE64Encoder encoder = new BASE64Encoder(); 
    return encoder.encode(publicKeyBytes); 

} 
public static String privateKeyToString(PrivateKey p) { 

    byte[] privateKeyBytes = p.getEncoded(); 
    BASE64Encoder encoder = new BASE64Encoder(); 
    return encoder.encode(privateKeyBytes); 
} 
} 

Và mã lái xe tôi đang sử dụng là:

import java.security.PrivateKey; 
import java.security.PublicKey; 

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


    ServerSQLManager s = new ServerSQLManager(); 
    ServerUser user = new ServerUser("testUser", "pass123"); 
    s.getKeys(user); 


    PublicKey testKey = user.getPublicKey(); 
    System.out.println(testKey); 
    PrivateKey testKey2 = user.getPrivateKey(); 
    System.out.println(testKey2); 
} 
} 

Khi tôi chạy này, tôi nhận được một đặc điểm kỹ thuật quan trọng không phù hợp, bất cứ ai có thể chỉ cho tôi đi đúng hướng với điều này? Đầu ra của bàn điều khiển là:

Error in stringToPrivateKey 


java.security.spec.InvalidKeySpecException: Inappropriate key specification 
at sun.security.provider.DSAKeyFactory.engineGeneratePrivate(Unknown Source) 
at java.security.KeyFactory.generatePrivate(Unknown Source) 
at KeyConvert.stringToPrivateKey(KeyConvert.java:61) 
at ServerSQLManager.getKeys(ServerSQLManager.java:128) 
at SQLServerTest.main(SQLServerTest.java:20) 

Sun DSA Public Key 
    Parameters:DSA 
    p:  fd7f5381 1d751229 52df4a9c 2eece4e7 f611b752 3cef4400 c31e3f80 b6512669 
    455d4022 51fb593d 8d58fabf c5f5ba30 f6cb9b55 6cd7813b 801d346f f26660b7 
    6b9950a5 a49f9fe8 047b1022 c24fbba9 d7feb7c6 1bf83b57 e7c6a8a6 150f04fb 
    83f6d3c5 1ec30235 54135a16 9132f675 f3ae2b61 d72aeff2 2203199d d14801c7 
q:  9760508f 15230bcc b292b982 a2eb840b f0581cf5 
g:  f7e1a085 d69b3dde cbbcab5c 36b857b9 7994afbb fa3aea82 f9574c0b 3d078267 
    5159578e bad4594f e6710710 8180b449 167123e8 4c281613 b7cf0932 8cc8a6e1 
    3c167a8b 547c8d28 e0a3ae1e 2bb3a675 916ea37f 0bfa2135 62f1fb62 7a01243b 
    cca4f1be a8519089 a883dfe1 5ae59f06 928b665e 807b5525 64014c3b fecf492a 

    y: 
    5210e849 24d90208 56802887 dfaededf 78d3e6d0 d2e59a1c fb52dde8 96147784 
    b2589365 2529414c 8265b61d c1fe6d98 cee0eea3 cce1e366 cd621ca7 41e3a94f 
    9c15bfcb eb860d19 21efd574 79bb5b15 8159b1cb e3fc7f76 f85a6fc1 8d65afc6 
    7c4fafda 503b01b5 99752ee4 2408ad80 1d983579 b00e2120 6d735874 ccaea1c0 

null 

Tôi biết mình không nên sử dụng thư viện sun.misc. * Và chúng tôi đang tìm kiếm phiên bản Apache, nhưng vẫn muốn tìm hiểu điều này (cũng nó sẽ không bao giờ là mã sản xuất)

Bất kỳ lời khuyên nào được đánh giá cao.

+0

Đừng làm điều này. Chuỗi không phải là vùng chứa cho dữ liệu nhị phân. – EJP

+0

Bạn đã cố gắng gỡ lỗi này để thu hẹp nguyên nhân của điều này? Không ai ở đây sẽ gỡ lỗi mã của bạn cho bạn khi bạn chỉ đổ tất cả các tệp của mình. – Jochen

Trả lời

44

Khóa công khai được lưu trữ bằng cách sử dụng X509EncodedKeySpec như bạn có, nhưng Khóa riêng sử dụng PKCS8EncodedKeySpec. Ví dụ như thế này:

public static PrivateKey loadPrivateKey(String key64) throws GeneralSecurityException { 
    byte[] clear = base64Decode(key64); 
    PKCS8EncodedKeySpec keySpec = new PKCS8EncodedKeySpec(clear); 
    KeyFactory fact = KeyFactory.getInstance("DSA"); 
    PrivateKey priv = fact.generatePrivate(keySpec); 
    Arrays.fill(clear, (byte) 0); 
    return priv; 
} 


public static PublicKey loadPublicKey(String stored) throws GeneralSecurityException { 
    byte[] data = base64Decode(stored); 
    X509EncodedKeySpec spec = new X509EncodedKeySpec(data); 
    KeyFactory fact = KeyFactory.getInstance("DSA"); 
    return fact.generatePublic(spec); 
} 

public static String savePrivateKey(PrivateKey priv) throws GeneralSecurityException { 
    KeyFactory fact = KeyFactory.getInstance("DSA"); 
    PKCS8EncodedKeySpec spec = fact.getKeySpec(priv, 
      PKCS8EncodedKeySpec.class); 
    byte[] packed = spec.getEncoded(); 
    String key64 = base64Encode(packed); 

    Arrays.fill(packed, (byte) 0); 
    return key64; 
} 


public static String savePublicKey(PublicKey publ) throws GeneralSecurityException { 
    KeyFactory fact = KeyFactory.getInstance("DSA"); 
    X509EncodedKeySpec spec = fact.getKeySpec(publ, 
      X509EncodedKeySpec.class); 
    return base64Encode(spec.getEncoded()); 
} 


public static void main(String[] args) throws Exception { 
    KeyPairGenerator gen = KeyPairGenerator.getInstance("DSA"); 
    KeyPair pair = gen.generateKeyPair(); 

    String pubKey = savePublicKey(pair.getPublic()); 
    PublicKey pubSaved = loadPublicKey(pubKey); 
    System.out.println(pair.getPublic()+"\n"+pubSaved); 

    String privKey = savePrivateKey(pair.getPrivate()); 
    PrivateKey privSaved = loadPrivateKey(privKey); 
    System.out.println(pair.getPrivate()+"\n"+privSaved); 
} 
+1

Hoàn toàn tại chỗ trên giải pháp, cảm ơn bạn rất nhiều. –

+0

PKCS8EncodedKeySpec là thủ thuật. Thật không may của Sun/Oracle của hướng dẫn là sai. –

+0

Đây có phải là câu trả lời hợp lệ không? Khi tôi thực hiện phương thức loadPrivateKey (String key), tôi nhận được lỗi sau: java.security.spec.InvalidKeySpecException: Đặc tả khóa không thích hợp: Đầu vào DER, lỗi thẻ Integer –

Các vấn đề liên quan