gpt4 book ai didi

spring - 在 Spring 3.1 中使用单独的 persistence.xml 文件进行生产和测试

转载 作者:行者123 更新时间:2023-11-28 19:47:43 26 4
gpt4 key购买 nike

好的,抱歉,我花了几个小时寻找这个问题的答案,但我输入了整个问题,让 StackOverflow 冒出我正在寻找的链接。可以看很多相关资料here .


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

这有点额外困难,因为我要切换整个数据库提供程序,这意味着不同的 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 中特别覆盖的配置。 有办法实现吗?

感谢您提供的任何见解!

最佳答案

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

  1. > PropertyPlaceholderConfigurer - 使用不同的属性文件来覆盖数据库连接或 ORM 方言的值。您可以使用 resource filter如果您使用 Maven,则生成单个属性文件的不同值
  2. > Bean definition inheritance - 使用另一个上下文配置来“覆盖”默认配置。

以下代码摘 self 的应用程序上下文:

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

关于spring - 在 Spring 3.1 中使用单独的 persistence.xml 文件进行生产和测试,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/9917344/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com