diff --git a/bom/application/pom.xml b/bom/application/pom.xml index 8414ce058191b..7f17278bbe353 100644 --- a/bom/application/pom.xml +++ b/bom/application/pom.xml @@ -337,6 +337,12 @@ ${smallrye-certificate-generator.version} + + io.smallrye.certs + smallrye-private-key-pem-parser + ${smallrye-certificate-generator.version} + + com.fasterxml.jackson diff --git a/docs/src/main/asciidoc/tls-registry-reference.adoc b/docs/src/main/asciidoc/tls-registry-reference.adoc index 2e7322e56b72f..d9f9fe505f84d 100644 --- a/docs/src/main/asciidoc/tls-registry-reference.adoc +++ b/docs/src/main/asciidoc/tls-registry-reference.adoc @@ -228,6 +228,21 @@ For example, `quarkus.tls.key-store.pem.order=b,c,a`. This setting is important when using SNI, because it uses the first specified pair as the default. ==== +When using PEM keystore, the following formats are supported: + +- PKCS#8 private key (unencrypted) +- PKCS#1 RSA private key (unencrypted) +- Encrypted PKCS#8 private key (encrypted with AES-128-CBC) + +In the later case, the `quarkus.tls.key-store.pem.password` (or `quarkus.tls.key-store.pem..password`) property must be set to the password used to decrypt the private key: + +[source,properties] +---- +quarkus.tls.http.key-store.pem.cert=certificate.crt +quarkus.tls.http.key-store.pem.key=key.key +quarkus.tls.http.key-store.pem.password=password +---- + ==== PKCS12 keystores PKCS12 keystores are single files that contain the certificate and the private key. diff --git a/extensions/tls-registry/cli/src/main/java/io/quarkus/tls/cli/letsencrypt/LetsEncryptHelpers.java b/extensions/tls-registry/cli/src/main/java/io/quarkus/tls/cli/letsencrypt/LetsEncryptHelpers.java index be91516fa9916..43d7188717b00 100644 --- a/extensions/tls-registry/cli/src/main/java/io/quarkus/tls/cli/letsencrypt/LetsEncryptHelpers.java +++ b/extensions/tls-registry/cli/src/main/java/io/quarkus/tls/cli/letsencrypt/LetsEncryptHelpers.java @@ -44,7 +44,7 @@ public static void writePrivateKeyAndCertificateChainsAsPem(PrivateKey pk, X509C throw new IllegalArgumentException("The certificate chain cannot be null or empty"); } - CertificateUtils.writePrivateKeyToPem(pk, privateKeyFile); + CertificateUtils.writePrivateKeyToPem(pk, null, privateKeyFile); if (chain.length == 1) { CertificateUtils.writeCertificateToPEM(chain[0], certificateChainFile); diff --git a/extensions/tls-registry/deployment/src/test/java/io/quarkus/tls/DefaultEncryptedPemKeyStoreTest.java b/extensions/tls-registry/deployment/src/test/java/io/quarkus/tls/DefaultEncryptedPemKeyStoreTest.java new file mode 100644 index 0000000000000..c73679a177791 --- /dev/null +++ b/extensions/tls-registry/deployment/src/test/java/io/quarkus/tls/DefaultEncryptedPemKeyStoreTest.java @@ -0,0 +1,58 @@ +package io.quarkus.tls; + +import static org.assertj.core.api.Assertions.assertThat; + +import java.security.KeyStoreException; +import java.security.cert.CertificateParsingException; +import java.security.cert.X509Certificate; + +import jakarta.inject.Inject; + +import org.jboss.shrinkwrap.api.ShrinkWrap; +import org.jboss.shrinkwrap.api.asset.StringAsset; +import org.jboss.shrinkwrap.api.spec.JavaArchive; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +import io.quarkus.test.QuarkusUnitTest; +import io.smallrye.certs.Format; +import io.smallrye.certs.junit5.Certificate; +import io.smallrye.certs.junit5.Certificates; + +@Certificates(baseDir = "target/certs", certificates = { + @Certificate(name = "test-formats-encrypted-pem", password = "password", formats = { Format.JKS, Format.ENCRYPTED_PEM, + Format.PKCS12 }) +}) +public class DefaultEncryptedPemKeyStoreTest { + + private static final String configuration = """ + quarkus.tls.key-store.pem.foo.cert=target/certs/test-formats-encrypted-pem.crt + quarkus.tls.key-store.pem.foo.key=target/certs/test-formats-encrypted-pem.key + quarkus.tls.key-store.pem.foo.password=password + """; + + @RegisterExtension + static final QuarkusUnitTest config = new QuarkusUnitTest().setArchiveProducer( + () -> ShrinkWrap.create(JavaArchive.class) + .add(new StringAsset(configuration), "application.properties")); + + @Inject + TlsConfigurationRegistry certificates; + + @Test + void test() throws KeyStoreException, CertificateParsingException { + TlsConfiguration def = certificates.getDefault().orElseThrow(); + + assertThat(def.getKeyStoreOptions()).isNotNull(); + assertThat(def.getKeyStore()).isNotNull(); + + // dummy-entry-x is the alias of the certificate in the keystore generated by Vert.x. + + X509Certificate certificate = (X509Certificate) def.getKeyStore().getCertificate("dummy-entry-0"); + assertThat(certificate).isNotNull(); + assertThat(certificate.getSubjectAlternativeNames()).anySatisfy(l -> { + assertThat(l.get(0)).isEqualTo(2); + assertThat(l.get(1)).isEqualTo("localhost"); + }); + } +} diff --git a/extensions/tls-registry/deployment/src/test/java/io/quarkus/tls/EncryptedPemWithNoPasswordTest.java b/extensions/tls-registry/deployment/src/test/java/io/quarkus/tls/EncryptedPemWithNoPasswordTest.java new file mode 100644 index 0000000000000..b254644cfd78a --- /dev/null +++ b/extensions/tls-registry/deployment/src/test/java/io/quarkus/tls/EncryptedPemWithNoPasswordTest.java @@ -0,0 +1,41 @@ +package io.quarkus.tls; + +import static org.assertj.core.api.Assertions.assertThat; +import static org.assertj.core.api.Assertions.fail; + +import java.security.KeyStoreException; +import java.security.cert.CertificateParsingException; + +import org.jboss.shrinkwrap.api.ShrinkWrap; +import org.jboss.shrinkwrap.api.asset.StringAsset; +import org.jboss.shrinkwrap.api.spec.JavaArchive; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +import io.quarkus.test.QuarkusUnitTest; +import io.smallrye.certs.Format; +import io.smallrye.certs.junit5.Certificate; +import io.smallrye.certs.junit5.Certificates; + +@Certificates(baseDir = "target/certs", certificates = { + @Certificate(name = "test-formats-encrypted-pem", password = "password", formats = { Format.JKS, Format.ENCRYPTED_PEM, + Format.PKCS12 }) +}) +public class EncryptedPemWithNoPasswordTest { + + private static final String configuration = """ + quarkus.tls.key-store.pem.foo.cert=target/certs/test-formats-encrypted-pem.crt + quarkus.tls.key-store.pem.foo.key=target/certs/test-formats-encrypted-pem.key + """; + + @RegisterExtension + static final QuarkusUnitTest config = new QuarkusUnitTest().setArchiveProducer( + () -> ShrinkWrap.create(JavaArchive.class) + .add(new StringAsset(configuration), "application.properties")) + .assertException(t -> assertThat(t.getMessage()).contains("key/certificate pair", "default")); + + @Test + void test() throws KeyStoreException, CertificateParsingException { + fail("Should not be called as the extension should fail before."); + } +} diff --git a/extensions/tls-registry/deployment/src/test/java/io/quarkus/tls/EncryptedPemWithWrongPasswordTest.java b/extensions/tls-registry/deployment/src/test/java/io/quarkus/tls/EncryptedPemWithWrongPasswordTest.java new file mode 100644 index 0000000000000..34e027fb030ce --- /dev/null +++ b/extensions/tls-registry/deployment/src/test/java/io/quarkus/tls/EncryptedPemWithWrongPasswordTest.java @@ -0,0 +1,42 @@ +package io.quarkus.tls; + +import static org.assertj.core.api.Assertions.assertThat; +import static org.assertj.core.api.Assertions.fail; + +import java.security.KeyStoreException; +import java.security.cert.CertificateParsingException; + +import org.jboss.shrinkwrap.api.ShrinkWrap; +import org.jboss.shrinkwrap.api.asset.StringAsset; +import org.jboss.shrinkwrap.api.spec.JavaArchive; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +import io.quarkus.test.QuarkusUnitTest; +import io.smallrye.certs.Format; +import io.smallrye.certs.junit5.Certificate; +import io.smallrye.certs.junit5.Certificates; + +@Certificates(baseDir = "target/certs", certificates = { + @Certificate(name = "test-formats-encrypted-pem", password = "password", formats = { Format.JKS, Format.ENCRYPTED_PEM, + Format.PKCS12 }) +}) +public class EncryptedPemWithWrongPasswordTest { + + private static final String configuration = """ + quarkus.tls.key-store.pem.foo.cert=target/certs/test-formats-encrypted-pem.crt + quarkus.tls.key-store.pem.foo.key=target/certs/test-formats-encrypted-pem.key + quarkus.tls.key-store.pem.foo.password=wrong + """; + + @RegisterExtension + static final QuarkusUnitTest config = new QuarkusUnitTest().setArchiveProducer( + () -> ShrinkWrap.create(JavaArchive.class) + .add(new StringAsset(configuration), "application.properties")) + .assertException(t -> assertThat(t.getMessage()).contains("key/certificate pair", "default")); + + @Test + void test() throws KeyStoreException, CertificateParsingException { + fail("Should not be called as the extension should fail before."); + } +} diff --git a/extensions/tls-registry/deployment/src/test/java/io/quarkus/tls/NamedEncryptedPemKeyStoreTest.java b/extensions/tls-registry/deployment/src/test/java/io/quarkus/tls/NamedEncryptedPemKeyStoreTest.java new file mode 100644 index 0000000000000..d3c199bac8c8e --- /dev/null +++ b/extensions/tls-registry/deployment/src/test/java/io/quarkus/tls/NamedEncryptedPemKeyStoreTest.java @@ -0,0 +1,60 @@ +package io.quarkus.tls; + +import static org.assertj.core.api.Assertions.assertThat; + +import java.security.KeyStoreException; +import java.security.cert.CertificateParsingException; +import java.security.cert.X509Certificate; + +import jakarta.inject.Inject; + +import org.jboss.shrinkwrap.api.ShrinkWrap; +import org.jboss.shrinkwrap.api.asset.StringAsset; +import org.jboss.shrinkwrap.api.spec.JavaArchive; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.RegisterExtension; + +import io.quarkus.test.QuarkusUnitTest; +import io.smallrye.certs.Format; +import io.smallrye.certs.junit5.Certificate; +import io.smallrye.certs.junit5.Certificates; + +@Certificates(baseDir = "target/certs", certificates = { + @Certificate(name = "test-formats-encrypted-pem", password = "password", formats = { Format.JKS, Format.ENCRYPTED_PEM, + Format.PKCS12 }) +}) +public class NamedEncryptedPemKeyStoreTest { + + private static final String configuration = """ + quarkus.tls.http.key-store.pem.foo.cert=target/certs/test-formats-encrypted-pem.crt + quarkus.tls.http.key-store.pem.foo.key=target/certs/test-formats-encrypted-pem.key + quarkus.tls.http.key-store.pem.foo.password=password + """; + + @RegisterExtension + static final QuarkusUnitTest config = new QuarkusUnitTest().setArchiveProducer( + () -> ShrinkWrap.create(JavaArchive.class) + .add(new StringAsset(configuration), "application.properties")); + + @Inject + TlsConfigurationRegistry certificates; + + @Test + void test() throws KeyStoreException, CertificateParsingException { + TlsConfiguration def = certificates.getDefault().orElseThrow(); + TlsConfiguration named = certificates.get("http").orElseThrow(); + + assertThat(def.getKeyStoreOptions()).isNull(); + assertThat(def.getKeyStore()).isNull(); + + assertThat(named.getKeyStoreOptions()).isNotNull(); + assertThat(named.getKeyStore()).isNotNull(); + + X509Certificate certificate = (X509Certificate) named.getKeyStore().getCertificate("dummy-entry-0"); + assertThat(certificate).isNotNull(); + assertThat(certificate.getSubjectAlternativeNames()).anySatisfy(l -> { + assertThat(l.get(0)).isEqualTo(2); + assertThat(l.get(1)).isEqualTo("localhost"); + }); + } +} diff --git a/extensions/tls-registry/runtime/pom.xml b/extensions/tls-registry/runtime/pom.xml index 930ce841182e1..b28c0f94577a3 100644 --- a/extensions/tls-registry/runtime/pom.xml +++ b/extensions/tls-registry/runtime/pom.xml @@ -26,6 +26,10 @@ io.quarkus quarkus-credentials + + io.smallrye.certs + smallrye-private-key-pem-parser + diff --git a/extensions/tls-registry/runtime/src/main/java/io/quarkus/tls/runtime/config/PemKeyCertConfig.java b/extensions/tls-registry/runtime/src/main/java/io/quarkus/tls/runtime/config/PemKeyCertConfig.java index 73d2e4608305b..6288e6b233a70 100644 --- a/extensions/tls-registry/runtime/src/main/java/io/quarkus/tls/runtime/config/PemKeyCertConfig.java +++ b/extensions/tls-registry/runtime/src/main/java/io/quarkus/tls/runtime/config/PemKeyCertConfig.java @@ -2,6 +2,7 @@ import static io.quarkus.tls.runtime.config.TlsConfigUtils.read; +import java.nio.charset.StandardCharsets; import java.nio.file.Path; import java.util.ArrayList; import java.util.List; @@ -10,6 +11,7 @@ import java.util.TreeMap; import io.quarkus.runtime.annotations.ConfigGroup; +import io.smallrye.certs.pem.parsers.EncryptedPKCS8Parser; import io.smallrye.config.WithParentName; import io.vertx.core.buffer.Buffer; import io.vertx.core.net.PemKeyCertOptions; @@ -62,7 +64,17 @@ default PemKeyCertOptions toOptions() { for (KeyCertConfig config : orderedListOfPair) { options.addCertValue(Buffer.buffer(read(config.cert()))); - options.addKeyValue(Buffer.buffer(read(config.key()))); + if (config.password().isPresent()) { + byte[] content = read(config.key()); + String contentAsString = new String(content, StandardCharsets.UTF_8); + Buffer decrypted = new EncryptedPKCS8Parser().decryptKey(contentAsString, config.password().get()); + if (decrypted == null) { + throw new IllegalArgumentException("Unable to decrypt the key file: " + config.key()); + } + options.addKeyValue(decrypted); + } else { + options.addKeyValue(Buffer.buffer(read(config.key()))); + } } return options; } @@ -70,7 +82,7 @@ default PemKeyCertOptions toOptions() { interface KeyCertConfig { /** - * The path to the key file (in PEM format). + * The path to the key file (in PEM format: PKCS#8, PKCS#1 or encrypted PKCS#8). */ Path key(); @@ -78,6 +90,11 @@ interface KeyCertConfig { * The path to the certificate file (in PEM format). */ Path cert(); + + /** + * When the key is encrypted (encrypted PKCS#8), the password to decrypt it. + */ + Optional password(); } } diff --git a/pom.xml b/pom.xml index 94c31a88da386..21e15f2a56d1a 100644 --- a/pom.xml +++ b/pom.xml @@ -87,7 +87,7 @@ 2.46.0 - 0.8.1 + 0.9.2 7.8.0