All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.kiwiproject.validation.group.KiwiValidationGroups Maven / Gradle / Ivy

Go to download

Kiwi is a utility library. We really like Google's Guava, and also use Apache Commons. But if they don't have something we need, and we think it is useful, this is where we put it.

There is a newer version: 4.5.2
Show newest version
package org.kiwiproject.validation.group;

import static org.kiwiproject.base.KiwiPreconditions.checkArgumentNotNull;

import jakarta.validation.ConstraintViolation;
import jakarta.validation.Validator;
import jakarta.validation.groups.Default;
import lombok.experimental.UtilityClass;
import org.kiwiproject.validation.KiwiValidations;

import java.util.Arrays;
import java.util.Set;

/**
 * An opinionated class that makes working with validation groups easier...if you are OK with the restrictions this
 * utility imposes. Specifically, it assumes you only need three validation groups:
 * 
    *
  1. * a group that should be included only when validating "new" objects (e.g., objects that have been * instantiated but not yet persisted to some data store) *
  2. *
  3. * a group that should be included only when validating "existing" objects (e.g., an object was retrieved * from an external data source and has been changed with new information *
  4. *
  5. * a group that should always be included (this is the {@link Default} validation group) *
  6. *
* To use this utility, annotate your classes with Jakarta Beans Validation annotations. If you only want an * annotation to be included for "new" objects, set the {@code groups} property to {@link NewObject}. If instead * you only want an annotation to be included for "existing" objects, set the {@code groups} property to * {@link ExistingObject}. If you always want a validation annotation to be included during validation, don't set * the {@code groups} property. Here's an example: *
 * {@literal @}Null(groups = NewObject.class)
 * {@literal @}NotNull(groups = ExistingObject.class)
 *  private Long id;
 *
 * {@literal @}NotBlank
 *  private String content;
 * 
* In this example code, {@code id} must be null when the object is "new" but must be non-null when the object "exists". * The {@code content} is always required. When performing the logic to insert a "new" object you would pass * {@link #newObjectGroups()} as the groups in {@link Validator#validate(Object, Class[])}. Similarly, you would * pass {@link #existingObjectGroups()} as the groups in {@link Validator#validate(Object, Class[])} when performing * update logic on "existing" objects. For example, assuming a static import: *
 * // validate a new object
 * var violations = validator.validate(aNewObject, newObjectGroups());
 *
 * // validate an existing object
 * var violations = validator.validate(anExistingObject, existingObjectGroups());
 * 
* For even more convenience, you can use the various {@code validateXxx} methods, assuming as mentioned earlier * that you have annotated your classes with Jakarta Beans Validator annotations. A typical example is using * annotations such as {@link jakarta.validation.constraints.NotNull} and {@link jakarta.validation.constraints.Null} * and setting {@link NewObject} and {@link ExistingObject} as the value of the {@code groups} property, as shown * in the above code example. To validate a new object, you would then simply do the following: *
 * var violations = KiwiValidationGroups.validateNewObject(aNewObject);
 * 
* And to validate an existing object, you would do: *
 * var violations = KiwiValidationGroups.validateExistingObject(anExistingObject);
 * 
* Using this utility ensures objects are validated against all groups, unlike {@link jakarta.validation.GroupSequence} * whose behavior is to stop validating any later groups once a group fails validation. For example, if the * sequence is {@code { Default.class, NewObject.class }} and a constraint fails for the {@code Default} group, then * the constraints for {@code NewObject} will not be evaluated. In general, this is not the behavior we want or * expect. *

* To reiterate, this utility is opinionated and therefore limited in that it only knows about {@link NewObject} and * {@link ExistingObject}, and expects them to always be validated along with the {@link Default} group. We have * found this to be useful in enough situations to include it in kiwi. But if you need more flexibility, don't * use this and instead pass specific validation groups manually to the validate method in {@link Validator}. */ @UtilityClass public class KiwiValidationGroups { private static final Class[] NEW_OBJECT_GROUPS = new Class[]{NewObject.class, Default.class}; private static final Class[] EXISTING_OBJECT_GROUPS = new Class[]{ExistingObject.class, Default.class}; private static final Validator DEFAULT_VALIDATOR = KiwiValidations.newValidator(); /** * Use this when validating a new (transient) object. The groups include {@link NewObject} and {@link Default}. * * @return a new array containing the new object group classes * @see NewObject * @see Default */ public static Class[] newObjectGroups() { return Arrays.copyOf(NEW_OBJECT_GROUPS, NEW_OBJECT_GROUPS.length); } /** * Use this when validating an existing (persistent) object. The groups include {@link ExistingObject} * and {@link Default}. * * @return a new array containing the existing object group classes * @see ExistingObject * @see Default */ public static Class[] existingObjectGroups() { return Arrays.copyOf(EXISTING_OBJECT_GROUPS, EXISTING_OBJECT_GROUPS.length); } /** * Validate a new (transient) object using a default {@link Validator} instance. * * @param object the object to validate * @param the type of object being validated * @return validation results */ public static Set> validateNewObject(T object) { return validateNewObject(DEFAULT_VALIDATOR, object); } /** * Validate a new (transient) object using the given {@link Validator} instance. * * @param validator the {@link Validator} instance to use * @param object the object to validate * @param the type of object being validated * @return validation results */ public static Set> validateNewObject(Validator validator, T object) { checkArgumentNotNull(validator); return validator.validate(object, NEW_OBJECT_GROUPS); } /** * Validate an existing (persistent) object using a default {@link Validator} instance. * * @param object the object to validate * @param the type of object being validated * @return validation results */ public static Set> validateExistingObject(T object) { return validateExistingObject(DEFAULT_VALIDATOR, object); } /** * Validate an existing (persistent) object using the given {@link Validator} instance. * * @param validator the {@link Validator} instance to use * @param object the object to validate * @param the type of object being validated * @return validation results */ public static Set> validateExistingObject(Validator validator, T object) { checkArgumentNotNull(validator); return validator.validate(object, EXISTING_OBJECT_GROUPS); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy