集成測試神器Testcontainers上手

1.Testcontainers介紹:

Testcontainers是一個Java庫,它支持JUnit測試,提供公共數據庫、SeleniumWeb瀏覽器或任何能夠在Docker容器中運行的輕量級、一次性實例。java

測試容器使如下類型的測試更加容易:mysql

數據訪問層集成測試:web

使用MySQL,PostgreSQL或Oracle數據庫的容器化實例測試您的數據訪問層代碼,但無需在開發人員的計算機上進行復雜的設置,而且測試將始終從已知的數據庫狀態開始,避免「垃圾」數據的干擾。也可使用任何其餘能夠容器化的數據庫類型。spring

應用程序集成測試:sql

用於在具備相關性(例如數據庫,消息隊列或Web服務器)的短時間測試模式下運行應用程序。數據庫

UI /驗收測試:瀏覽器

使用與Selenium兼容的容器化Web瀏覽器進行自動化UI測試。每一個測試均可以獲取瀏覽器的新實例,而無需擔憂瀏覽器狀態,插件版本或瀏覽器自動升級。您將得到每一個測試會話或測試失敗的視頻記錄。服務器

更多:app

能夠簽出各類貢獻的模塊,或使用 GenericContainer做爲基礎建立本身的自定義容器類。框架


2.Testcontainers實踐示例:

Testcontainers提供了多種現成的與測試關聯的應用程序容器,以下圖:

image-20200407113553146

在本文中,將演示集成postgresql容器和mockserver容器的測試。

Testcontainers必要條件:

1.Docker

2.支持的JVM測試框架:JUnit4,JUnit5,spock...

2.1 集成postgresql測試

依賴:

<dependency>
    <groupId>org.testcontainers</groupId>
    <artifactId>testcontainers</artifactId>
    <version>1.12.5</version>
    <scope>test</scope>
</dependency>
<dependency>
    <groupId>org.testcontainers</groupId>
  	<!--指定數據庫名稱,mysql,mariadb等等-->
    <artifactId>postgresql</artifactId>
    <version>1.12.5</version>
    <scope>test</scope>
</dependency>
複製代碼

配置:

在項目的src/test/resources/application.yml文件中配置postgresql相關信息

#將驅動程序設置爲org.testcontainers.jdbc.ContainerDatabaseDriver,它是一個Testcontainers JDBC代理驅動程序。初始化數據源時,此驅動程序將負責啓動所需的Docker容器。
spring.datasource.driver-class-name=org.testcontainers.jdbc.ContainerDatabaseDriver

#將JDBC URL設置爲JDBC:tc:<database image>:<version>:///以便Testcontainers知道要使用哪一個數據庫。
#TC_INITSCRIPT=指定的數據庫初始化的腳本文件位置
spring.datasource.url=jdbc:tc:postgresql:9.6:///?TC_INITSCRIPT=file:src/main/resources/init_db.sql

#將方言明確設置爲數據庫的方言實現,不然在啓動應用程序時會收到異常。當您在應用程序中使用JPA時(經過Spring Data JPA),此步驟是必需的
spring.jpa.database-platform=org.hibernate.dialect.PostgreSQL9Dialect
複製代碼

測試示例:

爲了在@DataJpaTest中使用TC,您須要確保使用了應用程序定義的(自動配置的)數據源。您能夠經過使用@AutoConfigureTestDatabase註釋測試來輕鬆完成此操做,以下所示:

@RunWith(SpringJUnit4ClassRunner.class)
@DataJpaTest
@AutoConfigureTestDatabase(replace = AutoConfigureTestDatabase.Replace.NONE)
public class OwnerRepositoryTests {

    @Autowired
    private OwnerRepository ownerRepository;

    @Test
    void findAllReturnsJohnDoe() { // as defined in tc-initscript.sql
        var owners = ownerRepository.findAll();
        assertThat(owners.size()).isOne();
        assertThat(owners.get(0).getFirstName()).isEqualTo("John");
        assertThat(owners.get(0).getLastName()).isEqualTo("Doe");
    }
}
複製代碼

