Hibernate 4 JPA EntityListeners не запускается в приложении Spring MVC/Spring Data

У меня есть Hibernate 4/Spring MVC 3.2.4/Spring Data 1.4.1 с приложением Spring Security, в котором я пытаюсь интегрировать EventListeners в свои классы сущностей. Я думаю, что у меня все настроено правильно, и когда я запускаю свои модульные тесты, я вижу, что мои прослушиватели событий запускаются. Однако, когда я запускаю свое полное приложение MVC, прослушиватели событий никогда не запускаются.

Я немного потерял, где/как отлаживать эту проблему. Мой EntityManagerFactory настроен практически одинаково в обоих случаях:

ПРОИЗВОДСТВО:

<beans profile="tomcat">
    <!-- application datasource -->
    <bean id="dataSource" class="org.springframework.jndi.JndiObjectFactoryBean" scope="singleton" lazy-init="true">
        <property name="jndiName" value="java:comp/env/jdbc/josak" />
    </bean>

    <!--  entity manager -->
    <bean id="entityManagerFactory" class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean">
        <property name="dataSource" ref="dataSource"/>
        <property name="jpaVendorAdapter" ref="hibernateJpaVendorAdapter"/>
        <property name="packagesToScan" value="com.ia.domain"/>
        <property name="jpaProperties">
            <props>
                <prop key="hibernate.dialect">org.hibernate.dialect.MySQL5InnoDBDialect</prop>
                <prop key="hibernate.query.substitutions">true '1', false '0'</prop>
                <prop key="hibernate.generate_statistics">true</prop>
                <prop key="hibernate.show_sql">false</prop>
                <prop key="hibernate.format_sql">true</prop>
                <prop key="hibernate.hbm2ddl.auto">update</prop>
                <prop key="hibernate.ejb.naming_strategy">org.hibernate.cfg.ImprovedNamingStrategy</prop>
                <prop key="hibernate.connection.charSet">UTF-8</prop>
            </props>
        </property>

    </bean>
</beans>

ЕДИНИЧНЫЕ ИСПЫТАНИЯ:

<beans profile="test">
    <jdbc:embedded-database id="dataSource" type="H2"></jdbc:embedded-database>
    <bean id="entityManagerFactory" class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean">
        <property name="dataSource" ref="dataSource"/>
        <property name="jpaVendorAdapter" ref="hibernateJpaVendorAdapter"/>
        <property name="packagesToScan" value="com.ia.domain"/>
        <property name="jpaProperties">
            <props>
                <prop key="hibernate.dialect">org.hibernate.dialect.H2Dialect</prop>
                <prop key="hibernate.query.substitutions">true '1', false '0'</prop>
                <prop key="hibernate.generate_statistics">true</prop>
                <prop key="hibernate.use_sql_comments">true</prop>
                <prop key="hibernate.show_sql">true</prop>
                <prop key="hibernate.format_sql">true</prop>
                <prop key="hibernate.hbm2ddl.auto">create</prop>
                <prop key="hibernate.ejb.naming_strategy">org.hibernate.cfg.ImprovedNamingStrategy</prop>
                <prop key="hibernate.connection.charSet">UTF-8</prop>
            </props>
        </property>
        </bean>
</beans>  

поэтому я предполагаю, что дело не в том, что модульный тест entityManagerFactory определяется по-разному.

Мой модульный тест довольно прост:

@Transactional
@RunWith(SpringJUnit4ClassRunner.class)
@WebAppConfiguration
@ContextConfiguration({"classpath:META-INF/spring/applicationContext*.xml"})
@ActiveProfiles("test")
public class UserServiceImplTest {
    @Test
    public void updateUser(){
        User newInfo = dod.getNewTransientUser(15);
        userService.saveUser(newInfo);
    }
}       

Если я поставлю точку останова в свой метод @PrePersist, я увижу, что этот метод вызывается.

Мой контроллер MVC также довольно прост:

@RequestMapping( method=RequestMethod.GET, value="getUserInfo/{userId}", produces=MediaType.APPLICATION_JSON_VALUE)
@ResponseBody
@PreAuthorize("hasRole('ROLE_PERMISSION_RATE_VIEW')")
@Transactional
public String getUserInfo( @PathVariable long userId ){

    userService.saveUser(createUser());
    return "done";
}

где createUser() просто возвращает новый объект User с заполненными полями. Однако моя точка останова в моем EventListener никогда не срабатывала, и я не вижу в своих журналах ничего, что указывало бы на это.

Если это полезно, я также могу опубликовать класс прослушивателя событий, но не думаю, что это будет иметь большое значение.

Мой класс Entity определяется как:

@Entity
@EntityListeners( AuditEventListener.class)
public class User {

    @TableGenerator( name="UUIDGenerator", pkColumnValue="user_id", table="uuid_generator", allocationSize=1)
    @Id
    @GeneratedValue(strategy = GenerationType.TABLE, generator="UUIDGenerator")
    @Column(name = "id")
    private Long id;

...
}

Я пропустил что-то очевидное здесь? Есть ли причина, по которой это не работает из приложения MVC, а из модульного теста? В моем модульном тесте также используется полная конфигурация контекста Spring, поэтому я очень запутался в разнице между двумя случаями.

Я понимаю, что это очень туманный вопрос, но любые рекомендации по его дальнейшей отладке будут высоко оценены.


person Eric B.    schedule 12.03.2014    source источник


Ответы (1)


Как бы грустно это ни звучало, я думаю, что я отследил ошибку до проблемы jRebel. Однако не в самом jRebel у меня был конфигурационный файл rein.xml, который указывал на путь к классам моих объектов, хотя я хотел, чтобы jRebel загружал объекты из моего пути к классу веб-приложения (они находятся в 2 отдельных модулях). С тех пор я изменил свой pom.xml и удалил файл rein.xml, и все, кажется, работает.

person Eric B.    schedule 17.03.2014