Spring 使用数据库启动应用程序 - 测试 class 在使用 @DirtiesContext 重新创建上下文后失败

Spring Boot Application with DB - Test class fails after context being recreated with @DirtiesContext

我正在尝试执行 class 一组 Spring 引导应用程序的 JUnit4 测试,该应用程序由多个 Web 服务组成并配置了数据库。

在每次测试后清除上下文很方便,所以我在每个测试中都添加了一个@DirtiesContext 注释class,因为此注释的默认行为设置为AFTER_CLASS。

我遇到的问题是第一个测试 class 运行良好,但随后的测试总是失败。

所以我创建了 2 个简单的 JUnit classes 来尝试解决这个问题。两者都是相等的,测试方法是空的,所以总是应该 return 成功:

import org.springframework.http.HttpStatus;
import org.springframework.http.MediaType;
import org.springframework.http.ResponseEntity;
import org.springframework.test.annotation.DirtiesContext;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
import org.springframework.test.context.web.WebAppConfiguration;

import urlshortener2014.goldenbrown.Application;
import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertTrue;

@RunWith(SpringJUnit4ClassRunner.class)
@SpringApplicationConfiguration(classes = Application.class)
@WebAppConfiguration
@IntegrationTest("server.port=0")
@DirtiesContext
public class ApplicationTests {

    @Value("${local.server.port}")
    private int port = 0;

    @Test
    public void testAlwaysOk() throws Exception {

    }
}

我已经在 eclipse 中和通过 "gradle test" 执行了两个测试 classes(ApplicationTests 和 SameApplicationTests),在这两种情况下,第二个和后续测试 classes 都失败了,在上下文被清洁。

我怀疑问题与应用程序的数据库有关,没有正确重新创建,因为输出跟踪多次指向与数据库相关的错误。但我不确定这是怎么发生的,为什么会发生,也不知道如何解决。

这是一个具有 "gradle test" 输出的要点(正常输出、--info 输出和--debug 输出): https://gist.github.com/jgbarcos/c8b34c5c292ca1fabc1d

这是正在使用的 build.gradle(仅用于测试 2 个简单的 classes):

eclipse {
   project {
      name = "UrlShortener2014.goldenBrown"
   }
}

dependencies {
   compile project(":common")
     // Provides java script libraries for static content
   compile("org.webjars:bootstrap:3.0.3")
   compile("org.webjars:jquery:2.0.3-1")
   compile 'org.apache.httpcomponents:httpclient:4.3.6'
   compile 'nl.bitwalker:UserAgentUtils:1.2.4'
   compile 'org.springframework:spring-context'
   compile 'org.springframework:spring-context-support'
   compile 'net.sf.ehcache:ehcache:2.7.4'
   compile("org.springframework.boot:spring-boot-starter-web:1.2.0.RELEASE")
   compile 'org.springframework:spring-test:4.1.4.RELEASE'
   testCompile 'junit:junit:4.10'
}

// Used for @DirtiesContext problem
test{
    scanForTestClasses = false
    // This should get only "ApplicationTests.class" and "SameApplicationTests.class"
    include "**/*ApplicationTests.class"
}

这是我创建的 GitHub 分支,用于重现团队项目文件夹 (goldenBrown) 的问题: https://github.com/jgbarcos/UrlShortener2014/tree/debug_branch/goldenBrown

(注意: 项目依赖于另一个文件夹“/common”而不是“/goldenBrown”中名为common的项目,这可能有点棘手)

希望这有助于理解问题,在此先感谢。

您的代码没问题。错误在schema-hsqldb.sql。只需在文件开头添加以下两行:

DROP TABLE CLICK IF EXISTS;
DROP TABLE SHORTURL IF EXISTS;

这确保每次重新创建数据库时,都会删除现有表。