我继承了一个项目,并且正在尝试针对内存中的h2数据库运行一组集成测试。为了让它们传递一些表,需要创建关系和参考数据。
我可以看到问题,因为RUNSCRIPT
中引用的脚本正在执行多次,因此会生成Index "XXX_IDX" already exists
错误和其他违规行为。那么有没有办法强制脚本只运行一次或者我需要一个外部数据库?似乎脚本运行在我认为是by design的每个连接上。
属性文件
my.datasource.url=jdbc:h2:mem:my_db;DB_CLOSE_DELAY=-1;MODE=Oracle;MVCC=TRUE;INIT=RUNSCRIPT FROM 'classpath:/create-tables-and-ref-data.sql'
XML配置
<bean id="myDataSource" class="org.apache.commons.dbcp.BasicDataSource">
<property name="url" value="${my.datasource.url}"/>
<!-- other properties for username, password etc... -->
</bean>
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="myDataSource"/>
</bean>
<bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
<property name="dataSource" ref="myDataSource"/>
</bean>
<tx:annotation-driven transaction-manager="transactionManager"/>
以下模式中的许多Java类
@Component
public class SomethingDAOImpl implements SomethingDAO {
@Autowired
public SomethingDAOImpl(JdbcTemplate jdbcTemplate) {
this.jdbcTemplate = jdbcTemplate;
}
}
@Component
public class SomethingElseDAOImpl implements SomethingElseDAO {
@Autowired
public SomethingElseDAOImpl(JdbcTemplate jdbcTemplate) {
this.jdbcTemplate = jdbcTemplate;
}
}
默认的bean范围是singleton,我认为这样可行,但我想我错过了一些东西。此外,如果我切换到已经具有表和参考数据设置的真实Oracle实例,则测试全部通过。
答案 0 :(得分:2)
在许多情况下,可以编写SQL脚本,以便不抛出任何异常:
create table if not exists test(id int, name varchar(255));
create index if not exists test_idx on test(name);
答案 1 :(得分:0)
我最终使用了另一种方法,因为我无法以可以无错误地重新应用的方式编写SQL。
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:jdbc="http://www.springframework.org/schema/jdbc"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.2.xsd
http://www.springframework.org/schema/jdbc http://www.springframework.org/schema/jdbc/spring-jdbc-3.2.xsd">
<jdbc:initialize-database data-source="myDataSource" enabled="true" ignore-failures="ALL">
<jdbc:script location="classpath:create-and-alter-tables-first-then-add-test-data.sql" />
</jdbc:initialize-database>
</beans>
在上下文初始化时执行一次。
注意:为简洁起见,省略了其他名称空间和bean。