I'm a bit of a Java EE/EJB noob, but from the docs and other posts I've gathered you cannot query the database using the same entitymanager/session during entity validation.
In general, the lifecycle method of a portable application should not invoke EntityManager or Query operations, access other entity instances, or modify relationships within the same persistence context.[43] A lifecycle callback method may modify the non-relationship state of the entity on which it is invoked.
Translation please?
This is pretty abstract...can it be explained in more concrete terms? It leads to more questions than it answers. For example, if my entity has a lazy-loaded collection am I allowed to access it during validation? The collection is 'another entity' and will require a DB query which seems to be in violation of the docs.
This 'lifecycle requirement' seems odd because it's just a fact of life that certain validations do indeed require querying the database.
From other posts I've also seen people get around this querying issue by creating a new entitymanager/session using the entitymanagerfactory.
This leads me to two questions about using EntityManagers and Hibernate Validation:
- Is it possible I have some sort of design flaw or am misusing Hibernate Validation because I need to query the database during validation?
- Given that I'm using Java EE with JBoss, how do I inject my validator with an EntityManagerFactory?
I've tried something like this:
@Stateless
public class UserValidator implements ConstraintValidator<ValidUser, User> {
@PersistenceUnit(unitName="blahblah")
EntityManagerFactory emf;
...
}
But the EMF never gets injected. I'm guessing the @Stateless tag becomes irrelevant because I'm implementing a ConstraintValidator interface which is needed for the Hibernate Validator stuff to work.
So what's the general pattern for getting at an EntityManagerFactory from a Validator?
Thanks!
Through some of the comments and enough scrounging around, I finally figured out a somewhat 'canonical' way to answer my question.
But to clear things up, my question really was asking two things which have 2 distinct answers:
Answering the second question first I'll simply say, it is strongly encouraged to use a second EntityManager to do queries during validation. That means you should be injecting an EntityManagerFactory and creating a new EntityManager for queries (rather than injecting an EntityManager which will be the same one that created the lifecycle event to begin with).
Generally speaking, for validation purposes, you'll only be querying the database anyway and not inserting/updating so this should be fairly safe to do.
I asked a very related SO question here.
Now to answer question 1.
Yes it is completely possible to inject things into Validators used in the Hibernate Validation framework. To accomplish this you need to do 3 things:
Here is an example custom ConstraintValidatorFactory that uses 'managed' (injectable) validators:
Here is an example validation.xml file that tells Hibernate Validator which class to use as the ValidatorFactory:
And finally a validator class with injection points:
I think I get your desire to do all the validation using the awesome bean validation API, but remember that is not required.
Furthermore, think about these two requirements:
The first clearly only depends on the password itself and I would classify it as validating data and therefore such validation belongs in the data layer.
The second depends on the relation of a piece of data with a number of other entities, or with the current state of the system. I would classify this as something that belongs in the business layer.
That said, instead of trying to put the validation constraints on the entity class, put them on some business layer class (yes, you can even use bean validation if you so desire now).
For example, say you have a
User
entity with current password field and aPasswords
entity from which you can query for a user's old passwords. Now make your user data access object:Create your class-level constraint:
And validator:
Something like this should do it. As a side effect, since the actual validation is done by an EJB now, the validation logic itself will be transacted, if you leave the default transnational attributes that is.
The lifecycle events should not use the entity manager since it could lead to regressions. Imagine that during a pre-update event, you modify another entity. This should generate another pre-update event within the previous pre-update event. To avoid such issues, the use of entity manager is discouraged.
But if all you want to do is read some additional data, conceptually there is not problem. Valation happens implicitly in pre-update and pre-insert events.
If you never use post-load events, then reading data in a lifecycle event shouldn't trigger nested lifecycle events. As far as I understand the spec, querying entities is not stricly forbidden but strongly discouraged. In this case it might be fine. Did you try if this works?
Injection works only in managed entities. When injection is not possible, you should be able to do a good old lookup to obtain an entity manager. When using the second entity manager, nested lifecycle events might be generated, though. But if you only do something trivial like reading a list of old password, that should be OK.