Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Enhance decimal validation to match actual Jakarta Validation behavior and fix related issues #1131

Open
wants to merge 9 commits into
base: main
Choose a base branch
from

Conversation

Seol-JY
Copy link

@Seol-JY Seol-JY commented Dec 23, 2024

Summary

This PR addresses several issues and enhances the decimal validation handling in Fixture Monkey, primarily focused on fixing #1126 and improving the overall validation behavior for numeric types.

Related issues:
Fixes #1126, #1127

Description

This PR implements comprehensive improvements to the decimal validation system:

1. Fixed Inclusive Property Handling

  • Resolved issues with incorrect handling of the inclusive property in validation annotations
  • Ensures proper behavior when minimum and maximum values are equal
  • Applies to BigDecimal, float, and double types

2. Enhanced Multiple Validation Handling

  • Implemented proper AND operation logic for multiple constraints
    (e.g., when @DecimalMax("11.5") @Max(10) is applied to a field, values should be less than or equal to 10.0, as it satisfies both constraints)
  • Ensures all validation conditions are satisfied simultaneously

3. Improved @Digits Annotation Processing

  • Aligned @Digits validation behavior with Jakarta Validation's actual implementation
  • Previous implementation was overly restrictive:
    • For @Digits(integer=3, fraction=2), strictly generated numbers in xxx.xx format
    • This forced exact digit counts for both integer and fraction parts
  • New implementation matches actual validation rules:
    • For @Digits(integer=3, fraction=2), correctly allows:
      • Integer part: any number up to 999 (≤ 3 digits)
      • Fraction part: up to 2 decimal places (not requiring exactly 2)
    • Example: Both 5.4 and 999.99 are now valid where previously only xxx.xx format was generated

4. Added Repeatable Annotation Support

  • Implemented collection of all instances from container annotations marked with @Repeatable
  • Ensures comprehensive validation when multiple constraints are present

5. Code Refactoring

  • Improved code organization and readability
  • Enhanced error handling and validation logic
  • Simplified constraint processing workflow

How Has This Been Tested?

Manually verified functionality for other changes through direct testing:

  • Tested @Digits annotation's new behavior with various combinations (e.g., integer=3,fraction=2 properly allowing values up to 999.99)
  • Verified proper handling of multiple validation constraints working together
  • Confirmed correct inclusive/exclusive boundary handling for decimal values
  • Checked behavior of repeatable annotations across different scenarios
  • Validated proper constraint combinations (e.g., @Min/@Max with @DecimalMin/@DecimalMax)

Is the Document updated?

No documentation update is required as this is a bug fix for internal implementation.

Comment on lines 56 to 62
@DecimalMax(value = "100.1")
@DecimalMin(value = "100.1")
private BigDecimal decimalEqual;

@Max(value = 100)
@Min(value = 100)
private BigDecimal integerEqual;
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I cannot infer your intent, could you create a new DTO and make a new concise and clear test to reveal your intent?

Comment on lines 114 to 136
public <T extends Annotation> List<T> findAnnotations(Class<T> annotationClass) {
List<T> results = this.getResolvedProperty().getAnnotations().stream()
.filter(it -> annotationClass.isAssignableFrom(it.annotationType()))
.map(annotationClass::cast).collect(toList());

Repeatable repeatable = annotationClass.getAnnotation(Repeatable.class);
if (repeatable != null) {
Class<? extends Annotation> containerClass = repeatable.value();
this.getResolvedProperty().getAnnotations().stream()
.filter(it -> containerClass.isAssignableFrom(it.annotationType()))
.findFirst()
.ifPresent(container -> {
try {
Method valueMethod = container.annotationType().getDeclaredMethod("value");
T[] values = (T[])valueMethod.invoke(container);
results.addAll(Arrays.asList(values));
} catch (Exception ignored) {
}
});
}

return results;
}
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It is not for common case.

You can not confirm that user uses the annotation @Repeatable , the name of property value .

I think it should be removed.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@seongahjo
Ah, I understand! Instead of defining a new findAnnotations in ArbitraryGeneratorContext, should I handle duplicate annotations within ArbitraryGeneratorContext?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Does this.getResolvedProperty().getAnnotations() not return the duplicate annotations?

Could you make a test to show when the duplicate annotation is an issue?

Copy link
Author

@Seol-JY Seol-JY Dec 23, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Does this.getResolvedProperty().getAnnotations() not return the duplicate annotations?

Yes.

@Getter
@Setter
public class TestSpec {
	@Max(value = 50)        // interface jakarta.validation.constraints.Max
	private BigDecimal a;

	@Max(value = 100)       // multiple @Max annotations - internally processed as container annotation
	@Max(value = 50)
	private BigDecimal b;     

	@Max.List({             // interface jakarta.validation.constraints.Max$List
		@Max(value = 100),
		@Max(value = 50)
	})
	private BigDecimal c;
}

Duplicate annotations appear to be automatically processed in the same way as the annotations in field c.

Therefore, I implemented findAnnotations to process the contents within Max$List in the same way as regular duplicate annotations, as shown above.

Without the Repeatable processing in findAnnotations, if we only use this.getResolvedProperty().getAnnotations(), we would need to search twice using getAnnotations(Max.class) and getAnnotations(Max.List.class). This is because we need to consider both cases where there is a single annotation and multiple annotations.
Since getAnnotationsByType cannot be used in the current implementation, I chose to implement findAnnotations to handle both single annotations and container annotations in a unified way.

