From 06be39e07cd5f2413976d930f792930ceec0e25c Mon Sep 17 00:00:00 2001 From: altro3 Date: Tue, 12 Nov 2024 15:54:20 +0700 Subject: [PATCH] Fix: Calling addConstraintViolation on a ConstraintValidatorContext results in a wrong validation message (#444) Fixed #397 --- .../validation/CustomerConstraintTest.java | 124 ++++++++++++++++++ .../DefaultConstraintViolationBuilder.java | 2 +- 2 files changed, 125 insertions(+), 1 deletion(-) create mode 100644 test-suite/src/test/java/io/micronaut/docs/validation/CustomerConstraintTest.java diff --git a/test-suite/src/test/java/io/micronaut/docs/validation/CustomerConstraintTest.java b/test-suite/src/test/java/io/micronaut/docs/validation/CustomerConstraintTest.java new file mode 100644 index 00000000..a57fef03 --- /dev/null +++ b/test-suite/src/test/java/io/micronaut/docs/validation/CustomerConstraintTest.java @@ -0,0 +1,124 @@ +package io.micronaut.docs.validation; + +import io.micronaut.context.annotation.Property; +import io.micronaut.context.annotation.Requires; +import io.micronaut.core.annotation.AnnotationValue; +import io.micronaut.core.annotation.Introspected; +import io.micronaut.core.annotation.NonNull; +import io.micronaut.core.annotation.Nullable; +import io.micronaut.test.extensions.junit5.annotation.MicronautTest; +import io.micronaut.validation.validator.Validator; +import io.micronaut.validation.validator.constraints.ConstraintValidator; +import io.micronaut.validation.validator.constraints.ConstraintValidatorContext; +import jakarta.inject.Inject; +import jakarta.inject.Singleton; +import jakarta.validation.Constraint; +import jakarta.validation.ConstraintViolation; +import jakarta.validation.Payload; +import jakarta.validation.constraints.Pattern; +import org.junit.jupiter.api.Test; + +import java.lang.annotation.Documented; +import java.lang.annotation.Retention; +import java.lang.annotation.Target; +import java.util.ArrayList; +import java.util.Collection; +import java.util.List; +import java.util.Set; + +import static java.lang.annotation.ElementType.FIELD; +import static java.lang.annotation.ElementType.PARAMETER; +import static java.lang.annotation.ElementType.TYPE_USE; +import static java.lang.annotation.RetentionPolicy.RUNTIME; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.containsString; +import static org.hamcrest.Matchers.empty; +import static org.hamcrest.Matchers.not; + +@Property(name = "spec.name", value = "CustomerConstraintTest") +@MicronautTest(startApplication = false) +class CustomerConstraintTest { + + @Inject + Validator validator; + + + @Test + void should_return_interpolated_message() { + // Given + var bean = new ExampleBean(); + bean.getList().add("11111"); + + // When + Set> violations = validator.validate(bean); + + // Then + assertThat("violations", violations, not(empty())); + assertThat("violations[0].message", violations.iterator().next().getMessage(), containsString("[a-z]")); + assertThat("violations[0].message", violations.iterator().next().getMessageTemplate(), containsString("regexp")); + } + + @Introspected + static class ExampleBean { + @CollectionPattern(regexp = "[a-z]") + private List list = new ArrayList<>(); + + public List getList() { + return list; + } + + public void setList(List list) { + this.list = list; + } + } + + @Target({FIELD, PARAMETER, TYPE_USE}) + @Retention(RUNTIME) + @Documented + @Constraint(validatedBy = {CollectionPatternValidator.class}) + @interface CollectionPattern { + + String regexp(); + + Pattern.Flag[] flags() default {}; + + String message() default "{jakarta.validation.constraints.Pattern.message}"; + + Class[] groups() default {}; + + Class[] payload() default {}; + } + + @Requires(property = "spec.name", value = "CustomerConstraintTest") + @Singleton + static class CollectionPatternValidator implements ConstraintValidator> { + + @Override + public boolean isValid(@Nullable Collection values, + @NonNull AnnotationValue annotationMetadata, + @NonNull ConstraintValidatorContext context) { + + if (values == null || values.isEmpty()) { + return true; + } + + final String regexp = annotationMetadata.getRequiredValue("regexp", String.class); + final java.util.regex.Pattern regex = java.util.regex.Pattern.compile(regexp); + + context.disableDefaultConstraintViolation(); + + boolean valid = true; + for (String value : values) { + if (value != null && !regex.matcher(value).matches()) { + + context.buildConstraintViolationWithTemplate("must match \"{regexp}\"") + .addPropertyNode("[" + value + "]") + .addConstraintViolation(); + + valid = false; + } + } + return valid; + } + } +} diff --git a/validation/src/main/java/io/micronaut/validation/validator/DefaultConstraintViolationBuilder.java b/validation/src/main/java/io/micronaut/validation/validator/DefaultConstraintViolationBuilder.java index eb262ef8..2ca941d4 100644 --- a/validation/src/main/java/io/micronaut/validation/validator/DefaultConstraintViolationBuilder.java +++ b/validation/src/main/java/io/micronaut/validation/validator/DefaultConstraintViolationBuilder.java @@ -121,7 +121,6 @@ public ConstraintValidatorContext addConstraintViolation() { constraintValidatorContext.getRootClass(), null, null, - messageTemplate, messageInterpolator.interpolate(messageTemplate, new MessageInterpolator.Context() { @Override public ConstraintDescriptor getConstraintDescriptor() { @@ -138,6 +137,7 @@ public T unwrap(Class type) { throw new ValidationException("Not supported!"); } }), + messageTemplate, validationPath.iterator().hasNext() ? validationPath : new ValidationPath(constraintValidatorContext.getCurrentPath()), constraintValidatorContext.constraint, null,