h2数据库访问来自单独线程的测试数据

时间:2019-02-14 12:05:27

标签: java spring testing integration-testing h2

我的集成测试方案:

  1. 在H2数据库中创建一行
  2. sleep(50000ms)(与此同时,由于配置了spring,因此调用了另一个线程,并且该线程应该找到在点1创建的行,并更新该行)
  3. 期望从点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"/>

从根本上讲-它调用receive bean的方法emailReceiverService。当然-我确保调用了该方法(通过向收件箱发送电子邮件)-但是,正如我所说的-我认为创建第二个线程的方式并不复杂。基本上来说,它是由Spring配置调用的(spring配置通过

传递给测试方法
@ContextConfiguration(locations = { "classpath:spring/test-lottery-context.xml", "classpath:spring/test-sms-context.xml" })
@Transactional
public class QueueServiceImplIntegrationTest extends AbstractTransactionalTestNGSpringContextTests {

)。

1 个答案:

答案 0 :(得分:0)

我在那里找到了根本问题:交易

我的测试正在扩展AbstractTransactionalTestNGSpringContextTests,因此事务范围是整个@Test注释的方法。我将测试更改为仅扩展AbstractTestNGSpringContextTests-然后将事务的范围缩小到从@Test带注释的方法调用的特定Service方法(我正在谈论常规MVC模式)。那解决了我的问题。

干杯!