I am writing this so that I never, ever, ever forget how the new default lazy-load behavior in Hibernate3 can completely mess with your head for hours and hours of debugging when you are expecting Spring's HibernateTemplate
's load
method to throw a HibernateObjectRetrievalFailureException
in a unit test, only to find the reason is quite simple but subtle! Oh, and of course if anyone else is reading, or more importantly, Googling, then hopefully this will help you too.
I have an implementation of a DAO that extends Spring's HibernateDaoSupport
class which has a finder method for an entity given the unique identifier, which is of type Long
. That finder method basically does this:
public Entity findEntity(Long id) {
return (Entity)getHibernateTemplate().load(Entity.class, id);
}
Note specifically that I am using the load()
method which is defined to "Return the persistent instance of the given entity class with the given identifier, throwing an exception if not found." Specifically, this method should catch any HibernateException
subclass thrown by the internal HibernateCallback
and convert it into the appropriate class in the Spring DataAccessException
hierarchy, e.g. in this case it should be converted to a HibernateObjectRetrievalFailureException
if I pass in an identifier for which there is no persistent entity. So far, so good.
Next I have a simple unit test that calls my finder method using an invalid identifier, and then it asserts that the appropriate exception was thrown. Basically it looks something like this:
public void testFindEntityUsingInvalidIdentifier() {
final Class expectedExceptionType = HibernateObjectRetrievalFailureException.class;
try {
dao.findEntity(-9999L);
fail("Should have thrown an " + expectedExceptionType.getName());
} catch (Exception e) {
assertEquals(expectedExceptionType, e.getClass());
}
}
So I ran this test thinking it was a no-brainer. It failed. In fact, it failed with the message "junit.framework.AssertionFailedError: Should have thrown a org.springframework.orm.hibernate3.HibernateObjectRetrievalFailureException." Um, what? I was about 100% sure there was no such row in the database, since I was using Spring's AbstractTransactionalDataSourceSpringContextTests
test class which ensures transactions are rolled back after each test, and because I knew I didn't put any data in the database with that identifier. So that meant the findEntity
method did not throw any exception. I started adding the old fallback System.out.println()
statements all over the place to see what exactly was going on. The finder method actually was returning a non-null object, but when I tried to call any method on it, like toString()
, it then threw a raw Hibernate ObjectNotFoundException
, which as of Hibernate3 is unchecked not checked. Hmmm. Performed some initial debugging using a real debugger no less and found that proxy objects were being returned, and then looked in the stack traces and saw some CGLIB stuff in there, meaning the entity object was in fact proxied.
Since the object was actually a proxy, that explained why no exception was thrown by HibernateTemplate.load()
- since no methods had been called on the proxy yet, Hibernate3 was happily returning a proxy object for an identifier which does not really exist. The second you call any method on that proxy, you get the Hibernate ObjectNotFoundException
. Did a bunch of research and finally found that the Hibernate3 reference guide, in Section 11.3 (Loading an object) mentions that "load() will throw an unrecoverable exception if there is no matching database row. If the class is mapped with a proxy, load() just returns an uninitialized proxy and does not actually hit the database until you invoke a method of the proxy." I then did more research and arrived at a resource I should probably have looked at much sooner, as this is my first time using Hibernate3 (I've been using Hibernate2.x for a while now). That resource is the Hibernate3 migration guide which mentions that Hibernate3 now defaults to lazy="true"
for everything, both classes and collections. I even remember reading this a while back but it didn't occur to me that the load()
method in the Hibernate Session
would behave like that.
In any case, with lazy loading set to "true" because of the changed default value in Hibernate3, the Spring HibernateTemplate.load()
basically is useless for lazily initialized objects, since it will not catch any Hibernate exceptions and thus won't ever do the conversion into one of the DataAccessException
subclasses. There are a few solutions or workarounds or hacks or whatever you want to call them. First, you could set default-lazy="false"
in your root hibernate-mapping
element, which will switch the behavior back to what it was in Hibernate2.x. That is what I did to verify that my test would work properly when there was no proxying involved. Second, you can use the HibernateTemplate.get()
method instead of load()
(which delegates to the Hibernate Session.get()
method) because get()
"hits the database immediately and returns null if there is no matching row" and then if the result is null, throw the appropriate exception yourself. Third, you can leave Hibernate3's default behavior alone and override it for specific classes and/or collections. For collections I almost always use lazy loading but I had never done it for classes. So you could set lazy="true"
for specific mapped classes in your Hibernate mappings.
I fully understand why you want to lazily initialize your collections, but I am not sure why I would want the default behavior for normal classes to be lazy load. When I perform a find for a specific object, I pretty much know I want that object since I am going to do something with it, like display it in a web page or something. I suppose one use case for lazy initializing classes would be if you perform a query and get back a list of objects, and you don't want to initialize all of them until the user actually needs them. But even in that case I generally am going to display the search results and I will be using those objects. So I am still somewhat at odds for when I would realistically want this behavior; even for search results I can retrieve only the results I need by using Hibernate's ability to set the start row and number of rows returned for a query.
So, the main point of all this is that the default lazy loading of everything in Hibernate3 may cause some unexpected problems, and if you start seeing CGLIB in stack traces and weird things are happening, chances are you've got an object proxy rather than the actual objct you thought you were getting, and that the proxy is causing some weird behavior. Whew!