使用单独的persistence.xml文件进行生产并使用Spring 3.1进行测试

时间:2012-03-28 23:57:16

标签: spring testing persistence.xml

好的,对不起,我找了好几个小时的答案,但是我为StackOverflow输入了整个问题来冒泡我正在寻找的链接。您可以阅读大量相关信息here


我有一个用Spring Roo创建的Spring项目来使用Hibernate和MySQL。但是,为了进行测试,我想在内存中使用HSQLDB,因为Roo集成测试使用ID(主键)0到10删除数据(而不是使用数据库为他们已经创建的数据分配的数据删除数据),这意味着它删除已存在于数据库中的数据,在我的情况下会导致约束违规,然后才能回滚事务。

这有点困难,因为我正在切换整个数据库提供程序,这意味着不同的Hibernate方言以及不同的DDL设置(在生产中验证,在测试中创建 - 丢弃)。但它并没有像我期望的那样工作,而且我为什么感到难过。

如果你知道它为什么不起作用,请说明,即使你没有解决方案。

这是一个Roo项目,我当然正在使用Maven。所以我尝试的第一件事就是拥有一个特定于测试的src/test/resources/META-INF/persistence.xml文件,同样也是一个特定于测试的src/test/resources/META-INF/spring/database.properties文件。这没有用,就像我跑mvn test时一切都坏了,相关信息是

Conflicting persistence unit definitions for name 'persistenceUnit'

为什么mvn test仍在使用非测试资源?

然后我将src/test/resources/META-INF/spring重命名为spring-test并将applicationContext.xml复制到其中。我将测试类中的上下文配置更改为

@ContextConfiguration(locations = "classpath:/META-INF/spring-test/applicationContext*.xml")

完成(或者我认为)分离,我对spring-test/applicationContext.xml进行了一些编辑:

更改了属性文件的路径:

<context:property-placeholder location="classpath*:META-INF/spring/*.properties"/>

<context:property-placeholder location="classpath*:META-INF/spring-test/*.properties"/>

更改了持久性单元的名称:

<bean class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean" id="entityManagerFactory">
    <property name="persistenceUnitName" value="persistenceUnit"/>
    <property name="dataSource" ref="dataSource"/>
</bean>

<bean class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean" id="entityManagerFactory">
    <property name="persistenceUnitName" value="testPersistenceUnit"/>
    <property name="dataSource" ref="dataSource"/>
</bean>

我将持久性单位名称的相应更改改为src/test/resources/META-INF/persistence.xml

好吧,好吧,现在没有冲突,但Hibernate失去了实体映射(例如Product实体),我得到了:

org.springframework.dao.InvalidDataAccessApiUsageException:
org.hibernate.hql.ast.QuerySyntaxException: Product is not mapped [SELECT o FROM Product o]; 

为什么Spring / Hibernate在此配置中丢失了实体映射?

接下来我尝试合并两个persistence.xml文件,以便src/main/resources/META-INF下的一个文件包含两个持久性单元。

有效!! ??

我认为这很难看,因为现在我在我的生产代码中有测试配置,但这就是我所接受的。

什么是更好的方式?

据我所知,persistence.xml中的属性不像Spring XML文件中那样。所以我认为只用特定于测试的属性文件就可以做我想做的事。

理想情况下,我使用src / main / resources下的所有配置运行测试,除了在src / test / resources中专门覆盖的内容。 有没有办法实现这个目标?

感谢您提供的任何见解!

1 个答案:

答案 0 :(得分:1)

在我的工作中,我曾经在没有数据库连接信息的情况下配置persistence.xml。数据库连接由Spring的上下文配置定义。在Spring框架中,有几种方法可以更改对象的属性:

  1. PropertyPlaceholderConfigurer - 使用不同的属性文件覆盖ORM的数据库连接或方言的值。如果您使用resource filter
  2. ,则可以使用Maven生成单个属性文件的不同值
  3. Bean definition inheritance - 使用其他上下文配置“覆盖”默认配置。
  4. 以下代码摘自我的申请背景:

    <bean class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer"
        p:location="classpath:frontend.properties" p:ignore-resource-not-found="true"
        p:systemPropertiesModeName="SYSTEM_PROPERTIES_MODE_OVERRIDE"
    />
    
    <util:properties id="jpaProperties">
        <prop key="hibernate.dialect">${hibernate.dialect}</prop>
    </util:properties>
    
    <!-- Global entity manager factory(may not be overrided) -->
    <bean id="entityManagerFactory" class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean"
        p:dataSource-ref="dataSource" p:jpaProperties-ref="jpaProperties"
    >
        <property name="JpaDialect">
            <bean class="org.springframework.orm.jpa.vendor.HibernateJpaDialect" />
        </property>
    </bean>
    <!-- :~) -->
    
    <!-- Data Source -->
    <bean id="dataSource" class="com.jolbox.bonecp.BoneCPDataSource" destroy-method="close"
        p:driverClass="${database.driverClass}" p:jdbcUrl="${database.url}"
        p:username="${database.username}" p:password="${database.password}"
        p:partitionCount="${database_conn.pooling.partition_count:2}"
        p:maxConnectionsPerPartition="64" p:minConnectionsPerPartition="${database_conn.pooling.min_connections:4}"
        p:acquireIncrement="4" p:statementsCacheSize="64"
        p:connectionTimeoutInMs="1800000" p:IdleConnectionTestPeriodInMinutes="420"
    />
    <!-- :~) -->
    

    以下是测试配置:

    <!-- import configuration of application -->
    <import resource="classpath:database.xml" />
    
    <!--
      - I override some of settings of connection pooling while testing
      -->
    <bean id="dataSource" class="com.jolbox.bonecp.BoneCPDataSource" destroy-method="close"
        p:driverClass="${database.driverClass}" p:jdbcUrl="${database.url}"
        p:username="${database.username}" p:password="${database.password}"
        p:maxConnectionsPerPartition="8" p:minConnectionsPerPartition="2"
        p:acquireIncrement="2" p:statementsCacheSize="32"
    />
    

    当我运行测试时,我在Maven surefire中设置系统属性以配置不同的数据库。如下例:

    <plugin>
         <groupId>org.apache.maven.plugins</groupId>
         <artifactId>maven-surefire-plugin</artifactId>
         <version>2.8.1</version>
         <configuration>
             <systemPropertyVariables>
                 <!-- Using HSQLDB as test database system -->
                 <database.driverClass>org.hsqldb.jdbc.JDBCDriver</database.driverClass>
                 <database.url>${database.hsqldb.url}</database.url>
                 <database.username>any</database.username>
                 <database.password>any</database.password
                 <hibernate.dialect>org.hibernate.dialect.HSQLDialect</hibernate.dialect>
                 <!-- :~) -->
             </systemPropertyVariables>
         </configuration>
    </plugin>