diff --git a/bom/application/pom.xml b/bom/application/pom.xml index fb54502ae14b7..3168e48b4a206 100644 --- a/bom/application/pom.xml +++ b/bom/application/pom.xml @@ -47,7 +47,7 @@ 2.0 4.0.2 2.8.0 - 3.10.2 + 3.10.3-SNAPSHOT 4.1.0 4.0.0 4.0.3 diff --git a/extensions/hibernate-validator/deployment/src/test/java/io/quarkus/hibernate/validator/test/config/ConfigMappingInvalidTest.java b/extensions/hibernate-validator/deployment/src/test/java/io/quarkus/hibernate/validator/test/config/ConfigMappingInvalidTest.java index 696b7a2dc61f9..cedf4a2bc434c 100644 --- a/extensions/hibernate-validator/deployment/src/test/java/io/quarkus/hibernate/validator/test/config/ConfigMappingInvalidTest.java +++ b/extensions/hibernate-validator/deployment/src/test/java/io/quarkus/hibernate/validator/test/config/ConfigMappingInvalidTest.java @@ -1,56 +1,76 @@ package io.quarkus.hibernate.validator.test.config; -import static org.junit.jupiter.api.Assertions.assertThrows; - +import static org.junit.jupiter.api.Assertions.assertEquals; +import static org.junit.jupiter.api.Assertions.assertTrue; +import static org.junit.jupiter.api.Assertions.fail; + +import java.lang.annotation.ElementType; +import java.lang.annotation.Retention; +import java.lang.annotation.RetentionPolicy; +import java.lang.annotation.Target; +import java.time.Duration; +import java.time.Period; import java.util.List; - -import jakarta.inject.Inject; +import java.util.Map; +import java.util.Optional; +import java.util.OptionalInt; +import java.util.function.Consumer; + +import jakarta.validation.Constraint; +import jakarta.validation.ConstraintValidator; +import jakarta.validation.ConstraintValidatorContext; +import jakarta.validation.Payload; import jakarta.validation.constraints.Max; import jakarta.validation.constraints.Min; import jakarta.validation.constraints.NotEmpty; import jakarta.validation.constraints.Size; 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.Disabled; import org.junit.jupiter.api.Test; import org.junit.jupiter.api.extension.RegisterExtension; import io.quarkus.arc.Unremovable; +import io.quarkus.runtime.configuration.DurationConverter; import io.quarkus.test.QuarkusUnitTest; import io.smallrye.config.ConfigMapping; import io.smallrye.config.ConfigValidationException; -import io.smallrye.config.SmallRyeConfig; +import io.smallrye.config.WithConverter; +import io.smallrye.config.WithDefault; +import io.smallrye.config.WithName; +import io.smallrye.config.WithParentName; public class ConfigMappingInvalidTest { @RegisterExtension - static final QuarkusUnitTest UNIT_TEST = new QuarkusUnitTest().setArchiveProducer( + static final QuarkusUnitTest TEST = new QuarkusUnitTest().setArchiveProducer( () -> ShrinkWrap.create(JavaArchive.class) - .addAsResource(new StringAsset("validator.server.host=localhost\n" + - "validator.server.services=redis,postgresql\n" + - "validator.hierarchy.number=1\n" + - "validator.repeatable.name=a"), "application.properties")); - - @Inject - SmallRyeConfig config; + .addAsResource("application-mappings-validation.properties", "application.properties")) + .assertException(new Consumer() { + @Override + public void accept(final Throwable throwable) { + // can't use instanceOf because of different ClassLoaders + assertEquals(ConfigValidationException.class.getName(), throwable.getClass().getName()); + String message = throwable.getMessage(); + assertTrue(message.contains("validator.server.host must be less than or equal to 3")); + assertTrue(message.contains("validator.hierarchy.number must be greater than or equal to 10")); + assertTrue(message.contains("validator.repeatable.name size must be between 2")); + + assertTrue(message.contains("cloud.port must be greater than or equal to 8000")); + assertTrue(message.contains("cloud.log.days must be less than or equal to 15")); + assertTrue(message.contains("cloud.cors.origins[1].port must be greater than or equal to 8000")); + assertTrue(message.contains("cloud.info.name size must be between 0 and 3")); + assertTrue(message.contains("cloud.info.code must be less than or equal to 3")); + assertTrue(message.contains("cloud.info.alias[0] size must be between 0 and 3")); + assertTrue(message.contains("cloud.info.admins.root[1].username size must be between 0 and 4")); + assertTrue(message.contains("cloud.info.firewall.accepted[1] size must be between 8 and 15")); + assertTrue(message.contains("cloud.proxy.timeout must be less than or equal to 10")); + assertTrue(message.contains("cloud server is not prod")); + } + }); @Test void invalid() { - assertThrows(ConfigValidationException.class, () -> config.getConfigMapping(Server.class), - "validator.server.host must be less than or equal to 3"); - } - - @Test - @Disabled("Requires https://github.com/smallrye/smallrye-config/pull/923") - void invalidHierarchy() { - assertThrows(ConfigValidationException.class, () -> config.getConfigMapping(Child.class), - "validator.hierarchy.number must be greater than or equal to 10"); - } - - @Test - void repeatable() { - assertThrows(ConfigValidationException.class, () -> config.getConfigMapping(Repeatable.class)); + fail(); } @Unremovable @@ -80,4 +100,138 @@ public interface Repeatable { @Size(min = 2) String name(); } + + @Unremovable + @ConfigMapping(prefix = "cloud") + @Prod + public interface Cloud { + String host(); + + @Min(8000) + int port(); + + @WithConverter(DurationConverter.class) + Duration timeout(); + + @WithName("io-threads") + int threads(); + + @WithParentName + Map form(); + + Optional ssl(); + + Optional proxy(); + + Optional cors(); + + Log log(); + + Info info(); + + interface Form { + String loginPage(); + + String errorPage(); + + String landingPage(); + + Optional cookie(); + + @WithDefault("1") + List positions(); + } + + interface Ssl { + int port(); + + String certificate(); + + @WithDefault("TLSv1.3,TLSv1.2") + List protocols(); + } + + interface Proxy { + boolean enable(); + + @Max(10) + int timeout(); + } + + interface Log { + @WithDefault("false") + boolean enabled(); + + @WithDefault(".log") + String suffix(); + + @WithDefault("true") + boolean rotate(); + + @WithDefault("COMMON") + Pattern pattern(); + + Period period(); + + @Max(15) + int days(); + + enum Pattern { + COMMON, + SHORT, + COMBINED, + LONG; + } + } + + interface Cors { + List origins(); + + List<@Size(min = 2) String> methods(); + + interface Origin { + @Size(min = 5) + String host(); + + @Min(8000) + int port(); + } + } + + interface Info { + Optional<@Size(max = 3) String> name(); + + @Max(3) + OptionalInt code(); + + Optional> alias(); + + Map> admins(); + + Map> firewall(); + + interface Admin { + @Size(max = 4) + String username(); + } + } + } + + @Target({ ElementType.TYPE_USE, ElementType.FIELD }) + @Retention(RetentionPolicy.RUNTIME) + @Constraint(validatedBy = Prod.Validator.class) + public @interface Prod { + String message() default "server is not prod"; + + Class[] groups() default {}; + + Class[] payload() default {}; + + class Validator implements ConstraintValidator { + @Override + public boolean isValid(final Cloud value, final ConstraintValidatorContext context) { + return value.host().equals("prod"); + } + } + } } diff --git a/extensions/hibernate-validator/deployment/src/test/resources/application-mappings-validation.properties b/extensions/hibernate-validator/deployment/src/test/resources/application-mappings-validation.properties new file mode 100644 index 0000000000000..a94e71916cc37 --- /dev/null +++ b/extensions/hibernate-validator/deployment/src/test/resources/application-mappings-validation.properties @@ -0,0 +1,34 @@ +validator.server.host=localhost +validator.server.services=redis,postgresql +validator.hierarchy.number=1 +validator.repeatable.name=a + +cloud.host=localhost +cloud.port=5000 +cloud.io-threads=200 +cloud.timeout=60s +cloud.form.error-page=error.html +cloud.form.landing-page=index.html +cloud.form.login-page=login.html +cloud.info.admins.root[0].username=root +cloud.info.admins.root[1].username=super +cloud.info.alias=James +cloud.info.alias[0]=James +cloud.info.code=007 +cloud.info.firewall.accepted=127.0.0.1,8.8.8 +cloud.info.firewall.accepted[0]=127.0.0.1 +cloud.info.firewall.accepted[1]=8.8.8 +cloud.info.name=Bond +cloud.log.days=20 +cloud.log.period=P1D +cloud.proxy.enable=true +cloud.proxy.timeout=20 +cloud.ssl.certificate=certificate +cloud.ssl.port=8443 +cloud.cors.methods=GET,POST +cloud.cors.methods[0]=GET +cloud.cors.methods[1]=POST +cloud.cors.origins[0].host=some-server +cloud.cors.origins[0].port=9000 +cloud.cors.origins[1].host=localhost +cloud.cors.origins[1].port=1 diff --git a/integration-tests/smallrye-config/src/main/java/io/quarkus/it/smallrye/config/Cloud.java b/integration-tests/smallrye-config/src/main/java/io/quarkus/it/smallrye/config/Cloud.java deleted file mode 100644 index 74fd78e03b5e5..0000000000000 --- a/integration-tests/smallrye-config/src/main/java/io/quarkus/it/smallrye/config/Cloud.java +++ /dev/null @@ -1,136 +0,0 @@ -package io.quarkus.it.smallrye.config; - -import java.time.Duration; -import java.time.Period; -import java.util.List; -import java.util.Map; -import java.util.Optional; -import java.util.OptionalInt; - -import jakarta.validation.constraints.Max; -import jakarta.validation.constraints.Min; -import jakarta.validation.constraints.Size; - -import io.quarkus.arc.Unremovable; -import io.quarkus.runtime.configuration.DurationConverter; -import io.smallrye.config.ConfigMapping; -import io.smallrye.config.WithConverter; -import io.smallrye.config.WithDefault; -import io.smallrye.config.WithName; -import io.smallrye.config.WithParentName; - -@Unremovable -@ConfigMapping(prefix = "cloud") -@Prod -public interface Cloud { - String host(); - - @Min(8000) - int port(); - - @WithConverter(DurationConverter.class) - Duration timeout(); - - @WithName("io-threads") - int threads(); - - @WithParentName - Map form(); - - Optional ssl(); - - Optional proxy(); - - Optional cors(); - - Log log(); - - Info info(); - - interface Form { - String loginPage(); - - String errorPage(); - - String landingPage(); - - Optional cookie(); - - @WithDefault("1") - List positions(); - } - - interface Ssl { - int port(); - - String certificate(); - - @WithDefault("TLSv1.3,TLSv1.2") - List protocols(); - } - - interface Proxy { - boolean enable(); - - @Max(10) - int timeout(); - } - - interface Log { - @WithDefault("false") - boolean enabled(); - - @WithDefault(".log") - String suffix(); - - @WithDefault("true") - boolean rotate(); - - @WithDefault("COMMON") - Pattern pattern(); - - Period period(); - - @Max(15) - int days(); - - enum Pattern { - COMMON, - SHORT, - COMBINED, - LONG; - } - } - - interface Cors { - List origins(); - - List<@Size(min = 2) String> methods(); - - interface Origin { - @Size(min = 5) - String host(); - - @Min(8000) - int port(); - } - } - - interface Info { - Optional<@Size(max = 3) String> name(); - - @Max(3) - OptionalInt code(); - - Optional> alias(); - - Map> admins(); - - Map> firewall(); - - interface Admin { - @Size(max = 4) - String username(); - } - } -} diff --git a/integration-tests/smallrye-config/src/main/java/io/quarkus/it/smallrye/config/Prod.java b/integration-tests/smallrye-config/src/main/java/io/quarkus/it/smallrye/config/Prod.java deleted file mode 100644 index 0387cabd46b58..0000000000000 --- a/integration-tests/smallrye-config/src/main/java/io/quarkus/it/smallrye/config/Prod.java +++ /dev/null @@ -1,29 +0,0 @@ -package io.quarkus.it.smallrye.config; - -import java.lang.annotation.ElementType; -import java.lang.annotation.Retention; -import java.lang.annotation.RetentionPolicy; -import java.lang.annotation.Target; - -import jakarta.validation.Constraint; -import jakarta.validation.ConstraintValidator; -import jakarta.validation.ConstraintValidatorContext; -import jakarta.validation.Payload; - -@Target({ ElementType.TYPE_USE, ElementType.FIELD }) -@Retention(RetentionPolicy.RUNTIME) -@Constraint(validatedBy = Prod.Validator.class) -public @interface Prod { - String message() default "server is not prod"; - - Class[] groups() default {}; - - Class[] payload() default {}; - - class Validator implements ConstraintValidator { - @Override - public boolean isValid(final Cloud value, final ConstraintValidatorContext context) { - return value.host().equals("prod"); - } - } -} diff --git a/integration-tests/smallrye-config/src/main/java/io/quarkus/it/smallrye/config/ServerResource.java b/integration-tests/smallrye-config/src/main/java/io/quarkus/it/smallrye/config/ServerResource.java index 69205f27a8553..ca507f61583c2 100644 --- a/integration-tests/smallrye-config/src/main/java/io/quarkus/it/smallrye/config/ServerResource.java +++ b/integration-tests/smallrye-config/src/main/java/io/quarkus/it/smallrye/config/ServerResource.java @@ -4,24 +4,15 @@ import jakarta.enterprise.inject.Instance; import jakarta.inject.Inject; -import jakarta.json.Json; -import jakarta.json.JsonArrayBuilder; -import jakarta.json.JsonObjectBuilder; import jakarta.ws.rs.GET; import jakarta.ws.rs.Path; import jakarta.ws.rs.core.Response; -import org.eclipse.microprofile.config.Config; import org.eclipse.microprofile.config.inject.ConfigProperties; import org.eclipse.microprofile.config.inject.ConfigProperty; -import io.smallrye.config.ConfigValidationException; -import io.smallrye.config.SmallRyeConfig; - @Path("/server") public class ServerResource { - @Inject - Config config; @Inject Server server; @Inject @@ -45,26 +36,6 @@ public Response getServerProperties() { return Response.ok(serverProperties).build(); } - @GET - @Path("/validator/cloud") - public Response validator() { - try { - config.unwrap(SmallRyeConfig.class).getConfigMapping(Cloud.class, "cloud"); - } catch (ConfigValidationException e) { - JsonArrayBuilder jsonArrayBuilder = Json.createArrayBuilder(); - for (int i = 0; i < e.getProblemCount(); i++) { - jsonArrayBuilder.add(e.getProblem(i).getMessage()); - } - JsonObjectBuilder jsonObjectBuilder = Json.createObjectBuilder(); - jsonObjectBuilder.add("errors", jsonArrayBuilder); - return Response.ok().entity(jsonObjectBuilder.build().toString()).build(); - } catch (Exception e) { - e.printStackTrace(); - } - - return Response.serverError().build(); - } - @GET @Path("/info") public String info() { diff --git a/integration-tests/smallrye-config/src/main/resources/application.yaml b/integration-tests/smallrye-config/src/main/resources/application.yaml index 7da81f0bee2be..81e3e4e758693 100644 --- a/integration-tests/smallrye-config/src/main/resources/application.yaml +++ b/integration-tests/smallrye-config/src/main/resources/application.yaml @@ -34,55 +34,6 @@ http: period: P1D days: 10 -cloud: - host: localhost - port: 5000 - timeout: 60s - io-threads: 200 - - form: - login-page: login.html - error-page: error.html - landing-page: index.html - - ssl: - port: 8443 - certificate: certificate - - cors: - origins: - - host: some-server - port: 9000 - - host: localhost - port: 1 - methods: - - GET - - POST - - proxy: - enable: true - timeout: 20 - - log: - period: P1D - days: 20 - - info: - name: Bond - code: 007 - alias: - - James - admins: - root: - - - username: root - - - username: super - firewall: - accepted: - - 127.0.0.1 - - 8.8.8 - profile: main: yaml: main diff --git a/integration-tests/smallrye-config/src/test/java/io/quarkus/it/smallrye/config/ServerResourceTest.java b/integration-tests/smallrye-config/src/test/java/io/quarkus/it/smallrye/config/ServerResourceTest.java index 400df4f21f128..0951ad1e8f5f8 100644 --- a/integration-tests/smallrye-config/src/test/java/io/quarkus/it/smallrye/config/ServerResourceTest.java +++ b/integration-tests/smallrye-config/src/test/java/io/quarkus/it/smallrye/config/ServerResourceTest.java @@ -1,19 +1,13 @@ package io.quarkus.it.smallrye.config; import static io.restassured.RestAssured.given; -import static jakarta.ws.rs.core.HttpHeaders.ACCEPT; -import static jakarta.ws.rs.core.HttpHeaders.CONTENT_TYPE; -import static jakarta.ws.rs.core.MediaType.APPLICATION_JSON; import static jakarta.ws.rs.core.Response.Status.OK; import static org.hamcrest.Matchers.containsString; import static org.hamcrest.Matchers.equalTo; -import static org.hamcrest.Matchers.hasItem; -import static org.hamcrest.Matchers.hasSize; import org.junit.jupiter.api.Test; import io.quarkus.test.junit.QuarkusTest; -import io.restassured.http.Header; @QuarkusTest class ServerResourceTest { @@ -75,25 +69,4 @@ void info() { .header("X-VERSION", "1.2.3.4") .body(containsString("My application info")); } - - @Test - void invalid() { - given() - .header(new Header(CONTENT_TYPE, APPLICATION_JSON)) - .header(new Header(ACCEPT, APPLICATION_JSON)) - .get("/server/validator/{prefix}", "cloud") - .then() - .statusCode(OK.getStatusCode()) - .body("errors", hasSize(10)) - .body("errors", hasItem("cloud.port must be greater than or equal to 8000")) - .body("errors", hasItem("cloud.log.days must be less than or equal to 15")) - .body("errors", hasItem("cloud.cors.origins[1].port must be greater than or equal to 8000")) - .body("errors", hasItem("cloud.info.name size must be between 0 and 3")) - .body("errors", hasItem("cloud.info.code must be less than or equal to 3")) - .body("errors", hasItem("cloud.info.alias[0] size must be between 0 and 3")) - .body("errors", hasItem("cloud.info.admins.root[1].username size must be between 0 and 4")) - .body("errors", hasItem("cloud.info.firewall.accepted[1] size must be between 8 and 15")) - .body("errors", hasItem("cloud.proxy.timeout must be less than or equal to 10")) - .body("errors", hasItem("cloud server is not prod")); - } }