Тест интеграции Spring mvc с hibernate и jackson-hibernate-mapper: недопустимая попытка связать коллекцию с двумя сессиями

Исключение "незаконная попытка связать коллекцию с двумя сеансами" хорошо известно и возникает, когда на одну и ту же сущность ссылаются в двух разных сеансах. У меня есть весенний интеграционный тест MVC, где это происходит, и мне нужен совет, как заставить его работать правильно.

Также важно отметить, что я использую https://github.com/FasterXML/jackson-datatype-hibernate что помогает избежать исключения LazyInitializationException при преобразовании объектов домена в json. Я предполагаю, что он внутренне открывает сеанс Hibernate для извлечения ленивых отношений.

Вот тест:

@RunWith(SpringJunit4ClassRunner.class)
@WebAppConfiguration
@ContextHierarchy({
    @ContextConfiguration(name = "root", locations = "classpath:application.xml"),
    @ContextConfiguration(name = "servlet", locations = "classpath:servlet.xml")
})
@TransactionalConfiguration
public class IntegrationTest {
    private MockMvc mockMvc;

    //some initialization and autowiring

    @Test
    @Transactional // It is important to rollback all the db changes after the test. So, it's a common pattern to make all tests transactional.
    // So here we have HibernateSession#1
    public void testController() {
        // repository is spring-data-jpa repository
        // but you may think of it as a DAO which returns persisted 
        // hibernate entity
        ChildEntity child = new ChildEntity();
        child = childRepository.save();

        MyEntity entity = new MyEntity();
        entity.setChildEntities(Collections.singletoneList(child));
        entity = repository.save(entity);
        // entity and its child are still preserved in the HibernateSession#1

        mockMvc.perform(get("/entity/by_child_id/" + child.getId()).andExpect(status().is("200"));
    }

}

@RestController
@RequestMapping("/entity")
public class MyController {

    @RequestMapping("by_child_id/{id}")
    // My guess: Jackson hibernate mapper will open another session
    // and try to fetch children  
    // So here we have HibernateSession#2
    public MyEntity getByChildId(@PathVariable("id") childId) {
         return repository.findByChildrenId(childId);
    }

}

@Entity 
public class MyEntity {
    @OneToMany(fetch = FetchType.EAGER)
    private List<ChildEntity> children;
}

//and finally here is a piece of servlet.xml
<mvc:annotation-driven>
    <mvc:message-converters>
        <!-- Use the HibernateAware mapper instead of the default -->
        <bean class="org.springframework.http.converter.json.MappingJackson2HttpMessageConverter">
            <property name="objectMapper">
                <bean class="path.to.your.HibernateAwareObjectMapper" />
            </property>
        </bean>
    </mvc:message-converters>
</mvc:annotation-driven> 

У меня есть исключение:

Caused by org.hibernate.HibernateException: Illegal attempt to associate a collection with two open sessions: [MyEntity.children#1]
    at com.fasterxml.jackson.datatype.hibernate4.PersistentCollectionSerializer.inistializeCollection(PersistentCollectionSerializer.java:195)
    at com.fasterxml.jackson.datatype.hibernate4.PersistentCollectionSerializer.findLazyValue(PersistentCollectionSerializer.java:160)
    at com.fasterxml.jackson.datatype.hibernate4.PersistentCollectionSerializer.serialize(PersistentCollectionSerializer.java:115)
    at com.fasterxml.jackson.databind.ser.BeanPropertyWriter.serializeAsField(BeanPropertyWriter.java:505)
    at com.fasterxml.jackson.databind.ser.std.BeanSerializerBase.serializeFields(BeanSerializerBase:639)

jackson-databind-2.4.2.jar

Итак, есть ли способ избежать этой проблемы, кроме как избегать использования jackson-hibernate-mapper для тестов? Использование DTO и Dozer вместо доменного объекта не рассматривается как решение (это будет сделано в будущем, но не сейчас).

заранее спасибо

1 ответ

Решение

Благодаря M. Deinum я решил использовать самое простое решение и создал test-servlet-context.xml:

<beans>

   <import resource="classpath:servlet.xml">   

   <bean id="jsonConverter" class="org.springframework.http.converter.json.MappingJackson2HttpMessageConverter"/>

</beans>

Также я изменил определение конвертеров в servlet.xml:

<mvc:annotation-driven>
    <mvc:message-converters>
        <!-- Use the HibernateAware mapper instead of the default -->
        <ref bean="jsonConverter"/>
    </mvc:message-converters>
</mvc:annotation-driven> 

<bean class="org.springframework.http.converter.json.MappingJackson2HttpMessageConverter">
    <property name="objectMapper">
        <bean class="path.to.your.HibernateAwareObjectMapper" />
    </property>
</bean>

Теперь мои тесты не используют магию спящего Джексона. Это бесполезно, поскольку все тесты сами являются транзакционными.

О дефекте сообщили здесь: https://github.com/FasterXML/jackson-datatype-hibernate/issues/74

Другие вопросы по тегам