java單元/整合測試中使用Testcontainers
阿新 • • 發佈:2020-04-07
## 1.Testcontainers介紹:
Testcontainers是一個Java庫,它支援JUnit測試,提供公共資料庫、SeleniumWeb瀏覽器或任何可以在Docker容器中執行的輕量級、一次性例項。
**測試容器使以下型別的測試更加容易:**
**資料訪問層整合測試:**
使用MySQL,PostgreSQL或Oracle資料庫的容器化例項測試您的資料訪問層程式碼,但無需在開發人員的計算機上進行復雜的設定,並且測試將始終從已知的資料庫狀態開始,避免“垃圾”資料的干擾。也可以使用任何其他可以容器化的資料庫型別。
**應用程式整合測試:**
用於在具有相關性(例如資料庫,訊息佇列或Web伺服器)的短期測試模式下執行應用程式。
**UI /驗收測試:**
使用與Selenium相容的容器化Web瀏覽器進行自動化UI測試。每個測試都可以獲取瀏覽器的新例項,而無需擔心瀏覽器狀態,外掛版本或瀏覽器自動升級。您將獲得每個測試會話或測試失敗的視訊記錄。
**更多:**
可以簽出各種貢獻的模組,或使用 [GenericContainer](https://www.testcontainers.org/features/creating_container/)作為基礎建立自己的自定義容器類。
---
## 2.Testcontainers實踐示例:
Testcontainers提供了多種現成的與測試關聯的應用程式容器,如下圖:
![image-20200407113553146](https://images.cnblogs.com/cnblogs_com/dongxishaonian/1691435/o_200407033632image-20200407113553146.png)
在本文中,將演示整合postgresql容器和mockserver容器的測試。
> **Testcontainers必要條件:**
>
> 1.Docker
>
> 2.支援的JVM測試框架:JUnit4,JUnit5,spock...
### 2.1 整合postgresql測試
#### **依賴:**
```xml
org.testcontainers
testcontainers
1.12.5
test
org.testcontainers
postgresql
1.12.5
test
```
#### **配置:**
在專案的src/test/resources/application.properties檔案中配置postgresql相關資訊
```properties
#將驅動程式設定為org.testcontainers.jdbc.ContainerDatabaseDriver,它是一個Testcontainers JDBC代理驅動程式。初始化資料來源時,此驅動程式將負責啟動所需的Docker容器。
spring.datasource.driver-class-name=org.testcontainers.jdbc.ContainerDatabaseDriver
#將JDBC URL設定為JDBC:tc:::///以便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註釋測試來輕鬆完成此操作,如下所示:
```java
@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替換如上兩個註解即可:
```java
@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服務。
#### **依賴:**
```xml
org.testcontainers
mockserver
1.12.5
test
org.mock-server
mockserver-netty
5.5.4
org.mock-server
mockserver-client-java
5.5.4
```
#### **測試示例:**
```java
//建立一個MockServer容器
@Rule
public MockServerContainer mockServer = new MockServerContainer();
```
以及使用Java MockServerClient設定簡單的期望。
```java
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進行測試):
```java
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!"));
```
完整程式碼如下:
```java
@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輕鬆的解決了筆者的問題。
---
**關注筆者公眾號,推送各類原創/優質技術文章 ⬇️**
![](https://images.cnblogs.com/cnblogs_com/dongxishaonian/1686573/o_200401060637WechatIMG6.jpeg)