解析加密的 PKCS#8 编码的 PEM 文件程序化

huangapple go评论87阅读模式
英文:

Parsing encrypted PKCS#8 encoded pem file programatically

问题

I am having trouble figuring out how to properly read a private key of a PEM file. I have gone through different topics on Stack Overflow, but I couldn't find the solution for it. What I want to achieve is reading an encrypted private key in PKCS#8 encoding file from a classpath and load it as a key-entry in an in-memory keystore. Below is an example private key which I try to parse, the password is secret. It is purely created for sharing here, so it is not a private key used on a production machine.

I created it from a P12 file with the following command: openssl pkcs12 -in key-pair.p12 -out key-pair.pem

Valid example (throw-away) key-pair

  1. -----BEGIN ENCRYPTED PRIVATE KEY-----
  2. MIIFDjBABgkqhkiG9w0BBQ0wMzAbBgkqhkiG9w0BBQwwDgQI9/FSBonUacYCAggA
  3. MBQGCCqGSIb3DQMHBAidGkS8wYhOpwSCBMi8JaSYOKudMNNFRpzL7QMIZgFtzDay
  4. MmOroy3lW34dOa7dusqDl4d2gklKcHCpbEaTYxm5aQJ1LuiOdGtFy7HwxOvKU5xz
  5. 4qsJoeBIpE0eCTKjQW7/I38DzLXx2wUURqhMWOtFsWZEyR5Dqok3N9mIKKKBXAFG
  6. ...
  7. SRo=
  8. -----END ENCRYPTED PRIVATE KEY-----
  9. -----BEGIN CERTIFICATE-----
  10. MIIDTjCCAjagAwIBAgIEVnHI3TANBgkqhkiG9w0BAQsFADBIMQswCQYDVQQGEwJO
  11. ...
  12. 7LCdg==
  13. -----END CERTIFICATE-----

I know BouncyCastle can parse it, but I want to avoid using additional libraries. So I am wondering if it is possible with just plain JDK or some other lightweight libraries.

I am already able to parse private keys with the following headers/footers:

  1. -----BEGIN PRIVATE KEY-----
  2. *
  3. -----END PRIVATE KEY-----

and

  1. -----BEGIN RSA PRIVATE KEY-----
  2. *
  3. -----END RSA PRIVATE KEY-----

I already use the following snippet to accomplish that:

  1. import ... // (imports)
  2. class App {
  3. // ... (other constants and methods)
  4. public static void main(String[] args) throws ... {
  5. String privateKeyContent = "..."; // (private key content)
  6. String certificateContent = "..."; // (certificate content)
  7. PrivateKey privateKey = parsePrivateKey(privateKeyContent);
  8. Certificate[] certificates = parseCertificate(certificateContent).values()
  9. .toArray(new Certificate[]{});
  10. KeyStore keyStore = createEmptyKeyStore();
  11. keyStore.setKeyEntry("client", privateKey, null, certificates);
  12. }
  13. private static PrivateKey parsePrivateKey(String identityContent) throws ... {
  14. // ... (current parsing logic)
  15. }
  16. private static KeySpec getKeySpecFromAsn1StructuredData(byte[] decodedPrivateKeyContent) throws ... {
  17. // ... (current ASN.1 parsing logic)
  18. }
  19. // ... (other methods)
  20. private static Map<String, Certificate> parseCertificate(String certificateContent) throws ... {
  21. // ... (current certificate parsing logic)
  22. }
  23. }

I also need to handle an ASN.1 encoded private key. I could use BouncyCastle, but again I wanted to avoid it because I am curious about alternatives. I found the asn-one Java library to parse ASN.1 encoded private keys, and it works quite well.

I have shared my full snippet of handling the other cases, so it will be easier to try out when someone wants to help me.

From the article ASN.1 key structures in DER and PEM, I learned that a private key with the following header/footer: -----BEGIN ENCRYPTED PRIVATE KEY----- * -----END ENCRYPTED PRIVATE KEY----- is also an ASN.1 encoded private key with a specific structure.

I've tried different approaches, but I am not sure how to parse it correctly to a Java object and load it as a KeySpec. Any help is welcome!

英文:

I am having trouble figuring out how to properly read a private key of a pem file. I have gone through different topics on stackoverflow, but I couldn't find the solution for it. What I want to achieve is reading an encrypted private key in pkcs#8 encoding file from a classpath and load it as a key-entry in a in memory keystore. Below is an example private key which I try to parse, the password is secret. It is purely created for sharing here, so it is not a private key which is used on a production machine.

I created it from a p12 file with the following command: openssl pkcs12 -in key-pair.p12 -out key-pair.pem

Valid example (throw-away) key-pair

  1. -----BEGIN ENCRYPTED PRIVATE KEY-----
  2. MIIFDjBABgkqhkiG9w0BBQ0wMzAbBgkqhkiG9w0BBQwwDgQI9/FSBonUacYCAggA
  3. MBQGCCqGSIb3DQMHBAidGkS8wYhOpwSCBMi8JaSYOKudMNNFRpzL7QMIZgFtzDay
  4. MmOroy3lW34dOa7dusqDl4d2gklKcHCpbEaTYxm5aQJ1LuiOdGtFy7HwxOvKU5xz
  5. 4qsJoeBIpE0eCTKjQW7/I38DzLXx2wUURqhMWOtFsWZEyR5Dqok3N9mIKKKBXAFG
  6. AwNjlTRW2LyPSttiIUGN01lthjifMWoLTWB1aSGOmGeJRBdSZeqZ15xKneR4H5ja
  7. yE88YcpOHCDKMIxi6ZVoKs7jDQhu8bBKqS8NsYyh1AlP9QkvWNal36jWSzhqYNzk
  8. NRWUOZngfkdbMALVfRtbrC215jHGWVwosPIIs8rkoarRv8s6QWS1Rg3YfQ3qgcRf
  9. s7hkDFKJf3TUXr+askfamV5hc300ZG64+ldX1YxWXY8Vd/wIvHAc/YE/lTyCgYrY
  10. 19Am6MNBfp8/kXvzKj+PizB8oNDO4S8sSShEEzOQ5a/+MTC6bqB0DLWYGUqRbjLc
  11. PyYTC2C4i9Agx/GeGVE3c1UdtXiwwnt2XUn7Y1YGqABk0xGIY4J1NFTbSOxKl9hO
  12. arwopAFrZU5nsjjFzv1DJvhfQWnYX18kPSKNHDlia019M118qZ8ERwD9tH8ix9Fa
  13. R2tQdxn1aRGmvXSw+zFkbWD8aWs9n/B+QN1yllJqVoWypOld1yj+fVYYnYOtV1gK
  14. eiygrtrh3JJCvLbEQl4nOgJM3PlEtfBHSaunehIXQMD1z/NDUqgBYjuDPyqRxJeH
  15. Va5k72Nds5PeySKJJnICB3nZKjqgfLhNUrXa1SAQ4vqr0Ik/Lu9P7T+B1XiYwuUT
  16. a20+bxi/x89ZZqwp3jnDuHup7XcO1MtqsoOKP/JgkjVMesb8Q1W8i2dXzg+l4gkk
  17. l1ipreEGtT1YfFTq0DFelz6CjZFLDlGGeGWob94sW94DWTW0nsLPhQWEnwW1CcyJ
  18. oJbJdDEgdiIbRJoABDkTuVXLwTlgzHSHh6zeJvNvcojI7UI3nWYCVYvD3kwghXiP
  19. 67sKGL3ug7PFDqLia46AudGY7CFh4+wpxyH+fidLC3FMdkDBA6xR6mGgEjRLXR9M
  20. TnJ/eSYP7eqYZeKn9EarcI7v1zM2IG0/PDQCetiI0ABiHpdKyRQuuiEavp3xC5Vi
  21. h7UmJNYt8Zsz3rwqAQ4FR2+Su5R34OOdRmxTaYLe96PXTpLcLef5TkYixSY7Tzgd
  22. PMyRxRPrywklUEFe4KK/KOcdolxybfsIsxQnupLAMEsO7/Cs7mouNHISK51haDRc
  23. vNbKQ5E4xOq1U4ThW5dHR29cGZillfmMzj05ZQh3ZX2TQJP45ahFET3v9kInWCwQ
  24. 8atqclVPOSnASsJZ0PxjYgKZuY8QWYM6zpfWyWnfu/CHhWbRS/qX8T1ow2SMyPBL
  25. CQbZ+MhcdP0IrjoXhDFQsns16i/BPK5TTVqtEC2ywDf5P4/BOEZkySG9YNOd6THp
  26. VA/dVPafzmLy3ltqH+jG8ZH2+RtWx7kwBjiDWs5cF33BFrPS7AZlzMzZoCHLXD/r
  27. T/SmisybUKHMqri0x0RHeIByW0hogSByWiyIn8POabDzJV6Df9nQPziDGcSsvWfG
  28. 7q+hizh6+nnXOY+GZx3ptwg9mA9R4QyCiFNQradOaXSPxyEL2IC77/srFfVEIaU4
  29. SRo=
  30. -----END ENCRYPTED PRIVATE KEY-----
  31. -----BEGIN CERTIFICATE-----
  32. MIIDTjCCAjagAwIBAgIEVnHI3TANBgkqhkiG9w0BAQsFADBIMQswCQYDVQQGEwJO
  33. TDEVMBMGA1UEChMMVGh1bmRlcmJlcnJ5MRIwEAYDVQQLEwlBbXN0ZXJkYW0xDjAM
  34. BgNVBAMTBUhha2FuMB4XDTIwMDgzMTA4MDczOVoXDTMwMDgyOTA4MDczOVowSDEL
  35. MAkGA1UEBhMCTkwxFTATBgNVBAoTDFRodW5kZXJiZXJyeTESMBAGA1UECxMJQW1z
  36. dGVyZGFtMQ4wDAYDVQQDEwVIYWthbjCCASIwDQYJKoZIhvcNAQEBBQADggEPADCC
  37. AQoCggEBAJZ+pqirEqEk4k1ow8vryld79oCO4P/1y+v60CkLpS2MpeHE3BogTr7g
  38. WWP5HdHBMU5l8yT5tXuyVZZqgyzo0q2Sdm7GGrNunHf6Z8vPQFu69sQC1utDM9u8
  39. ZFKiKsTNJ+5QS6KsOtlACyhaLoaPAWVjtvueMjVwmM9hfv/Gq6VyyuBm2x1C4HTj
  40. zPCLHE2G1D13EJaWsvyZLGSbl0GGXZGPhaDd/vnw5TW36mvNTWW+37ZIEk4kXANa
  41. FUNsJemm8HjB/PzHs3/SXGxuD8NKobg3+cNXYwAz2s2DI0W6Xw2g5bbrMQAdBRvn
  42. 9/kNftMymDORw3RGwDM2ld4zQfIkNrkCAwEAAaNAMD4wHQYDVR0OBBYEFHhT7ATg
  43. oVVFIsglxD/1iUBRB6gDMB0GA1UdEQEB/wQTMBGCCWxvY2FsaG9zdIcEfwAAATAN
  44. BgkqhkiG9w0BAQsFAAOCAQEAhVWH/CgZ0ZNmTfiFAOnWdaZVaa7vAFPT2YbXuvlY
  45. YIRlru0B/zn2Mfwmn5W2o1CqoBuhyfErkkF4aRM1vduIirUjlcH4+cFXrV2gtlnf
  46. eWTg/sJJmYzkJTGeOIqRlB1HKCqoeNCrykkcsikECQ1nCqr1qLh9DXsUgWVW57YW
  47. qvP1P8xOO2/J9shMB6lOhftpawrqZ2hNG8fqMKjVVuUpFBNR+WODQ/rRRtqa6uU2
  48. V8aOOZx1QJUkTdN76YOCuGET7edevjpdbRXde+HQN6mbT9OLxSZHO0aQrDyDmNhp
  49. aVHuQn/KtYNWCZ78XKK8wtVnflmfqE/c9xO1n/EcVvLCdg==
  50. -----END CERTIFICATE-----

