Hibernate 4 JPA EntityListeners未在Spring MVC / Spring Data应用程序中触发

时间:2014-03-12 04:07:03

标签: java spring hibernate jpa entitylisteners

我有一个带有Spring Security应用程序的Hibernate 4 / Spring MVC 3.2.4 / Spring Data 1.4.1,其中我试图在我的实体类上集成EventListeners。我认为我已经正确配置了所有内容,当我运行单元测试时,我可以看到我的事件监听器被触发了。但是,当我运行我的完整MVC应用程序时,事件侦听器永远不会被触发。

我有点迷失在哪里/如何调试这个问题。在这两种情况下,我的EntityManagerFactory设置几乎相同:

PRODUCTION:

<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>

UNIT TESTS:

<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
@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上下文配置,所以我对这两种情况之间的差异感到非常困惑。

我意识到这是一个非常模糊的问题,但是如何进一步调试这个问题的任何指导都将非常感激。

1 个答案:

答案 0 :(得分:0)

听起来很难过,我想我已经将错误跟踪为jRebel问题了。然而,不是在jRebel本身,我有一个指向我的实体的类路径的rebel.xml配置文件,即使我希望jRebel从我的webapp类路径加载实体(它们在2个单独的模块中)。我已经修改了我的pom.xml并删除了rebel.xml文件,一切似乎都有效。