H2访问数据库以从单独的线程访问测试数据
我的集成测试场景:
- 在H2数据库中创建行
sleep(50000ms)
(同时,由于Spring配置,将调用另一个线程,此线程应找到点1中创建的行,并更新此行)- 预期从点%1开始有行。已由点%2中提到的线程更新。
此方案同时测试配置和实现。这就是我想要实现的目标。
我在所有测试中都使用H2数据库,因此决定在这里也使用它。在调试测试场景时,我发现sleep
期间调用的新线程连接到数据库,但没有找到创建的行。我浏览了H2文档,并开始使用:
java -cp ~/.m2/repository/com/h2database/h2/1.4.194/h2-1.4.194.jar org.h2.tools.Server -tcp -web -browser -tcpAllowOthers -tcpPort 9092 -webPort 8082
和连接字符串:
DB_URL=jdbc:h2:tcp://localhost:9092/~/test2
和配置:
<bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource"
destroy-method="close" p:driverClassName="org.h2.Driver"
p:url="${DB_URL}"
p:username="${OPENSHIFT_MYSQL_DB_USERNAME}" p:password="${OPENSHIFT_MYSQL_DB_PASSWORD}"/>
<util:properties id="hibernateProperties">
<prop key="hibernate.dialect">org.hibernate.dialect.H2Dialect</prop>
<prop key="hibernate.hbm2ddl.auto">validate</prop>
<prop key="hibernate.show_sql">false</prop>
</util:properties>
<bean id="sessionFactory"
class="org.springframework.orm.hibernate5.LocalSessionFactoryBean"
p:dataSource-ref="dataSource" p:packagesToScan="com.fridayweekend.lottery.model"
p:hibernateProperties-ref="hibernateProperties"/>
<bean id="transactionManager"
class="org.springframework.orm.hibernate5.HibernateTransactionManager"
p:sessionFactory-ref="sessionFactory"/>
注意:我已经尝试过DDL-AUTO-我在第一次试运行时尝试创建架构,然后只验证它(以防止重新创建),但是没有帮助。
我可以通过H2 Web Console
检查数据库。我看到创建了Schema(基于我的带Java注释的模型),但是没有数据(即使在sleep
或调试断点期间)。
要使来自主线程(@Test注释)的数据对第二个线程可见,我应该做些什么?
PS。我认为这无关紧要,但是"第二个"线程是这样调用的:
<util:properties id="javaMailProperties">
<prop key="mail.imap.socketFactory.class">javax.net.ssl.SSLSocketFactory</prop>
<prop key="mail.imap.socketFactory.fallback">false</prop>
<prop key="mail.store.protocol">${imap.protocol}</prop>
<prop key="mail.debug">${imap.debug}</prop>
</util:properties>
<mail:inbound-channel-adapter id="imapAdapter"
store-uri="${imap.uri}"
channel="recieveEmailChannel"
should-delete-messages="false"
should-mark-messages-as-read="true"
auto-startup="true"
java-mail-properties="javaMailProperties">
<int:poller fixed-delay="${imap.poolerSecondsDelay}" time-unit="SECONDS"/>
</mail:inbound-channel-adapter>
<int:channel id="recieveEmailChannel">
<int:interceptors>
<int:wire-tap channel="logger"/>
</int:interceptors>
</int:channel>
<int:logging-channel-adapter id="logger" level="DEBUG"/>
<int:service-activator input-channel="recieveEmailChannel" ref="emailReceiverService" method="receive"/>
因此,它基本上调用emailReceiverService
bean的receive
方法。当然-我确保方法被调用(通过向收件箱发送电子邮件)-但是,正如我所说的-我不认为创建第二个线程的方式是相关的。基本上-它是由Spring配置调用的(Spring配置通过传递给测试方法
@ContextConfiguration(locations = { "classpath:spring/test-lottery-context.xml", "classpath:spring/test-sms-context.xml" })
@Transactional
public class QueueServiceImplIntegrationTest extends AbstractTransactionalTestNGSpringContextTests {
)。
解决方案
我在那里找到了根本问题:事务!
我的测试是扩展AbstractTransactionalTestNGSpringContextTests
,因此事务范围是整个@Test
带注释的方法。我将测试更改为简单地扩展AbstractTestNGSpringContextTests
-然后事务的范围缩小到从@Test
注释方法调用的特定服务方法(我说的是常规MVC模式)。这解决了我的问题。
干杯!
相关文章