I know BouncyCastle is able to parse it, but I want to avoid using additional libraries. So I am wondering if it is possible with just plain jdk or with some other lightweight libraries.

I am already able to parse private key with the following header/footer:

  1. -----BEGIN PRIVATE KEY-----
  2. *
  3. -----END PRIVATE KEY-----

and

  1. -----BEGIN RSA PRIVATE KEY-----
  2. *
  3. -----END RSA PRIVATE KEY-----

I already use the following snippet to accomplish that:

  1. import com.hierynomus.asn1.ASN1InputStream;
  2. import com.hierynomus.asn1.encodingrules.der.DERDecoder;
  3. import com.hierynomus.asn1.types.ASN1Object;
  4. import com.hierynomus.asn1.types.constructed.ASN1Sequence;
  5. import com.hierynomus.asn1.types.primitive.ASN1Integer;
  6. import java.io.ByteArrayInputStream;
  7. import java.io.IOException;
  8. import java.math.BigInteger;
  9. import java.security.KeyFactory;
  10. import java.security.KeyStore;
  11. import java.security.KeyStoreException;
  12. import java.security.NoSuchAlgorithmException;
  13. import java.security.PrivateKey;
  14. import java.security.cert.Certificate;
  15. import java.security.cert.CertificateException;
  16. import java.security.cert.CertificateFactory;
  17. import java.security.spec.InvalidKeySpecException;
  18. import java.security.spec.KeySpec;
  19. import java.security.spec.PKCS8EncodedKeySpec;
  20. import java.security.spec.RSAPrivateCrtKeySpec;
  21. import java.util.Base64;
  22. import java.util.HashMap;
  23. import java.util.Map;
  24. import java.util.Objects;
  25. import java.util.UUID;
  26. import java.util.regex.Matcher;
  27. import java.util.regex.Pattern;
  28. class App {
  29. private static final String KEYSTORE_TYPE = &quot;PKCS12&quot;;
  30. private static final String KEY_FACTORY_ALGORITHM = &quot;RSA&quot;;
  31. private static final String CERTIFICATE_TYPE = &quot;X.509&quot;;
  32. private static final Pattern CERTIFICATE_PATTERN = Pattern.compile(&quot;-----BEGIN CERTIFICATE-----(.*?)-----END CERTIFICATE-----&quot;, Pattern.DOTALL);
  33. private static final Pattern PRIVATE_KEY_PATTERN = Pattern.compile(&quot;-----BEGIN PRIVATE KEY-----(.*?)-----END PRIVATE KEY-----&quot;, Pattern.DOTALL);
  34. private static final Pattern ENCRYPTED_PRIVATE_KEY_PATTERN = Pattern.compile(&quot;-----BEGIN ENCRYPTED PRIVATE KEY-----(.*?)-----END ENCRYPTED PRIVATE KEY-----&quot;, Pattern.DOTALL);
  35. private static final Pattern RSA_PRIVATE_KEY_PATTERN = Pattern.compile(&quot;-----BEGIN RSA PRIVATE KEY-----(.*?)-----END RSA PRIVATE KEY-----&quot;, Pattern.DOTALL);
  36. private static final String NEW_LINE = &quot;\n&quot;;
  37. private static final String EMPTY = &quot;&quot;;
  38. public static void main(String[] args) throws CertificateException, NoSuchAlgorithmException, KeyStoreException, IOException, InvalidKeySpecException {
  39. String privateKeyContent = &quot;&quot;;
  40. String certificateContent = &quot;&quot;;
  41. PrivateKey privateKey = parsePrivateKey(privateKeyContent);
  42. Certificate[] certificates = parseCertificate(certificateContent).values()
  43. .toArray(new Certificate[]{});
  44. KeyStore keyStore = createEmptyKeyStore();
  45. keyStore.setKeyEntry(&quot;client&quot;, privateKey, null, certificates);
  46. }
  47. private static KeyStore createEmptyKeyStore() throws KeyStoreException, IOException, NoSuchAlgorithmException, CertificateException {
  48. KeyStore keyStore = KeyStore.getInstance(KEYSTORE_TYPE);
  49. keyStore.load(null, null);
  50. return keyStore;
  51. }
  52. private static PrivateKey parsePrivateKey(String identityContent) throws NoSuchAlgorithmException, InvalidKeySpecException, IOException {
  53. KeySpec keySpec = null;
  54. Matcher privateKeyMatcher = PRIVATE_KEY_PATTERN.matcher(identityContent);
  55. if (privateKeyMatcher.find()) {
  56. String privateKeyContent = privateKeyMatcher.group(1).replace(NEW_LINE, EMPTY).trim();
  57. byte[] decodedPrivateKeyContent = Base64.getDecoder().decode(privateKeyContent);
  58. keySpec = new PKCS8EncodedKeySpec(decodedPrivateKeyContent);
  59. }
  60. privateKeyMatcher = RSA_PRIVATE_KEY_PATTERN.matcher(identityContent);
  61. if (privateKeyMatcher.find()) {
  62. String privateKeyContent = privateKeyMatcher.group(1).replace(NEW_LINE, EMPTY).trim();
  63. byte[] decodedPrivateKeyContent = Base64.getDecoder().decode(privateKeyContent);
  64. keySpec = getKeySpecFromAsn1StructuredData(decodedPrivateKeyContent);
  65. }
  66. privateKeyMatcher = ENCRYPTED_PRIVATE_KEY_PATTERN.matcher(identityContent);
  67. if (privateKeyMatcher.find()) {
  68. String privateKeyContent = privateKeyMatcher.group(1).replace(NEW_LINE, EMPTY).trim();
  69. byte[] decodedPrivateKeyContent = Base64.getDecoder().decode(privateKeyContent);
  70. keySpec = null; //TODO
  71. }
  72. Objects.requireNonNull(keySpec);
  73. KeyFactory keyFactory = KeyFactory.getInstance(KEY_FACTORY_ALGORITHM);
  74. return keyFactory.generatePrivate(keySpec);
  75. }
  76. private static KeySpec getKeySpecFromAsn1StructuredData(byte[] decodedPrivateKeyContent) throws IOException {
  77. try(ByteArrayInputStream privateKeyAsInputStream = new ByteArrayInputStream(decodedPrivateKeyContent)) {
  78. ASN1InputStream stream = new ASN1InputStream(new DERDecoder(), privateKeyAsInputStream);
  79. ASN1Sequence asn1Sequence = stream.readObject();
  80. if (asn1Sequence.getValue().size() &lt; 9) {
  81. throw new RuntimeException(&quot;Parsed key content doesn&#39;t have the minimum required sequence size of 9&quot;);
  82. }
  83. BigInteger modulus = extractIntValueFrom(asn1Sequence.get(1));
  84. BigInteger publicExponent = extractIntValueFrom(asn1Sequence.get(2));
  85. BigInteger privateExponent = extractIntValueFrom(asn1Sequence.get(3));
  86. BigInteger primeP = extractIntValueFrom(asn1Sequence.get(4));
  87. BigInteger primeQ = extractIntValueFrom(asn1Sequence.get(5));
  88. BigInteger primeExponentP = extractIntValueFrom(asn1Sequence.get(6));
  89. BigInteger primeExponentQ = extractIntValueFrom(asn1Sequence.get(7));
  90. BigInteger crtCoefficient = extractIntValueFrom(asn1Sequence.get(8));
  91. return new RSAPrivateCrtKeySpec(
  92. modulus, publicExponent, privateExponent,
  93. primeP, primeQ, primeExponentP, primeExponentQ, crtCoefficient
  94. );
  95. }
  96. }
  97. private static BigInteger extractIntValueFrom(ASN1Object&lt;?&gt; asn1Object) {
  98. if (asn1Object instanceof ASN1Integer) {
  99. return ((ASN1Integer) asn1Object).getValue();
  100. } else {
  101. throw new RuntimeException(String.format(
  102. &quot;Unable to parse the provided value of the object type [%s]. The type should be an instance of [%s]&quot;,
  103. asn1Object.getClass().getName(), ASN1Integer.class.getName())
  104. );
  105. }
  106. }
  107. private static Map&lt;String, Certificate&gt; parseCertificate(String certificateContent) throws IOException, CertificateException {
  108. Map&lt;String, Certificate&gt; certificates = new HashMap&lt;&gt;();
  109. Matcher certificateMatcher = CERTIFICATE_PATTERN.matcher(certificateContent);
  110. while (certificateMatcher.find()) {
  111. String sanitizedCertificate = certificateMatcher.group(1).replace(NEW_LINE, EMPTY).trim();
  112. byte[] decodedCertificate = Base64.getDecoder().decode(sanitizedCertificate);
  113. try(ByteArrayInputStream certificateAsInputStream = new ByteArrayInputStream(decodedCertificate)) {
  114. CertificateFactory certificateFactory = CertificateFactory.getInstance(CERTIFICATE_TYPE);
  115. Certificate certificate = certificateFactory.generateCertificate(certificateAsInputStream);
  116. certificates.put(UUID.randomUUID().toString(), certificate);
  117. }
  118. }
  119. return certificates;
  120. }
  121. }

I also am required to handle asn.1 encoded private key. I could use BouncyCastle, but again I wanted to avoid it because I am curious for alternatives. I got inspired from this topic: https://stackoverflow.com/a/42733858/6777695 but DerInputStream and DerValue aren't accessible anymore from jdk 11 onwards. And the sun packages should be avoided... So I discovered asn-one java library to parse asn.1 encoded private key, and it works quite well.

I have shared my full snippet of handling the other cases, so it will be easier to try out when someone wants to help me.

From the following article: ASN.1 key structures in DER and PEM I learned that a private key having the following header/footer: -----BEGIN ENCRYPTED PRIVATE KEY----- * -----END ENCRYPTED PRIVATE KEY----- is also an asn.1 encoded private key with the following structure:

  1. EncryptedPrivateKeyInfo ::= SEQUENCE {
  2. encryptionAlgorithm EncryptionAlgorithmIdentifier,
  3. encryptedData EncryptedData
  4. }
  5. EncryptionAlgorithmIdentifier ::= AlgorithmIdentifier
  6. EncryptedData ::= OCTET STRING

I also tried the answers provided here https://stackoverflow.com/questions/2654949/how-to-read-a-password-encrypted-key-with-java but those also didn't work. However I am not quite sure how to parse it correctly to a java object and load it as a KeySpec. So any help is welcome!

======>
Updated progress at 12-09-2020 based on the input of dave_thompson_085

  1. Matcher privateKeyMatcher = Pattern.compile(&quot;-----BEGIN ENCRYPTED PRIVATE KEY-----(.*?)-----END ENCRYPTED PRIVATE KEY-----&quot;, Pattern.DOTALL)
  2. .matcher(identityContent);
  3. if (privateKeyMatcher.find()) {
  4. String privateKeyContent = privateKeyMatcher.group(1).replace(NEW_LINE, EMPTY).trim();
  5. byte[] decodedPrivateKeyContent = Base64.getDecoder().decode(privateKeyContent);
  6. try (ByteArrayInputStream privateKeyAsInputStream = new ByteArrayInputStream(decodedPrivateKeyContent)) {
  7. ASN1InputStream stream = new ASN1InputStream(new DERDecoder(), privateKeyAsInputStream);
  8. ASN1Sequence asn1Sequence = stream.readObject();
  9. ASN1OctetString privateKeyAsOctetString = (ASN1OctetString) asn1Sequence.get(1);
  10. ASN1OctetString saltAsOctetString = (ASN1OctetString) ((ASN1Sequence) ((ASN1Sequence) ((ASN1Sequence) ((ASN1Sequence) asn1Sequence.get(0)).get(1)).get(0)).get(1)).get(0);
  11. ASN1OctetString initializationVectorAsOctec = ((ASN1OctetString) ((ASN1Sequence) ((ASN1Sequence) ((ASN1Sequence) asn1Sequence.get(0)).get(1)).get(1)).get(1));
  12. ASN1Integer iterationCount = (ASN1Integer) ((ASN1Sequence) ((ASN1Sequence) ((ASN1Sequence) ((ASN1Sequence) asn1Sequence.get(0)).get(1)).get(0)).get(1)).get(1);
  13. IvParameterSpec ivParameterSpec = new IvParameterSpec(initializationVectorAsOctec.getValue());
  14. SecretKeyFactory secretKeyFactory = SecretKeyFactory.getInstance(&quot;PBKDF2withHmacSHA1&quot;);
  15. EncryptedPrivateKeyInfo encryptedPrivateKeyInfo = new EncryptedPrivateKeyInfo(secretKeyFactory.getAlgorithm(), privateKeyAsOctetString.getValue());
  16. int keyLength = encryptedPrivateKeyInfo.getEncoded().length;
  17. PBEKeySpec pbeKeySpec = new PBEKeySpec(keyPassword, saltAsOctetString.getValue(), iterationCount.getValue().intValue(), keyLength);
  18. Cipher cipher = Cipher.getInstance(&quot;DESede/CBC/PKCS5Padding/&quot;);
  19. SecretKey secretKey = secretKeyFactory.generateSecret(pbeKeySpec);
  20. cipher.init(Cipher.DECRYPT_MODE, secretKey, ivParameterSpec);
  21. PKCS8EncodedKeySpec keySpec = encryptedPrivateKeyInfo.getKeySpec(cipher);
  22. }
  23. }

Results into the following exception: java.security.InvalidKeyException: Wrong algorithm: DESede or TripleDES required on the statement cipher.init(Cipher.DECRYPT_MODE, secretKey, ivParameterSpec);

======>
Updated progress at 13-09-2020 based on the input of Michael Fehr

  1. Matcher privateKeyMatcher = Pattern.compile(&quot;-----BEGIN ENCRYPTED PRIVATE KEY-----(.*?)-----END ENCRYPTED PRIVATE KEY-----&quot;, Pattern.DOTALL)
  2. .matcher(identityContent);
  3. if (privateKeyMatcher.find()) {
  4. String privateKeyContent = privateKeyMatcher.group(1).replace(NEW_LINE, EMPTY).trim();
  5. byte[] decodedPrivateKeyContent = Base64.getDecoder().decode(privateKeyContent);
  6. try (ByteArrayInputStream privateKeyAsInputStream = new ByteArrayInputStream(decodedPrivateKeyContent)) {
  7. ASN1InputStream stream = new ASN1InputStream(new DERDecoder(), privateKeyAsInputStream);
  8. ASN1Sequence asn1Sequence = stream.readObject();
  9. ASN1OctetString privateKeyAsOctetString = (ASN1OctetString) asn1Sequence.get(1);
  10. ASN1OctetString saltAsOctetString = (ASN1OctetString) ((ASN1Sequence) ((ASN1Sequence) ((ASN1Sequence) ((ASN1Sequence) asn1Sequence.get(0)).get(1)).get(0)).get(1)).get(0);
  11. ASN1OctetString initializationVectorAsOctec = ((ASN1OctetString) ((ASN1Sequence) ((ASN1Sequence) ((ASN1Sequence) asn1Sequence.get(0)).get(1)).get(1)).get(1));
  12. ASN1Integer iterationCount = (ASN1Integer) ((ASN1Sequence) ((ASN1Sequence) ((ASN1Sequence) ((ASN1Sequence) asn1Sequence.get(0)).get(1)).get(0)).get(1)).get(1);
  13. IvParameterSpec ivParameterSpec = new IvParameterSpec(initializationVectorAsOctec.getValue());
  14. SecretKeyFactory secretKeyFactory = SecretKeyFactory.getInstance(&quot;PBKDF2withHmacSHA1&quot;);
  15. EncryptedPrivateKeyInfo encryptedPrivateKeyInfo = new EncryptedPrivateKeyInfo(secretKeyFactory.getAlgorithm(), privateKeyAsOctetString.getValue());
  16. int keyLength = 24 * 8;
  17. PBEKeySpec pbeKeySpec = new PBEKeySpec(keyPassword, saltAsOctetString.getValue(), iterationCount.getValue().intValue(), keyLength);
  18. Cipher cipher = Cipher.getInstance(&quot;DESede/CBC/PKCS5Padding/&quot;);
  19. SecretKey secretKeyPbkdf = secretKeyFactory.generateSecret(pbeKeySpec);
  20. SecretKey secretKey = new SecretKeySpec(secretKeyPbkdf.getEncoded(), &quot;DESede&quot;);
  21. cipher.init(Cipher.DECRYPT_MODE, secretKey, ivParameterSpec);
  22. PKCS8EncodedKeySpec keySpec = encryptedPrivateKeyInfo.getKeySpec(cipher);
  23. }
  24. }

答案1

得分: 4

一个更正式的关于PEM格式的参考资料是rfc7468的第11节,它指定了ASN.1内容(在去除PEM后)是PKCS8的加密形式,定义在rfc5208的第6节附录A中略微进行了修改:

  1. EncryptedPrivateKeyInfo ::= SEQUENCE {
  2. encryptionAlgorithm AlgorithmIdentifier {{KeyEncryptionAlgorithms}},
  3. encryptedData EncryptedData
  4. }
  5. EncryptedData ::= OCTET STRING
  6. (已删除部分)
  7. KeyEncryptionAlgorithms ALGORITHM-IDENTIFIER ::= {
  8. ... -- 对于本地配置文件
  9. }

但它未能解释在PKCS8中通常(除了PKCS12之外)使用的密钥加密算法是PKCS5 v2 rfc2898 或现在的v2.1 rfc8018中定义的基于密码的加密(次要差异在这里并不重要)。特别是对于您的问题中的密钥文件,AlgorithmIdentifier在OpenSSL中解析为:

  1. 4:d=1 hl=2 l= 64 cons: SEQUENCE
  2. 6:d=2 hl=2 l= 9 prim: OBJECT :PBES2
  3. 17:d=2 hl=2 l= 51 cons: SEQUENCE
  4. 19:d=3 hl=2 l= 27 cons: SEQUENCE
  5. 21:d=4 hl=2 l= 9 prim: OBJECT :PBKDF2
  6. 32:d=4 hl=2 l= 14 cons: SEQUENCE
  7. 34:d=5 hl=2 l= 8 prim: OCTET STRING [HEX DUMP]:F7F1520689D469C6
  8. 44:d=5 hl=2 l= 2 prim: INTEGER :0800
  9. 48:d=3 hl=2 l= 20 cons: SEQUENCE
  10. 50:d=4 hl=2 l= 8 prim: OBJECT :des-ede3-cbc
  11. 60:d=4 hl=2 l= 8 prim: OCTET STRING [HEX DUMP]:9D1A44BCC1884EA7

PBES2在第6.2节中定义,它由PBKDF2密钥派生后跟实际上任何对称加密/解密组成。

您应该能够看到顶层参数与附录A.4中的PBES2-Params匹配,而密钥派生功能部分中的参数与附录A.2中的PBKDF2-Params匹配(包括盐和迭代次数,省略了密钥长度和PRF),而用于加密方案部分的参数只是一个包含初始化向量(IV)的OCTET STRING,参见附录B.2.2

因此,要解密这个密钥,您需要根据第5.2节的规定实现PBKDF2密钥派生,使用那些参数,然后使用CBC模式中的三重DES(也称为TDEA)解密,密钥长度为三倍长度,使用PKCS5/7填充,Java加密支持的方式是Cipher.getInstance("DESede/CBC/PKCS5Padding"),使用从第一步派生的密钥和输入数据中的IV作为IVParameterSpec。结果将是一个_PKCS8未加密_密钥结构,您可以将其作为PKCS8EncodedKeySpec通过适当的KeyFactory运行,就像您已经为未加密的输入格式所做的那样。

尽管考虑到JRE的大小通常超过100MB,而Bouncy库只增加了大约5MB,而且使用起来更加容易,但我不能说我同意您的评估。

英文:

A more official reference for this PEM format is rfc7468 section 11 which specifies that the ASN.1 content (after de-PEM) is PKCS8's encrypted form defined in rfc5208 section 6 and appendix A which modifies it slightly to:

  1. EncryptedPrivateKeyInfo ::= SEQUENCE {
  2. encryptionAlgorithm AlgorithmIdentifier {{KeyEncryptionAlgorithms}},
  3. encryptedData EncryptedData
  4. }
  5. EncryptedData ::= OCTET STRING
  6. (redacted)
  7. KeyEncryptionAlgorithms ALGORITHM-IDENTIFIER ::= {
  8. ... -- For local profiles
  9. }

which fails to explain that the key-encryption algorithms commonly used for PKCS8 (except in PKCS12) are the password-based encryptions defined in PKCS5 v2 rfc2898 or now v2.1 rfc8018 (the minor differences don't matter here). In particular for the key file in your Q the AlgorithmIdentifier parses (with OpenSSL) as:

  1. 4:d=1 hl=2 l= 64 cons: SEQUENCE
  2. 6:d=2 hl=2 l= 9 prim: OBJECT :PBES2
  3. 17:d=2 hl=2 l= 51 cons: SEQUENCE
  4. 19:d=3 hl=2 l= 27 cons: SEQUENCE
  5. 21:d=4 hl=2 l= 9 prim: OBJECT :PBKDF2
  6. 32:d=4 hl=2 l= 14 cons: SEQUENCE
  7. 34:d=5 hl=2 l= 8 prim: OCTET STRING [HEX DUMP]:F7F1520689D469C6
  8. 44:d=5 hl=2 l= 2 prim: INTEGER :0800
  9. 48:d=3 hl=2 l= 20 cons: SEQUENCE
  10. 50:d=4 hl=2 l= 8 prim: OBJECT :des-ede3-cbc
  11. 60:d=4 hl=2 l= 8 prim: OCTET STRING [HEX DUMP]:9D1A44BCC1884EA7

PBES2 is defined in section 6.2 to consist of PBKDF2 key derivation followed by practically any symmetric encryption/decryption.
You should be able to see that the top-level parameters match PBES2-Params in appendix A.4 and the parameters within the keyDerivationFunc part match PBKDF2-Params in appendix A.2 (salt and iterationCount present, keyLength and prf omitted), while the parameter for the encryptionScheme part is simply an OCTET STRING containing the Initialization Vector (IV), see appendix B.2.2.

Thus to decrypt this you need to implement PBKDF2 key derivation as specified in section 5.2 with those parameters, <strike>which Java crypto supports as SecretKeyFactory.getInstance(&quot;PBKDF2withHmacSHA1&quot;) and PBEKeySpec,</strike> and then triple-DES aka TDEA decryption with triple-length key aka keying option 1 in CBC mode with PKCS5/7 padding, which Java crypto supports as Cipher.getInstance(&quot;DESede/CBC/PKCS5Padding&quot;), using the key derived from the first step and the IV from the input data as IVParameterSpec. The result will be a PKCS8-unencrypted key structure, which you can run through a suitable KeyFactory as PKCS8EncodedKeySpec, just as you already do for the unencrypted input format.

Although considering that a JRE is usually upward of 100MB to begin with, while Bouncy adds about 5MB and is shedloads easier to use, I can't say I agree with your assessment.


UPDATE: I didn't realize, but JCE's SKF for PBKDF2 doesn't actually do the key derivation, it only packages up the data elements for use by a suitable Cipher algorithm -- and it doesn't implement a Cipher algorithm for the case we need here. So you'll need to manually code PBKDF2 which fortunately(?) I already did for https://stackoverflow.com/questions/61613286/decrypt-file-encrypted-using-openssl-with-aes-cbc-256 . Putting those together, here's tested code:

  1. static void SO63832456ManualPKCS8EPBES2 (String[] args) throws Exception {
  2. String file = args[0], pass = args[1];
  3. // I use Bouncy to read because it&#39;s much easier for me,
  4. // but you should get the same results with any ASN.1 lib
  5. PemObject obj; try(Reader rdr = new FileReader(file);
  6. PEMParser p = new PEMParser(rdr) ){ obj = p.readPemObject(); }
  7. if( ! obj.getType().equals(&quot;ENCRYPTED PRIVATE KEY&quot;) ) throw new Exception (&quot;wrong type&quot;);
  8. org.bouncycastle.asn1.pkcs.EncryptedPrivateKeyInfo eki = org.bouncycastle.asn1.pkcs.EncryptedPrivateKeyInfo.getInstance(obj.getContent());
  9. AlgorithmIdentifier algid = eki.getEncryptionAlgorithm();
  10. if( ! algid.getAlgorithm().equals(PKCSObjectIdentifiers.id_PBES2) ) throw new Exception(&quot;not PBES2&quot;);
  11. PBES2Parameters top = PBES2Parameters.getInstance(algid.getParameters());
  12. KeyDerivationFunc kdf = top.getKeyDerivationFunc();
  13. if( ! kdf.getAlgorithm().equals(PKCSObjectIdentifiers.id_PBKDF2) ) throw new Exception(&quot;not PBKDF2&quot;);
  14. PBKDF2Params kdp = PBKDF2Params.getInstance(kdf.getParameters());
  15. EncryptionScheme esa = top.getEncryptionScheme();
  16. if( ! esa.getAlgorithm().equals(PKCSObjectIdentifiers.des_EDE3_CBC) ) throw new Exception(&quot;not DES3&quot;);
  17. ASN1OctetString esp = ASN1OctetString.getInstance(esa.getParameters());
  18. // these are the (only) items actually used
  19. byte[] salt = kdp.getSalt(), iv = esp.getOctets(), edata = eki.getEncryptedData();
  20. int iter = kdp.getIterationCount().intValueExact();
  21. //SecretKeyFactory fact = SecretKeyFactory.getInstance(&quot;PBKDF2withHmacSHA1&quot;);
  22. //SecretKey skey = fact.generateSecret(new PBEKeySpec(pass.toCharArray(), salt, iter, 24));
  23. byte[] skey = PBKDF2 (&quot;HmacSHA1&quot;, pass.getBytes(&quot;UTF-8&quot;), salt, iter, 24);
  24. Cipher ciph = Cipher.getInstance(&quot;DESEDE/CBC/PKCS5Padding&quot;);
  25. ciph.init(Cipher.DECRYPT_MODE, new SecretKeySpec(skey,&quot;DESEDE&quot;), new IvParameterSpec(iv));
  26. byte[] plain = ciph.doFinal(edata);
  27. KeyFactory fac2 = KeyFactory.getInstance(&quot;RSA&quot;);
  28. PrivateKey pkey = fac2.generatePrivate(new PKCS8EncodedKeySpec(plain));
  29. System.out.println ( ((java.security.interfaces.RSAPrivateKey)pkey).getModulus() );
  30. }
  31. public static byte[] PBKDF2 (String prf, byte[] pass, byte[] salt, int iter, int len)
  32. throws NoSuchProviderException, NoSuchAlgorithmException, InvalidKeyException {
  33. byte[] result = new byte[len];
  34. Mac mac = Mac.getInstance(prf);
  35. mac.init (new SecretKeySpec (pass,prf));
  36. byte[] saltcnt = Arrays.copyOf (salt, salt.length+4);
  37. while( /*remaining*/len&gt;0 ){
  38. for( int o = saltcnt.length; --o&gt;=saltcnt.length-4; ) if( ++saltcnt[o] != 0 ) break;
  39. byte[] u = saltcnt, x = new byte[mac.getMacLength()];
  40. for( int i = 1; i &lt;= iter; i++ ){
  41. u = mac.doFinal (u);
  42. for( int o = 0; o &lt; x.length; o++ ) x[o] ^= u[o];
  43. }
  44. int len2 = Math.min (len, x.length);
  45. System.arraycopy (x,0, result,result.length-len, len2);
  46. len -= len2;
  47. }
  48. return result;
  49. }
  50. --&gt;
  51. 18998199686208613730227071552042680566660281525162377778227794768879762338654464000476266705180956908841709177555966832423085875590556580445428066261332223955361646420713402269532603328557674453787190302285379571808527818182992417461094551685532849252957724361115461909983939654809533488667032982867230996411860328826589806804115764031480737011900431983025490936659008858316889367140757195359233313370592669605576636360123189540118611655293825465309125578750878878980002346272931570343463549347548711194000065166346537274382220211242975801253306067862461345994215674963367480607539333468674983712264803816027753559737

Bleah!

答案2

得分: 2

  1. privateKey:
  2. SunRsaSign RSA private CRT key, 2048 bits
  3. params: null
  4. modulus: 18998199686208613730227071552042680566660281525162377778227794768879762338654464000476266705180956908841709177555966832423085875590556580445428066261332223955361646420713402269532603328557674453787190302285379571808527818182992417461094551685532849252957724361115461909983939654809533488667032982867230996411860328826589806804115764031480737011900431983025490936659008858316889367140757195359233313370592669605576636360123189540118611655293825465309125578750878878980002346272931570343463549347548711194000065166346537274382220211242975801253306067862461345994215674963367480607539333468674983712264803816027753559737
  5. private exponent: 2470690692670644289832084636740463653655882622624760758103597888447170520657826825702415751838767348735186037205082706146786714644526202528897926495648786865000645626802269367582073915029633022103975204027927584273373553784482725392082470421529769049284506228997163157587212073758970565083984956787424047216319298607371397767542633302071090323391531162434678662485530085857362259661656308472034152915312425224731842425134593468185574918070451135641322780271191791839285884885643517240131520070881951542294552185519522325857178404160441369354693465035929601010771762928436963178293113507661955562266846499005032897533
  6. number of certificates: 1
  7. certificate: 0 data:
  8. [
  9. Version: V3
  10. Subject: CN=Hakan, OU=Amsterdam, O=Thunderberry, C=NL
  11. Signature Algorithm: SHA256withRSA, OID = 1.2.840.113549.1.1.11
  12. Key: Sun RSA public key, 2048 bits
  13. params: null
  14. modulus: 18998199686208613730227071552042680566660281525162377778227794768879762338654464000476266705180956908841709177555966832423085875590556580445428066261332223955361646420713402269532603328557674453787190302285379571808527818182992417461094551685532849252957724361115461909983939654809533488667032982867230996411860328826589806804115764031480737011900431983025490936659008858316889367140757195359233313370592669605576636360123189540118611655293825465309125578750878878980002346272931570343463549347548711194000065166346537274382220211242975801253306067862461345994215674963367480607539333468674983712264803816027753559737
  15. public exponent: 65537
  16. Validity: [From: Mon Aug 31 10:07:39 CEST 2020,
  17. To: Thu Aug 29 10:07:39 CEST 2030]
  18. Issuer: CN=Hakan, OU=Amsterdam, O=Thunderberry, C=NL
  19. SerialNumber: [ 5671c8dd]
  20. Certificate Extensions: 2
  21. [1]: ObjectId: 2.5.29.17 Criticality=true
  22. SubjectAlternativeName [
  23. DNSName: localhost
  24. IPAddress: 127.0.0.1
  25. ]
  26. [2]: ObjectId: 2.5.29.14 Criticality=false
  27. SubjectKeyIdentifier [
  28. KeyIdentifier [
  29. 0000: 78 53 EC 04 E0 A1 55 45 22 C8 25 C4 3F F5 89 40 xS....UE".%.?..@
  30. 0010: 51 07 A8 03 Q...
  31. ]
  32. ]
  33. ]

(Note: The code is quite long and complex. I've provided the output of the code execution along with the private key and certificate information.)

英文:

Your edited code gives an error "java.security.InvalidKeyException: Wrong key size" and that's because you "read" the keyLength with this line:

  1. int keyLength = encryptedPrivateKeyInfo.getEncoded().length;

With the given private key the length would be 1247 bits and that is not an allowed TripleDES length. You can fix it to (hardcoded):

  1. int keyLength = 24 * 8;

Now we get a keyLength of 192 bit = 24 byte and that's ok for an TripleDES-key but we receive a new error:

  1. java.security.InvalidKeyException: Wrong algorithm: DESede or TripleDES required

The reason for this behavior is the following line (commented out) that can get fixed with the 2 additional lines:

  1. // SecretKey secretKey = secretKeyFactory.generateSecret(pbeKeySpec); // ### old routine
  2. SecretKey secretKeyPbkdf = secretKeyFactory.generateSecret(pbeKeySpec); // ### new
  3. SecretKey secretKey = new SecretKeySpec(secretKeyPbkdf.getEncoded(), &quot;DESede&quot;); // ### new

Now your code is successfully running and print outs the private key and the certificate:

  1. original keyLength: 1247
  2. fixed keyLength(bit): 192
  3. privateKey:
  4. SunRsaSign RSA private CRT key, 2048 bits
  5. params: null
  6. modulus: 18998199686208613730227071552042680566660281525162377778227794768879762338654464000476266705180956908841709177555966832423085875590556580445428066261332223955361646420713402269532603328557674453787190302285379571808527818182992417461094551685532849252957724361115461909983939654809533488667032982867230996411860328826589806804115764031480737011900431983025490936659008858316889367140757195359233313370592669605576636360123189540118611655293825465309125578750878878980002346272931570343463549347548711194000065166346537274382220211242975801253306067862461345994215674963367480607539333468674983712264803816027753559737
  7. private exponent: 2470690692670644289832084636740463653655882622624760758103597888447170520657826825702415751838767348735186037205082706146786714644526202528897926495648786865000645626802269367582073915029633022103975204027927584273373553784482725392082470421529769049284506228997163157587212073758970565083984956787424047216319298607371397767542633302071090323391531162434678662485530085857362259661656308472034152915312425224731842425134593468185574918070451135641322780271191791839285884885643517240131520070881951542294552185519522325857178404160441369354693465035929601010771762928436963178293113507661955562266846499005032897533
  8. number of certificates: 1
  9. certificate: 0 data:
  10. [
  11. [
  12. Version: V3
  13. Subject: CN=Hakan, OU=Amsterdam, O=Thunderberry, C=NL
  14. Signature Algorithm: SHA256withRSA, OID = 1.2.840.113549.1.1.11
  15. Key: Sun RSA public key, 2048 bits
  16. params: null
  17. modulus: 18998199686208613730227071552042680566660281525162377778227794768879762338654464000476266705180956908841709177555966832423085875590556580445428066261332223955361646420713402269532603328557674453787190302285379571808527818182992417461094551685532849252957724361115461909983939654809533488667032982867230996411860328826589806804115764031480737011900431983025490936659008858316889367140757195359233313370592669605576636360123189540118611655293825465309125578750878878980002346272931570343463549347548711194000065166346537274382220211242975801253306067862461345994215674963367480607539333468674983712264803816027753559737
  18. public exponent: 65537
  19. Validity: [From: Mon Aug 31 10:07:39 CEST 2020,
  20. To: Thu Aug 29 10:07:39 CEST 2030]
  21. Issuer: CN=Hakan, OU=Amsterdam, O=Thunderberry, C=NL
  22. SerialNumber: [ 5671c8dd]
  23. Certificate Extensions: 2
  24. [1]: ObjectId: 2.5.29.17 Criticality=true
  25. SubjectAlternativeName [
  26. DNSName: localhost
  27. IPAddress: 127.0.0.1
  28. ]
  29. [2]: ObjectId: 2.5.29.14 Criticality=false
  30. SubjectKeyIdentifier [
  31. KeyIdentifier [
  32. 0000: 78 53 EC 04 E0 A1 55 45 22 C8 25 C4 3F F5 89 40 xS....UE&quot;.%.?..@
  33. 0010: 51 07 A8 03 Q...
  34. ]
  35. ]
  36. ]
  37. Algorithm: [SHA256withRSA]
  38. Signature:
  39. 0000: 85 55 87 FC 28 19 D1 93 66 4D F8 85 00 E9 D6 75 .U..(...fM.....u
  40. 0010: A6 55 69 AE EF 00 53 D3 D9 86 D7 BA F9 58 60 84 .Ui...S......X`.
  41. 0020: 65 AE ED 01 FF 39 F6 31 FC 26 9F 95 B6 A3 50 AA e....9.1.&amp;....P.
  42. 0030: A0 1B A1 C9 F1 2B 92 41 78 69 13 35 BD DB 88 8A .....+.Axi.5....
  43. 0040: B5 23 95 C1 F8 F9 C1 57 AD 5D A0 B6 59 DF 79 64 .#.....W.]..Y.yd
  44. 0050: E0 FE C2 49 99 8C E4 25 31 9E 38 8A 91 94 1D 47 ...I...%1.8....G
  45. 0060: 28 2A A8 78 D0 AB CA 49 1C B2 29 04 09 0D 67 0A (*.x...I..)...g.
  46. 0070: AA F5 A8 B8 7D 0D 7B 14 81 65 56 E7 B6 16 AA F3 .........eV.....
  47. 0080: F5 3F CC 4E 3B 6F C9 F6 C8 4C 07 A9 4E 85 FB 69 .?.N;o...L..N..i
  48. 0090: 6B 0A EA 67 68 4D 1B C7 EA 30 A8 D5 56 E5 29 14 k..ghM...0..V.).
  49. 00A0: 13 51 F9 63 83 43 FA D1 46 DA 9A EA E5 36 57 C6 .Q.c.C..F....6W.
  50. 00B0: 8E 39 9C 75 40 95 24 4D D3 7B E9 83 82 B8 61 13 .9.u@.$M......a.
  51. 00C0: ED E7 5E BE 3A 5D 6D 15 DD 7B E1 D0 37 A9 9B 4F ..^.:]m.....7..O
  52. 00D0: D3 8B C5 26 47 3B 46 90 AC 3C 83 98 D8 69 69 51 ...&amp;G;F..&lt;...iiQ
  53. 00E0: EE 42 7F CA B5 83 56 09 9E FC 5C A2 BC C2 D5 67 .B....V...\....g
  54. 00F0: 7E 59 9F A8 4F DC F7 13 B5 9F F1 1C 56 F2 C2 76 .Y..O.......V..v
  55. ]

Here is the full code that is running on Open Java 11:

  1. import com.hierynomus.asn1.ASN1InputStream;
  2. import com.hierynomus.asn1.encodingrules.der.DERDecoder;
  3. import com.hierynomus.asn1.types.ASN1Object;
  4. import com.hierynomus.asn1.types.constructed.ASN1Sequence;
  5. import com.hierynomus.asn1.types.primitive.ASN1Integer;
  6. import com.hierynomus.asn1.types.string.ASN1OctetString;
  7. import javax.crypto.*;
  8. import javax.crypto.spec.IvParameterSpec;
  9. import javax.crypto.spec.PBEKeySpec;
  10. import javax.crypto.spec.SecretKeySpec;
  11. import java.io.ByteArrayInputStream;
  12. import java.io.IOException;
  13. import java.math.BigInteger;
  14. import java.security.*;
  15. import java.security.cert.Certificate;
  16. import java.security.cert.CertificateException;
  17. import java.security.cert.CertificateFactory;
  18. import java.security.spec.InvalidKeySpecException;
  19. import java.security.spec.KeySpec;
  20. import java.security.spec.PKCS8EncodedKeySpec;
  21. import java.security.spec.RSAPrivateCrtKeySpec;
  22. import java.util.*;
  23. import java.util.regex.Matcher;
  24. import java.util.regex.Pattern;
  25. // https://mvnrepository.com/artifact/com.hierynomus/asn-one/0.4.0
  26. // additionally you need https://mvnrepository.com/artifact/org.slf4j/slf4j-api/1.7.30
  27. // and https://mvnrepository.com/artifact/org.slf4j/slf4j-log4j12/1.7.30
  28. // https://mvnrepository.com/artifact/org.eclipse.equinox/org.apache.log4j/1.2.13.v200706111418
  29. class MainOrg3 {
  30. private static final String KEYSTORE_TYPE = &quot;PKCS12&quot;;
  31. private static final String KEY_FACTORY_ALGORITHM = &quot;RSA&quot;;
  32. private static final String CERTIFICATE_TYPE = &quot;X.509&quot;;
  33. private static final Pattern CERTIFICATE_PATTERN = Pattern.compile(&quot;-----BEGIN CERTIFICATE-----(.*?)-----END CERTIFICATE-----&quot;, Pattern.DOTALL);
  34. private static final Pattern PRIVATE_KEY_PATTERN = Pattern.compile(&quot;-----BEGIN PRIVATE KEY-----(.*?)-----END PRIVATE KEY-----&quot;, Pattern.DOTALL);
  35. private static final Pattern ENCRYPTED_PRIVATE_KEY_PATTERN = Pattern.compile(&quot;-----BEGIN ENCRYPTED PRIVATE KEY-----(.*?)-----END ENCRYPTED PRIVATE KEY-----&quot;, Pattern.DOTALL);
  36. private static final Pattern RSA_PRIVATE_KEY_PATTERN = Pattern.compile(&quot;-----BEGIN RSA PRIVATE KEY-----(.*?)-----END RSA PRIVATE KEY-----&quot;, Pattern.DOTALL);
  37. private static final String NEW_LINE = &quot;\n&quot;;
  38. private static final String EMPTY = &quot;&quot;;
  39. static char[] keyPassword = &quot;secret&quot;.toCharArray();
  40. public static void main(String[] args) throws CertificateException, NoSuchAlgorithmException, KeyStoreException, IOException, InvalidKeySpecException {
  41. String privateKeyContent = &quot;-----BEGIN ENCRYPTED PRIVATE KEY-----\n&quot; +
  42. &quot;MIIFDjBABgkqhkiG9w0BBQ0wMzAbBgkqhkiG9w0BBQwwDgQI9/FSBonUacYCAggA\n&quot; +
  43. &quot;MBQGCCqGSIb3DQMHBAidGkS8wYhOpwSCBMi8JaSYOKudMNNFRpzL7QMIZgFtzDay\n&quot; +
  44. &quot;MmOroy3lW34dOa7dusqDl4d2gklKcHCpbEaTYxm5aQJ1LuiOdGtFy7HwxOvKU5xz\n&quot; +
  45. &quot;4qsJoeBIpE0eCTKjQW7/I38DzLXx2wUURqhMWOtFsWZEyR5Dqok3N9mIKKKBXAFG\n&quot; +
  46. &quot;AwNjlTRW2LyPSttiIUGN01lthjifMWoLTWB1aSGOmGeJRBdSZeqZ15xKneR4H5ja\n&quot; +
  47. &quot;yE88YcpOHCDKMIxi6ZVoKs7jDQhu8bBKqS8NsYyh1AlP9QkvWNal36jWSzhqYNzk\n&quot; +
  48. &quot;NRWUOZngfkdbMALVfRtbrC215jHGWVwosPIIs8rkoarRv8s6QWS1Rg3YfQ3qgcRf\n&quot; +
  49. &quot;s7hkDFKJf3TUXr+askfamV5hc300ZG64+ldX1YxWXY8Vd/wIvHAc/YE/lTyCgYrY\n&quot; +
  50. &quot;19Am6MNBfp8/kXvzKj+PizB8oNDO4S8sSShEEzOQ5a/+MTC6bqB0DLWYGUqRbjLc\n&quot; +
  51. &quot;PyYTC2C4i9Agx/GeGVE3c1UdtXiwwnt2XUn7Y1YGqABk0xGIY4J1NFTbSOxKl9hO\n&quot; +
  52. &quot;arwopAFrZU5nsjjFzv1DJvhfQWnYX18kPSKNHDlia019M118qZ8ERwD9tH8ix9Fa\n&quot; +
  53. &quot;R2tQdxn1aRGmvXSw+zFkbWD8aWs9n/B+QN1yllJqVoWypOld1yj+fVYYnYOtV1gK\n&quot; +
  54. &quot;eiygrtrh3JJCvLbEQl4nOgJM3PlEtfBHSaunehIXQMD1z/NDUqgBYjuDPyqRxJeH\n&quot; +
  55. &quot;Va5k72Nds5PeySKJJnICB3nZKjqgfLhNUrXa1SAQ4vqr0Ik/Lu9P7T+B1XiYwuUT\n&quot; +
  56. &quot;a20+bxi/x89ZZqwp3jnDuHup7XcO1MtqsoOKP/JgkjVMesb8Q1W8i2dXzg+l4gkk\n&quot; +
  57. &quot;l1ipreEGtT1YfFTq0DFelz6CjZFLDlGGeGWob94sW94DWTW0nsLPhQWEnwW1CcyJ\n&quot; +
  58. &quot;oJbJdDEgdiIbRJoABDkTuVXLwTlgzHSHh6zeJvNvcojI7UI3nWYCVYvD3kwghXiP\n&quot; +
  59. &quot;67sKGL3ug7PFDqLia46AudGY7CFh4+wpxyH+fidLC3FMdkDBA6xR6mGgEjRLXR9M\n&quot; +
  60. &quot;TnJ/eSYP7eqYZeKn9EarcI7v1zM2IG0/PDQCetiI0ABiHpdKyRQuuiEavp3xC5Vi\n&quot; +
  61. &quot;h7UmJNYt8Zsz3rwqAQ4FR2+Su5R34OOdRmxTaYLe96PXTpLcLef5TkYixSY7Tzgd\n&quot; +
  62. &quot;PMyRxRPrywklUEFe4KK/KOcdolxybfsIsxQnupLAMEsO7/Cs7mouNHISK51haDRc\n&quot; +
  63. &quot;vNbKQ5E4xOq1U4ThW5dHR29cGZillfmMzj05ZQh3ZX2TQJP45ahFET3v9kInWCwQ\n&quot; +
  64. &quot;8atqclVPOSnASsJZ0PxjYgKZuY8QWYM6zpfWyWnfu/CHhWbRS/qX8T1ow2SMyPBL\n&quot; +
  65. &quot;CQbZ+MhcdP0IrjoXhDFQsns16i/BPK5TTVqtEC2ywDf5P4/BOEZkySG9YNOd6THp\n&quot; +
  66. &quot;VA/dVPafzmLy3ltqH+jG8ZH2+RtWx7kwBjiDWs5cF33BFrPS7AZlzMzZoCHLXD/r\n&quot; +
  67. &quot;T/SmisybUKHMqri0x0RHeIByW0hogSByWiyIn8POabDzJV6Df9nQPziDGcSsvWfG\n&quot; +
  68. &quot;7q+hizh6+nnXOY+GZx3ptwg9mA9R4QyCiFNQradOaXSPxyEL2IC77/srFfVEIaU4\n&quot; +
  69. &quot;SRo=\n&quot; +
  70. &quot;-----END ENCRYPTED PRIVATE KEY-----&quot;;
  71. String certificateContent = &quot;-----BEGIN CERTIFICATE-----\n&quot; +
  72. &quot;MIIDTjCCAjagAwIBAgIEVnHI3TANBgkqhkiG9w0BAQsFADBIMQswCQYDVQQGEwJO\n&quot; +
  73. &quot;TDEVMBMGA1UEChMMVGh1bmRlcmJlcnJ5MRIwEAYDVQQLEwlBbXN0ZXJkYW0xDjAM\n&quot; +
  74. &quot;BgNVBAMTBUhha2FuMB4XDTIwMDgzMTA4MDczOVoXDTMwMDgyOTA4MDczOVowSDEL\n&quot; +
  75. &quot;MAkGA1UEBhMCTkwxFTATBgNVBAoTDFRodW5kZXJiZXJyeTESMBAGA1UECxMJQW1z\n&quot; +
  76. &quot;dGVyZGFtMQ4wDAYDVQQDEwVIYWthbjCCASIwDQYJKoZIhvcNAQEBBQADggEPADCC\n&quot; +
  77. &quot;AQoCggEBAJZ+pqirEqEk4k1ow8vryld79oCO4P/1y+v60CkLpS2MpeHE3BogTr7g\n&quot; +
  78. &quot;WWP5HdHBMU5l8yT5tXuyVZZqgyzo0q2Sdm7GGrNunHf6Z8vPQFu69sQC1utDM9u8\n&quot; +
  79. &quot;ZFKiKsTNJ+5QS6KsOtlACyhaLoaPAWVjtvueMjVwmM9hfv/Gq6VyyuBm2x1C4HTj\n&quot; +
  80. &quot;zPCLHE2G1D13EJaWsvyZLGSbl0GGXZGPhaDd/vnw5TW36mvNTWW+37ZIEk4kXANa\n&quot; +
  81. &quot;FUNsJemm8HjB/PzHs3/SXGxuD8NKobg3+cNXYwAz2s2DI0W6Xw2g5bbrMQAdBRvn\n&quot; +
  82. &quot;9/kNftMymDORw3RGwDM2ld4zQfIkNrkCAwEAAaNAMD4wHQYDVR0OBBYEFHhT7ATg\n&quot; +
  83. &quot;oVVFIsglxD/1iUBRB6gDMB0GA1UdEQEB/wQTMBGCCWxvY2FsaG9zdIcEfwAAATAN\n&quot; +
  84. &quot;BgkqhkiG9w0BAQsFAAOCAQEAhVWH/CgZ0ZNmTfiFAOnWdaZVaa7vAFPT2YbXuvlY\n&quot; +
  85. &quot;YIRlru0B/zn2Mfwmn5W2o1CqoBuhyfErkkF4aRM1vduIirUjlcH4+cFXrV2gtlnf\n&quot; +
  86. &quot;eWTg/sJJmYzkJTGeOIqRlB1HKCqoeNCrykkcsikECQ1nCqr1qLh9DXsUgWVW57YW\n&quot; +
  87. &quot;qvP1P8xOO2/J9shMB6lOhftpawrqZ2hNG8fqMKjVVuUpFBNR+WODQ/rRRtqa6uU2\n&quot; +
  88. &quot;V8aOOZx1QJUkTdN76YOCuGET7edevjpdbRXde+HQN6mbT9OLxSZHO0aQrDyDmNhp\n&quot; +
  89. &quot;aVHuQn/KtYNWCZ78XKK8wtVnflmfqE/c9xO1n/EcVvLCdg==\n&quot; +
  90. &quot;-----END CERTIFICATE-----&quot;;
  91. PrivateKey privateKey = parsePrivateKey(privateKeyContent);
  92. Certificate[] certificates = parseCertificate(certificateContent).values()
  93. .toArray(new Certificate[]{});
  94. KeyStore keyStore = createEmptyKeyStore();
  95. keyStore.setKeyEntry(&quot;client&quot;, privateKey, null, certificates);
  96. System.out.println(&quot;\nprivateKey:\n&quot; + privateKey);
  97. int certificatesLength = certificates.length;
  98. System.out.println(&quot;\nnumber of certificates: &quot; + certificatesLength);
  99. for (int i = 0; i &lt; certificatesLength; i++) {
  100. System.out.println(&quot;\ncertificate: &quot; + i + &quot; data:\n&quot; + certificates[i]);
  101. }
  102. }
  103. private static KeyStore createEmptyKeyStore() throws KeyStoreException, IOException, NoSuchAlgorithmException, CertificateException {
  104. KeyStore keyStore = KeyStore.getInstance(KEYSTORE_TYPE);
  105. keyStore.load(null, null);
  106. return keyStore;
  107. }
  108. private static PrivateKey parsePrivateKey(String identityContent) throws NoSuchAlgorithmException, InvalidKeySpecException, IOException {
  109. KeySpec keySpec = null;
  110. Matcher privateKeyMatcher = PRIVATE_KEY_PATTERN.matcher(identityContent);
  111. if (privateKeyMatcher.find()) {
  112. String privateKeyContent = privateKeyMatcher.group(1).replace(NEW_LINE, EMPTY).trim();
  113. byte[] decodedPrivateKeyContent = Base64.getDecoder().decode(privateKeyContent);
  114. keySpec = new PKCS8EncodedKeySpec(decodedPrivateKeyContent);
  115. }
  116. privateKeyMatcher = RSA_PRIVATE_KEY_PATTERN.matcher(identityContent);
  117. if (privateKeyMatcher.find()) {
  118. String privateKeyContent = privateKeyMatcher.group(1).replace(NEW_LINE, EMPTY).trim();
  119. byte[] decodedPrivateKeyContent = Base64.getDecoder().decode(privateKeyContent);
  120. keySpec = getKeySpecFromAsn1StructuredData(decodedPrivateKeyContent);
  121. }
  122. privateKeyMatcher = ENCRYPTED_PRIVATE_KEY_PATTERN.matcher(identityContent);
  123. if (privateKeyMatcher.find()) {
  124. String privateKeyContent = privateKeyMatcher.group(1).replace(NEW_LINE, EMPTY).trim();
  125. byte[] decodedPrivateKeyContent = Base64.getDecoder().decode(privateKeyContent);
  126. try (ByteArrayInputStream privateKeyAsInputStream = new ByteArrayInputStream(decodedPrivateKeyContent)) {
  127. ASN1InputStream stream = new ASN1InputStream(new DERDecoder(), privateKeyAsInputStream);
  128. ASN1Sequence asn1Sequence = stream.readObject();
  129. ASN1OctetString privateKeyAsOctetString = (ASN1OctetString) asn1Sequence.get(1);
  130. ASN1OctetString saltAsOctetString = (ASN1OctetString) ((ASN1Sequence) ((ASN1Sequence) ((ASN1Sequence) ((ASN1Sequence) asn1Sequence.get(0)).get(1)).get(0)).get(1)).get(0);
  131. ASN1OctetString initializationVectorAsOctec = ((ASN1OctetString) ((ASN1Sequence) ((ASN1Sequence) ((ASN1Sequence) asn1Sequence.get(0)).get(1)).get(1)).get(1));
  132. ASN1Integer iterationCount = (ASN1Integer) ((ASN1Sequence) ((ASN1Sequence) ((ASN1Sequence) ((ASN1Sequence) asn1Sequence.get(0)).get(1)).get(0)).get(1)).get(1);
  133. IvParameterSpec ivParameterSpec = new IvParameterSpec(initializationVectorAsOctec.getValue());
  134. SecretKeyFactory secretKeyFactory = SecretKeyFactory.getInstance(&quot;PBKDF2withHmacSHA1&quot;);
  135. EncryptedPrivateKeyInfo encryptedPrivateKeyInfo = new EncryptedPrivateKeyInfo(secretKeyFactory.getAlgorithm(), privateKeyAsOctetString.getValue());
  136. int keyLength = encryptedPrivateKeyInfo.getEncoded().length;
  137. System.out.println(&quot;original keyLength: &quot; + keyLength);
  138. keyLength = 24 * 8; // ### fixed
  139. System.out.println(&quot;fixed keyLength(bit): &quot; + keyLength);
  140. PBEKeySpec pbeKeySpec = new PBEKeySpec(keyPassword, saltAsOctetString.getValue(), iterationCount.getValue().intValue(), keyLength);
  141. Cipher cipher = Cipher.getInstance(&quot;DESede/CBC/PKCS5Padding/&quot;);
  142. // SecretKey secretKey = secretKeyFactory.generateSecret(pbeKeySpec); // ### old routine
  143. SecretKey secretKeyPbkdf = secretKeyFactory.generateSecret(pbeKeySpec); // ### new
  144. SecretKey secretKey = new SecretKeySpec(secretKeyPbkdf.getEncoded(), &quot;DESede&quot;); // ### new
  145. cipher.init(Cipher.DECRYPT_MODE, secretKey, ivParameterSpec);
  146. keySpec = encryptedPrivateKeyInfo.getKeySpec(cipher);
  147. //PKCS8EncodedKeySpec keySpec = encryptedPrivateKeyInfo.getKeySpec(cipher);
  148. } catch (NoSuchPaddingException | InvalidKeyException | InvalidAlgorithmParameterException e) {
  149. e.printStackTrace();
  150. }
  151. }
  152. Objects.requireNonNull(keySpec);
  153. KeyFactory keyFactory = KeyFactory.getInstance(KEY_FACTORY_ALGORITHM);
  154. return keyFactory.generatePrivate(keySpec);
  155. }
  156. private static KeySpec getKeySpecFromAsn1StructuredData(byte[] decodedPrivateKeyContent) throws IOException {
  157. try(ByteArrayInputStream privateKeyAsInputStream = new ByteArrayInputStream(decodedPrivateKeyContent)) {
  158. ASN1InputStream stream = new ASN1InputStream(new DERDecoder(), privateKeyAsInputStream);
  159. ASN1Sequence asn1Sequence = stream.readObject();
  160. if (asn1Sequence.getValue().size() &lt; 9) {
  161. throw new RuntimeException(&quot;Parsed key content doesn&#39;t have the minimum required sequence size of 9&quot;);
  162. }
  163. BigInteger modulus = extractIntValueFrom(asn1Sequence.get(1));
  164. BigInteger publicExponent = extractIntValueFrom(asn1Sequence.get(2));
  165. BigInteger privateExponent = extractIntValueFrom(asn1Sequence.get(3));
  166. BigInteger primeP = extractIntValueFrom(asn1Sequence.get(4));
  167. BigInteger primeQ = extractIntValueFrom(asn1Sequence.get(5));
  168. BigInteger primeExponentP = extractIntValueFrom(asn1Sequence.get(6));
  169. BigInteger primeExponentQ = extractIntValueFrom(asn1Sequence.get(7));
  170. BigInteger crtCoefficient = extractIntValueFrom(asn1Sequence.get(8));
  171. return new RSAPrivateCrtKeySpec(
  172. modulus, publicExponent, privateExponent,
  173. primeP, primeQ, primeExponentP, primeExponentQ, crtCoefficient
  174. );
  175. }
  176. }
  177. private static BigInteger extractIntValueFrom(ASN1Object&lt;?&gt; asn1Object) {
  178. if (asn1Object instanceof ASN1Integer) {
  179. return ((ASN1Integer) asn1Object).getValue();
  180. } else {
  181. throw new RuntimeException(String.format(
  182. &quot;Unable to parse the provided value of the object type [%s]. The type should be an instance of [%s]&quot;,
  183. asn1Object.getClass().getName(), ASN1Integer.class.getName())
  184. );
  185. }
  186. }
  187. private static Map&lt;String, Certificate&gt; parseCertificate(String certificateContent) throws IOException, CertificateException {
  188. Map&lt;String, Certificate&gt; certificates = new HashMap&lt;&gt;();
  189. Matcher certificateMatcher = CERTIFICATE_PATTERN.matcher(certificateContent);
  190. while (certificateMatcher.find()) {
  191. String sanitizedCertificate = certificateMatcher.group(1).replace(NEW_LINE, EMPTY).trim();
  192. byte[] decodedCertificate = Base64.getDecoder().decode(sanitizedCertificate);
  193. try(ByteArrayInputStream certificateAsInputStream = new ByteArrayInputStream(decodedCertificate)) {
  194. CertificateFactory certificateFactory = CertificateFactory.getInstance(CERTIFICATE_TYPE);
  195. Certificate certificate = certificateFactory.generateCertificate(certificateAsInputStream);
  196. certificates.put(UUID.randomUUID().toString(), certificate);
  197. }
  198. }
  199. return certificates;
  200. }
  201. }

答案3

得分: 1

  1. import javax.crypto.EncryptedPrivateKeyInfo;
  2. import javax.crypto.SecretKey;
  3. import javax.crypto.SecretKeyFactory;
  4. import javax.crypto.spec.PBEKeySpec;
  5. import java.io.IOException;
  6. import java.nio.file.Files;
  7. import java.nio.file.Paths;
  8. import java.security.*;
  9. import java.security.spec.InvalidKeySpecException;
  10. import java.security.spec.PKCS8EncodedKeySpec;
  11. import java.util.Base64;
  12. public class ReadPkcs8PrivateKeysSo {
  13. public static void main(String[] args) throws IOException, NoSuchAlgorithmException, InvalidKeySpecException, InvalidKeyException {
  14. String encrypted = new String(Files.readAllBytes(Paths.get("so_privatekey_3des_secret.pem")));
  15. String password = "secret";
  16. // Create object from encrypted private key
  17. encrypted = encrypted.replace("-----BEGIN ENCRYPTED PRIVATE KEY-----", "");
  18. encrypted = encrypted.replace("-----END ENCRYPTED PRIVATE KEY-----", "");
  19. EncryptedPrivateKeyInfo pkInfo = new EncryptedPrivateKeyInfo(Base64.getMimeDecoder().decode(encrypted));
  20. PBEKeySpec keySpec = new PBEKeySpec(password.toCharArray()); // password
  21. System.out.println("pkInfo.getAlgName: " + pkInfo.getAlgName());
  22. SecretKeyFactory pbeKeyFactory = SecretKeyFactory.getInstance(pkInfo.getAlgName());
  23. SecretKey secretKey = pbeKeyFactory.generateSecret(keySpec);
  24. PKCS8EncodedKeySpec encodedKeySpec = pkInfo.getKeySpec(secretKey);
  25. KeyFactory keyFactory = KeyFactory.getInstance("RSA");
  26. PrivateKey decryptedPrivateKey = keyFactory.generatePrivate(encodedKeySpec);
  27. System.out.println("decryptedPrivateKey: " + decryptedPrivateKey);
  28. }
  29. }
英文:

If you don't be shy to convert your given encrypted private key prior with openssl you can use the sample code for easy parsing.

To run my example I saved your Encrypted Private Key to a file 'so_privatekey_secret.pem'.

Using this comand line in openssl I converted the (already) encrypted key with another cipher and generated the (new) keyfile 'so_privatekey_3des_secret.pem':

  1. openssl pkcs8 -v1 PBE-SHA1-3DES -topk8 -in so_privatekey_secret.pem -out so_privatekey_3des_secret.pem

Your encrypted private key will look like:

  1. -----BEGIN ENCRYPTED PRIVATE KEY-----
  2. MIIE6jAcBgoqhkiG9w0BDAEDMA4ECAdT0CV4sPOPAgIIAASCBMi3H3WjyWEiAS/G
  3. gw6gZTHiZFmidehU8XPh1HPYVb385u36WO16CwZVIBperQoXptCaE2LRwxmrN7T/
  4. K2l59EHlV8i8FAfGKqvc7b1DW1fqx7jvw+/iT//3uNc3GHUNXfBrwHDi/MTghII4
  5. gEFNUm7ZO7rAqeR7pJnZxm3giCvRHFbFcZRkz6LhCif3eYF9Lx3gSe2x6SPeNG/T
  6. p0vrierxhHyCJk+S543J3cZ4vHUOjYzOsT1xz9hTbxAxeAEPkaCpD0P9jyth8M8k
  7. 2SRedRvR1Ia6r7sAfFz6XoHz5jH/lhd47EYpOc/A7rsgWkFLObwdFwGLRIvQsgeN
  8. VXnwg5UoxRdjE3ZXQMFgHgzc4igmoVCjwgNZW6nAMwlcTctjsQyvRnAEnQbqLUw1
  9. kddkjDzC1LFk+NiZ7L9oKVWAQuAQnLHMOxr8kJrTlSoS8zt5x5OEnR4I7JnUEHqs
  10. 2RythkJnoAsXaslBIkIXw4Kg6wisZjr/fL7ER0D0TBdZFPMvdMw8p4TxNxEF7v/K
  11. G97HmCe39cVl3DkYayK2Bs4T6FxqbqUzUMOQnNHpbj82+ybEhm/5duhD5Kd/Inz7
  12. 8mTpDM08P7OMG27AiJDReQJ18XQ5y3DxQJu8UgfQEtjpVW3BQIVzYRM4JqEEv72h
  13. YLELhtmBQom1SWadMW6jzRJRS+f+ivEApVyLcKlnPj2SY7niSmDjYChDx30fHoDt
  14. z25UYBb8LmZ6P5rwNpMgGutBYqg721ioYt86mQsDDaLYOxjlhPfwbrQ9HqUrPCyQ
  15. 0UBeP9hNYkGbZQ+erZmyJ4hhlGGXz9v1LZ4xqNelKuUUREZaLWNVO7UwX2AgUebv
  16. V7V3ne3ZNuLo5kDQqGz3hxwI4tcHgneNMpbxwdR7/B9QTXtirgZck9QIU6zxPhIf
  17. agyCfpqE5pE3ln5n+FfbOik1pNcQxFr0nLy6vnlcTATLHftX0JycFc9OcJ9eRnI/
  18. zk0ekYcw7QLxzNp/ews+Krwy+2SXaacTi+fXihCks10FZ/TEMYV0wC7OMW1yC35K
  19. jUU12u7kSccvUk4xQftyA39XYCN6g7kxDqLjCfc6pMMWZZ9anSo7rmuQ6scoMhdr
  20. 904PcX3ne9rBJLv3a3W25TolNKZx6JofCViyIipcJY4FRyIRoIfwUTZ643KLSZ4o
  21. nAPxtWHpbgA8ucrMKwQ0kbhT/qEvz3HxliL91N7qqMfZ3RfsgK2fUXQZ4zsU1oZ2
  22. f7UT9+D0w/Pusond7SHPFo9G68GjsE48CQ1J/l5QreZ+bLxBohs2xM1WIz2TBJX9
  23. ghdbo+0eatoxfq0o/H0vj66xMd8fvqePrzNw3kSqSpdQiZl95vqWM8Y7xEW0PxAU
  24. 78I0koI98v/3n9abSUNDhodPsk8I3abr7MoQsKe7VSu/x+mgMcfvNOiH6j6LuUDx
  25. Eii7nfHhC/M69hWoh8Y+a2qPdbT2WMZSgXdBC+dCQeL4o6+flht6/24Eg+HAQMyY
  26. 3X5aH2ZM4BJAyJvdtj7btMDKq6nunyMywYhIOgKTZUg8wytxDsi2F4KyNKpXXgsj
  27. 0yR2L6EUYeQ1ZLG/NcM86sHueF1x38RznpyD4rujRGslWAKRXk4D2O2GpYS4rYRk
  28. 4r4pPXltZ472q7PGBfM=
  29. -----END ENCRYPTED PRIVATE KEY-----

Running my sample program (that does have no exception handling) gives an output:

  1. pkInfo.getAlgName: PBEWithSHA1AndDESede
  2. decryptedPrivateKey: SunRsaSign RSA private CRT key, 2048 bits
  3. params: null
  4. modulus: 18998199686208613730227071552042680566660281525162377778227794768879762338654464000476266705180956908841709177555966832423085875590556580445428066261332223955361646420713402269532603328557674453787190302285379571808527818182992417461094551685532849252957724361115461909983939654809533488667032982867230996411860328826589806804115764031480737011900431983025490936659008858316889367140757195359233313370592669605576636360123189540118611655293825465309125578750878878980002346272931570343463549347548711194000065166346537274382220211242975801253306067862461345994215674963367480607539333468674983712264803816027753559737
  5. private exponent: 2470690692670644289832084636740463653655882622624760758103597888447170520657826825702415751838767348735186037205082706146786714644526202528897926495648786865000645626802269367582073915029633022103975204027927584273373553784482725392082470421529769049284506228997163157587212073758970565083984956787424047216319298607371397767542633302071090323391531162434678662485530085857362259661656308472034152915312425224731842425134593468185574918070451135641322780271191791839285884885643517240131520070881951542294552185519522325857178404160441369354693465035929601010771762928436963178293113507661955562266846499005032897533

code:

  1. import javax.crypto.EncryptedPrivateKeyInfo;
  2. import javax.crypto.SecretKey;
  3. import javax.crypto.SecretKeyFactory;
  4. import javax.crypto.spec.PBEKeySpec;
  5. import java.io.IOException;
  6. import java.nio.file.Files;
  7. import java.nio.file.Paths;
  8. import java.security.*;
  9. import java.security.spec.InvalidKeySpecException;
  10. import java.security.spec.PKCS8EncodedKeySpec;
  11. import java.util.Base64;
  12. public class ReadPkcs8PrivateKeysSo {
  13. public static void main(String[] args) throws IOException, NoSuchAlgorithmException, InvalidKeySpecException, InvalidKeyException {
  14. String encrypted = new String(Files.readAllBytes(Paths.get(&quot;so_privatekey_3des_secret.pem&quot;)));
  15. String password = &quot;secret&quot;;
  16. //Create object from encrypted private key
  17. encrypted = encrypted.replace(&quot;-----BEGIN ENCRYPTED PRIVATE KEY-----&quot;, &quot;&quot;);
  18. encrypted = encrypted.replace(&quot;-----END ENCRYPTED PRIVATE KEY-----&quot;, &quot;&quot;);
  19. EncryptedPrivateKeyInfo pkInfo = new EncryptedPrivateKeyInfo(Base64.getMimeDecoder().decode(encrypted));
  20. PBEKeySpec keySpec = new PBEKeySpec(password.toCharArray()); // password
  21. System.out.println(&quot;pkInfo.getAlgName: &quot; + pkInfo.getAlgName());
  22. SecretKeyFactory pbeKeyFactory = SecretKeyFactory.getInstance(pkInfo.getAlgName());
  23. SecretKey secretKey = pbeKeyFactory.generateSecret(keySpec);
  24. PKCS8EncodedKeySpec encodedKeySpec = pkInfo.getKeySpec(secretKey);
  25. KeyFactory keyFactory = KeyFactory.getInstance(&quot;RSA&quot;);
  26. PrivateKey decryptedPrivateKey = keyFactory.generatePrivate(encodedKeySpec);
  27. System.out.println(&quot;decryptedPrivateKey: &quot; + decryptedPrivateKey);
  28. }
  29. }

答案4

得分: 1

我花了几个星期的时间在我的pem-utils项目上工作,现在我拥有了纯Java代码,没有使用任何外部库,似乎可以加载我投入其中的任何内容。

我希望这能帮助原始发布者(在18个月前!),以及其他跟随我们脚步的人。

英文:

I spent a number of weeks working on my pem-utils project on GitHub and I now have pure-Java code with no external libraries that seems to be able to load anything I throw at it.

I hope this helps the original poster (from 18 months ago!) as well as anyone else following in our footsteps.

答案5

得分: 1

下面是翻译好的内容:

最终我决定使用BC来解析PEM文件中的证书和私钥。在我的使用情况下,我需要处理各种不同类型的私钥,因此复杂性也增加了,但我希望能够尽可能地简化自己和其他人的操作,因此我将其封装成了一个库。下面是一个示例用法。该库在这里可用:GitHub - SSLContext Kickstart

  1. import javax.net.ssl.SSLContext;
  2. import java.security.PrivateKey;
  3. import java.security.cert.X509Certificate;
  4. import java.util.List;
  5. import nl.altindag.ssl.SSLFactory;
  6. import nl.altindag.ssl.util.PemUtils;
  7. class App {
  8. public static void main(String[] args) {
  9. PrivateKey privateKey = PemUtils.parsePrivateKey(privateKeyContent, privateKeyPassword);
  10. List<X509Certificate> certificateChain = PemUtils.parseCertificate(certificateChainContent);
  11. SSLFactory sslFactory = SSLFactory.builder()
  12. .withIdentityMaterial(privateKey, privateKeyPassword, certificateChain)
  13. .build();
  14. SSLContext sslContext = sslFactory.getSslContext();
  15. }
  16. private static char[] privateKeyPassword = "secret".toCharArray();
  17. private static String privateKeyContent = "..."; // 省略私钥内容
  18. private static String certificateChainContent = "..."; // 省略证书链内容
  19. }
英文:

In the end I decided to use BC to parse the PEM files for certificates and private keys. In my use case I needed to handle all different kind of private keys and therefor the complexity also grew, but I wanted to make it as easy as possible for my self and others, so I simplified it by creating a library out of it, see below for an example usage. The library is available here: GitHub - SSLContext Kickstart

  1. import javax.net.ssl.SSLContext;
  2. import java.security.PrivateKey;
  3. import java.security.cert.X509Certificate;
  4. import java.util.List;
  5. import nl.altindag.ssl.SSLFactory;
  6. import nl.altindag.ssl.util.PemUtils;
  7. class App {
  8. public static void main(String[] args) {
  9. PrivateKey privateKey = PemUtils.parsePrivateKey(privateKeyContent, privateKeyPassword);
  10. List&lt;X509Certificate&gt; certificateChain = PemUtils.parseCertificate(certificateChainContent);
  11. SSLFactory sslFactory = SSLFactory.builder()
  12. .withIdentityMaterial(privateKey, privateKeyPassword, certificateChain)
  13. .build();
  14. SSLContext sslContext = sslFactory.getSslContext();
  15. }
  16. private static char[] privateKeyPassword = &quot;secret&quot;.toCharArray();
  17. private static String privateKeyContent = &quot;&quot;&quot;
  18. -----BEGIN ENCRYPTED PRIVATE KEY-----
  19. MIIFDjBABgkqhkiG9w0BBQ0wMzAbBgkqhkiG9w0BBQwwDgQI9/FSBonUacYCAggA
  20. MBQGCCqGSIb3DQMHBAidGkS8wYhOpwSCBMi8JaSYOKudMNNFRpzL7QMIZgFtzDay
  21. MmOroy3lW34dOa7dusqDl4d2gklKcHCpbEaTYxm5aQJ1LuiOdGtFy7HwxOvKU5xz
  22. 4qsJoeBIpE0eCTKjQW7/I38DzLXx2wUURqhMWOtFsWZEyR5Dqok3N9mIKKKBXAFG
  23. AwNjlTRW2LyPSttiIUGN01lthjifMWoLTWB1aSGOmGeJRBdSZeqZ15xKneR4H5ja
  24. yE88YcpOHCDKMIxi6ZVoKs7jDQhu8bBKqS8NsYyh1AlP9QkvWNal36jWSzhqYNzk
  25. NRWUOZngfkdbMALVfRtbrC215jHGWVwosPIIs8rkoarRv8s6QWS1Rg3YfQ3qgcRf
  26. s7hkDFKJf3TUXr+askfamV5hc300ZG64+ldX1YxWXY8Vd/wIvHAc/YE/lTyCgYrY
  27. 19Am6MNBfp8/kXvzKj+PizB8oNDO4S8sSShEEzOQ5a/+MTC6bqB0DLWYGUqRbjLc
  28. PyYTC2C4i9Agx/GeGVE3c1UdtXiwwnt2XUn7Y1YGqABk0xGIY4J1NFTbSOxKl9hO
  29. arwopAFrZU5nsjjFzv1DJvhfQWnYX18kPSKNHDlia019M118qZ8ERwD9tH8ix9Fa
  30. R2tQdxn1aRGmvXSw+zFkbWD8aWs9n/B+QN1yllJqVoWypOld1yj+fVYYnYOtV1gK
  31. eiygrtrh3JJCvLbEQl4nOgJM3PlEtfBHSaunehIXQMD1z/NDUqgBYjuDPyqRxJeH
  32. Va5k72Nds5PeySKJJnICB3nZKjqgfLhNUrXa1SAQ4vqr0Ik/Lu9P7T+B1XiYwuUT
  33. a20+bxi/x89ZZqwp3jnDuHup7XcO1MtqsoOKP/JgkjVMesb8Q1W8i2dXzg+l4gkk
  34. l1ipreEGtT1YfFTq0DFelz6CjZFLDlGGeGWob94sW94DWTW0nsLPhQWEnwW1CcyJ
  35. oJbJdDEgdiIbRJoABDkTuVXLwTlgzHSHh6zeJvNvcojI7UI3nWYCVYvD3kwghXiP
  36. 67sKGL3ug7PFDqLia46AudGY7CFh4+wpxyH+fidLC3FMdkDBA6xR6mGgEjRLXR9M
  37. TnJ/eSYP7eqYZeKn9EarcI7v1zM2IG0/PDQCetiI0ABiHpdKyRQuuiEavp3xC5Vi
  38. h7UmJNYt8Zsz3rwqAQ4FR2+Su5R34OOdRmxTaYLe96PXTpLcLef5TkYixSY7Tzgd
  39. PMyRxRPrywklUEFe4KK/KOcdolxybfsIsxQnupLAMEsO7/Cs7mouNHISK51haDRc
  40. vNbKQ5E4xOq1U4ThW5dHR29cGZillfmMzj05ZQh3ZX2TQJP45ahFET3v9kInWCwQ
  41. 8atqclVPOSnASsJZ0PxjYgKZuY8QWYM6zpfWyWnfu/CHhWbRS/qX8T1ow2SMyPBL
  42. CQbZ+MhcdP0IrjoXhDFQsns16i/BPK5TTVqtEC2ywDf5P4/BOEZkySG9YNOd6THp
  43. VA/dVPafzmLy3ltqH+jG8ZH2+RtWx7kwBjiDWs5cF33BFrPS7AZlzMzZoCHLXD/r
  44. T/SmisybUKHMqri0x0RHeIByW0hogSByWiyIn8POabDzJV6Df9nQPziDGcSsvWfG
  45. 7q+hizh6+nnXOY+GZx3ptwg9mA9R4QyCiFNQradOaXSPxyEL2IC77/srFfVEIaU4
  46. SRo=
  47. -----END ENCRYPTED PRIVATE KEY-----
  48. &quot;&quot;&quot;;
  49. private static String certificateChainContent = &quot;&quot;&quot;
  50. -----BEGIN CERTIFICATE-----
  51. MIIDTjCCAjagAwIBAgIEVnHI3TANBgkqhkiG9w0BAQsFADBIMQswCQYDVQQGEwJO
  52. TDEVMBMGA1UEChMMVGh1bmRlcmJlcnJ5MRIwEAYDVQQLEwlBbXN0ZXJkYW0xDjAM
  53. BgNVBAMTBUhha2FuMB4XDTIwMDgzMTA4MDczOVoXDTMwMDgyOTA4MDczOVowSDEL
  54. MAkGA1UEBhMCTkwxFTATBgNVBAoTDFRodW5kZXJiZXJyeTESMBAGA1UECxMJQW1z
  55. dGVyZGFtMQ4wDAYDVQQDEwVIYWthbjCCASIwDQYJKoZIhvcNAQEBBQADggEPADCC
  56. AQoCggEBAJZ+pqirEqEk4k1ow8vryld79oCO4P/1y+v60CkLpS2MpeHE3BogTr7g
  57. WWP5HdHBMU5l8yT5tXuyVZZqgyzo0q2Sdm7GGrNunHf6Z8vPQFu69sQC1utDM9u8
  58. ZFKiKsTNJ+5QS6KsOtlACyhaLoaPAWVjtvueMjVwmM9hfv/Gq6VyyuBm2x1C4HTj
  59. zPCLHE2G1D13EJaWsvyZLGSbl0GGXZGPhaDd/vnw5TW36mvNTWW+37ZIEk4kXANa
  60. FUNsJemm8HjB/PzHs3/SXGxuD8NKobg3+cNXYwAz2s2DI0W6Xw2g5bbrMQAdBRvn
  61. 9/kNftMymDORw3RGwDM2ld4zQfIkNrkCAwEAAaNAMD4wHQYDVR0OBBYEFHhT7ATg
  62. oVVFIsglxD/1iUBRB6gDMB0GA1UdEQEB/wQTMBGCCWxvY2FsaG9zdIcEfwAAATAN
  63. BgkqhkiG9w0BAQsFAAOCAQEAhVWH/CgZ0ZNmTfiFAOnWdaZVaa7vAFPT2YbXuvlY
  64. YIRlru0B/zn2Mfwmn5W2o1CqoBuhyfErkkF4aRM1vduIirUjlcH4+cFXrV2gtlnf
  65. eWTg/sJJmYzkJTGeOIqRlB1HKCqoeNCrykkcsikECQ1nCqr1qLh9DXsUgWVW57YW
  66. qvP1P8xOO2/J9shMB6lOhftpawrqZ2hNG8fqMKjVVuUpFBNR+WODQ/rRRtqa6uU2
  67. V8aOOZx1QJUkTdN76YOCuGET7edevjpdbRXde+HQN6mbT9OLxSZHO0aQrDyDmNhp
  68. aVHuQn/KtYNWCZ78XKK8wtVnflmfqE/c9xO1n/EcVvLCdg==
  69. -----END CERTIFICATE-----
  70. &quot;&quot;&quot;;
  71. }

huangapple
  • 本文由 发表于 2020年9月10日 23:00:06
  • 转载请务必保留本文链接:https://go.coder-hub.com/63832456.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定