以上測試將使用Testcontainers提供的postgresql容器進行測試,從而排除了外部環境對測試的干擾。

當須要用本地數據庫進行集成測試時,咱們只要使用@SpringBootTest替換如上兩個註解便可:

@SpringBootTest(webEnvironment = SpringBootTest.WebEnvironment.RANDOM_PORT)
@AutoConfigureMockMvc
public class OwnerResourceTests {

    @Autowired
    WebApplicationContext wac;

    @Test
    void findAllReturnsJohnDoe() throws Exception {
        given()
               .webAppContextSetup(wac)
        .when()
                .get("/owners")
        .then()
                .status(HttpStatus.OK)
                .body(
                        "_embedded.owners.firstName", containsInAnyOrder("John"),
                        "_embedded.owners.lastName", containsInAnyOrder("Doe")
                );
    }
}
複製代碼

以上測試將使用真實運行環境的數據庫進行測試。


2.2 集成mockServer測試

Mock Server可用於經過將請求與用戶定義的指望進行匹配來模擬HTTP服務。

依賴:

<dependency>
            <groupId>org.testcontainers</groupId>
            <artifactId>mockserver</artifactId>
            <version>1.12.5</version>
            <scope>test</scope>
        </dependency>
        <dependency>
            <groupId>org.mock-server</groupId>
            <artifactId>mockserver-netty</artifactId>
            <version>5.5.4</version>
        </dependency>
        <dependency>
            <groupId>org.mock-server</groupId>
            <artifactId>mockserver-client-java</artifactId>
            <version>5.5.4</version>
        </dependency>
複製代碼

測試示例:

//建立一個MockServer容器
@Rule
public MockServerContainer mockServer = new MockServerContainer();
複製代碼

以及使用Java MockServerClient設置簡單的指望。

new MockServerClient(mockServer.getContainerIpAddress(), mockServer.getServerPort())
                .when(request()
                        .withPath("/person")
                        .withQueryStringParameter("name", "peter"))
                .respond(response()
                        .withBody("Peter the person!"));
//...當一個get請求至'/person?name=peter' 時會返回 "Peter the person!"
複製代碼

測試(使用restassured進行測試):

RestAssured.baseURI = "http://" + mockServer.getContainerIpAddress();
RestAssured.port = mockServer.getServerPort();
given().queryParam("name", "peter")
                .get("/person")
                .then()
                .statusCode(HttpStatus.OK.value())
                .body(is("Peter the person!"));
複製代碼

完整代碼以下:

@RunWith(SpringJUnit4ClassRunner.class)
public class OneTests {
    @Rule
    public MockServerContainer mockServer = new MockServerContainer();

    @Test
    public void v() {
        RestAssured.baseURI = "http://" + mockServer.getContainerIpAddress();
        RestAssured.port = mockServer.getServerPort();

        new MockServerClient(mockServer.getContainerIpAddress(), mockServer.getServerPort())
                .when(request()
                        .withPath("/person")
                        .withQueryStringParameter("name", "peter"))
                .respond(response()
                        .withBody("Peter the person!"));

        given().queryParam("name", "peter")
                .get("/person")
                .then()
                .statusCode(HttpStatus.OK.value())
                .body(is("Peter the person!"));
    }
}
複製代碼

3.總結:

Testcontainers輕鬆的解決了集成測試時測試代碼與本地組件耦合,從而出現各類意外失敗的問題(好比本地數據庫中存在髒數據影響到了集成測試,多個集成測試同時運行時相互干擾致使測試結果意外失敗)。筆者以前專門爲集成測試準備了一套數據庫,使數據和其餘環境隔離掉,但仍是會遇到多個集成測試一塊兒跑相互干擾的問題,Testcontainers輕鬆的解決了筆者的問題。


🌟🌟🌟🌟🌟🌟🌟🌟🌟🌟🌟🌟🌟🌟🌟🌟🌟🌟

歡迎訪問筆者博客:blog.dongxishaonian.tech

關注筆者公衆號,推送各種原創/優質技術文章 ⬇️

WechatIMG6
相關文章
相關標籤/搜索