Copy link
Contributor

@seongahjo seongahjo Dec 24, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It is not clear to me how it works when there are multiple constraints on the same property.

Does the JSR-303 support multiple annotations on the same property within the same group?

I only found the applying multiple constraints within the different group.
https://beanvalidation.org/1.0/spec/#constraintsdefinitionimplementation-multipleconstraints

Copy link
Author

@Seol-JY Seol-JY Dec 24, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Since the @Repeatable annotation allows multiple annotations to be applied, it was confirmed during testing that all validation annotations (including duplicates) are processed as expected.

However, JSR-303 does not define a specific approach for handling duplicate annotations, meaning implementations can vary.

Given this, rather than implementing complex logic to handle duplicates, it may be more practical to revert to the original approach of simply checking for the presence of annotations.

@Seol-JY Seol-JY marked this pull request as draft December 24, 2024 07:01
Comment on lines 405 to 416
return new JavaDecimalConstraint(
positiveMin,
positiveMinInclusive,
positiveMax,
positiveMaxInclusive,
negativeMin,
negativeMinInclusive,
negativeMax,
negativeMaxInclusive,
isPositiveMin ? min : null,
isPositiveMin ? minInclusive : null,
isPositiveMax ? max : null,
isPositiveMax ? maxInclusive : null,

isNegativeMin ? min : null,
isNegativeMin ? minInclusive : null,
isNegativeMax ? max : null,
isNegativeMax ? maxInclusive : null,
scale
);
Copy link
Author

@Seol-JY Seol-JY Dec 24, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I don't understand why the JavaDecimalConstraint class needs to manage separate min/max values for negative and positive numbers. In the previous implementation with the @Digits annotation, this was necessary, but with the "3. Improved @Digits Annotation Processing," the method for enforcing maximum digit limits has changed. In this case, wouldn't it be sufficient to just manage the min/max values and their inclusiveness to generate random numbers? What do you think about changing this approach?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Is there a better idea for addressing both positive and negative values in the case of @Digits?

For example, in the case of @Digits(integer=3, fraction=2), it should be the value that satisfies the following constraints.

  1. 0 <= value <= 999.99
  2. -999.99 <= value <= 0

Copy link
Author

@Seol-JY Seol-JY Dec 29, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Looking at your example with @Digits(integer=3, fraction=2), I believe we could satisfy these constraints simply using a minimum (-999.99) and maximum (999.99) value. Is there any specific case where we need to manage positive and negative ranges separately in the current implementation?

Moreover, simplifying to a single range would actually help prevent potential distribution issues. For instance, with the current implementation, when we have constraints like:

@Digits(integer=3, fraction=2)
@Min(-2)
@Max(100)
private BigDecimal b;

The resolveArbitrary method in JqwikJavaArbitraryResolver splits the generation between negative and positive ranges with equal probability, leading to disproportionate distribution where values between -2 and 0 have the same total probability as values between 0 and 100. By using a single range approach, we could ensure truly uniform distribution across the entire valid range.
If I'm missing any important considerations, please let me know.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

To maintain compatibility with the existing structure (as it seems there would be many affected areas due to changes), I decided to continue using the existing JavaDecimalConstraint.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Sorry for the delay. I have misunderstood the validation with @Digits.
It is ok to set the limit within a minimum (-999.99) and maximum (999.99) value.

To maintain compatibility with the existing structure (as it seems there would be many affected areas due to changes)

Can you tell me what the breaking change is?

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

No problem! The JavaDecimalConstraint class appears to be used in the following methods:

  1. ValidateArbitraryGenerator#generate()
  2. JqwikJavaArbitraryResolver#floats()
  3. JqwikJavaArbitraryResolver#doubles()
  4. JqwikJavaArbitraryResolver#bigDecimals()
  5. SimpleValueJqwikPlugin.SimpleJavaConstraintGenerator#generateDecimalConstraint()

I want to reduce the fields in JavaDecimalConstraint as follows:

public JavaDecimalConstraint(
    @Nullable BigDecimal min
    @Nullable Boolean minInclusive,
    @Nullable BigDecimal max,
    @Nullable Boolean maxInclusive,
    @Nullable Integer scale
) {
    this.min = min;
    this.minInclusive = minInclusive;
    this.max = max;
    this.maxInclusive = maxInclusive;
    this.scale = scale;
}

Would it be okay to make this change and modify the implementation of related methods? I'm a bit concerned about potential side effects. Also, I'm curious whether JavaDecimalConstraint is also being used in any Kotlin implementations!

@Seol-JY Seol-JY requested a review from seongahjo January 17, 2025 15:38
@Seol-JY Seol-JY marked this pull request as ready for review January 25, 2025 18:54
Comment on lines +402 to +405
boolean isPositiveMin = min != null && min.compareTo(BigDecimal.ZERO) >= 0;
boolean isPositiveMax = max != null && max.compareTo(BigDecimal.ZERO) >= 0;
boolean isNegativeMin = min != null && min.compareTo(BigDecimal.ZERO) < 0;
boolean isNegativeMax = max != null && max.compareTo(BigDecimal.ZERO) < 0;
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

NIT, in convention of FixtureMonkey, only the boolean parameter allows the prefix of is.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

IllegalArgumentException when generating non-integer numeric values with equal min and max values
2 participants