Question

I have a validator for my Roo model.

When it finds an error (returns false), it works as expected.

However, when it succeeds, the controller enters an infinite loop resulting in a StackOverflowError:

at org.hibernate.validator.internal.engine.ValidatorImpl.validateConstraintsForDefaultGroup(ValidatorImpl.java:387)
at org.hibernate.validator.internal.engine.ValidatorImpl.validateConstraintsForCurrentGroup(ValidatorImpl.java:351)
at org.hibernate.validator.internal.engine.ValidatorImpl.validateInContext(ValidatorImpl.java:303)
at org.hibernate.validator.internal.engine.ValidatorImpl.validate(ValidatorImpl.java:133)
at org.hibernate.cfg.beanvalidation.BeanValidationEventListener.validate(BeanValidationEventListener.java:136)
at org.hibernate.cfg.beanvalidation.BeanValidationEventListener.onPreInsert(BeanValidationEventListener.java:94)
at org.hibernate.action.internal.EntityInsertAction.preInsert(EntityInsertAction.java:181)
at org.hibernate.action.internal.EntityInsertAction.execute(EntityInsertAction.java:81)
at org.hibernate.engine.spi.ActionQueue.execute(ActionQueue.java:377)
at org.hibernate.engine.spi.ActionQueue.executeActions(ActionQueue.java:369)
at org.hibernate.engine.spi.ActionQueue.executeActions(ActionQueue.java:286)
at org.hibernate.event.internal.AbstractFlushingEventListener.performExecutions(AbstractFlushingEventListener.java:339)
at org.hibernate.event.internal.DefaultAutoFlushEventListener.onAutoFlush(DefaultAutoFlushEventListener.java:62)
at org.hibernate.internal.SessionImpl.autoFlushIfRequired(SessionImpl.java:1205)
at org.hibernate.internal.SessionImpl.list(SessionImpl.java:1262)
at org.hibernate.internal.QueryImpl.list(QueryImpl.java:101)
at org.hibernate.ejb.QueryImpl.getSingleResult(QueryImpl.java:283)
at com.foo.domain.validate.UniqueNameValidator.isValid(UniqueNameValidator.java:30)
at com.foo.domain.validate.UniqueNameValidator.isValid(UniqueNameValidator.java:1)
at org.hibernate.validator.internal.engine.ConstraintTree.validateSingleConstraint(ConstraintTree.java:308)
at org.hibernate.validator.internal.engine.ConstraintTree.validateConstraints(ConstraintTree.java:180)
at org.hibernate.validator.internal.engine.ConstraintTree.validateConstraints(ConstraintTree.java:124)
at org.hibernate.validator.internal.metadata.core.MetaConstraint.validateConstraint(MetaConstraint.java:86)
at org.hibernate.validator.internal.engine.ValidatorImpl.validateConstraint(ValidatorImpl.java:442)
at org.hibernate.validator.internal.engine.ValidatorImpl.validateConstraintsForDefaultGroup(ValidatorImpl.java:387)

Note that the first and last lines are the same. This stack trace repeats many times before the stack overflow occurs.

My UniqueNameValidator looks like this:

public class UniqueNameValidator implements ConstraintValidator<UniqueFoo, Foo> {
  public boolean isValid(Foo value, ConstraintValidatorContext context) {
    TypedQuery<Long> q = ... // checks several conditions with other models in actual code
    return q.getSingleResult() == 0;
  }
}

My Roo MVC controller looks like this:

@RequestMapping(method = RequestMethod.POST, produces = "text/html")
public String create(@Valid Foo foo,
                     BindingResult bindingResult,
                     Model uiModel,
                     HttpServletRequest httpServletRequest) {
    if (bindingResult.hasErrors()) {
        populateCreateForm(uiModel, foo);
        return "foos/create";
    }
    uiModel.asMap().clear();

    foo.persist();

    return Session.getStoredURI("/" + encodeUrlPathSegment(foo.getId().toString(), httpServletRequest));
}

My model looks like this:

@RooJavaBean
@RooJpaActiveRecord
@UniqueName
public class Foo {
  @NotNull
  private String parentName;
  ...
}

What am I doing wrong?

Was it helpful?

Solution

The query in the UniqueNameValidator is causing the recursive validation.

TypedQuery<Long> q = ... 
return q.getSingleResult() == 0;

needs to be:

TypedQuery<Long> q = ...
q.setFlushMode(FlushModeType.COMMIT);
return q.getSingleResult() == 0;
Licensed under: CC-BY-SA with attribution
Not affiliated with StackOverflow
scroll top