【Spring Boot】第10.1課 - 使用 MockMvc 進行 API 整合測試

本文最後更新於:2025-03-25

在先前文章的練習中,若要進行測試,都是先啟動程式,再使用 Postman 這項工具呼叫 API,隨後確認結果。

但隨著功能越來越多,手動測試會非常麻煩。因此本文將利用 Spring Boot 提供的 MockMvc,幫助我們以程式化的方式進行測試。


一、測試的概念

(一)為什麼要寫測試

測試的目的是確認功能可以正常運作,例如先前的文章對後端 API 進行測試時,會手動操作 Postman 這項工具。然而逐一測試會很費時,因此我們可以透過撰寫「測試程式」來自動化。

每當開發新功能,或修復 Bug 時,我們可針對該次的改動撰寫測試程式,並與舊有的測試一起執行。除了證明該次交付的程式能達成目的,也要確保不會影響現有的功能,讓 bug 能盡早被發現。

根據測試的範圍,可分為以下幾種。

  • 單元測試(unit test):測試的對象是程式中的一個方法(method)。若該方法會使用到其他依賴的元件或外部服務(如資料庫),這部份會透過「模擬」(mock)的手法來處理,而不是真的進去執行。
  • 整合測試(integration test):與單元測試相比,整合測試會真正去使用所依賴的元件或服務,確保整體運作上沒有問題。
  • 端對端測試(end to end test):就像使用者操作前端畫面那樣。範圍包含前端發送 HTTP request 到後端,收到 response 後再驗證畫面上的結果。

本文所要介紹的是整合測試,這會利用 Spring Boot 提供的 MockMvc。它和 Postman 一樣,可以指定 HTTP 方法與 API 路徑,並攜帶 request header、body 與 query string。

(二)寫測試的缺點

雖說撰寫測試程式,能夠幫助我們在未來更有效率地測試,但也要權衡利弊:

  • 寫測試畢竟也是一種寫程式的動作,有時花費的時間可能還會大於開發業務邏輯的時間呢!當工作時程緊湊,可能會沒時間寫。除非團隊能將測試時間納人開發時程,否則若硬要做,勢必會很有壓力。
  • 有些團隊會有 code review(程式碼審查)的流程,同事得多花時間看這些測試程式。
  • 未來需求如果改變,測試程式也要隨之修改,要多花額外的時間來維護。

基於這些工作上可能遇到的情況,讀者不妨自己斟酌測試的量。比方說優先測試重要的情境,行有餘力再考慮其他的。

二、程式專案介紹

本節筆者將介紹程式專案,該專案在 Controller 提供了 RESTful API,以便進行測試。

以下是產品資料的類別,包含 id、名稱與價格。

public class Product {
    private String id;
    private String name;
    private int price;

    // getter, setter ...
}

以下類別是用來在 Controller 接收 query string,包含排序的欄位與方向。

public class ProductRequestParameter {
    private String sortField;
    private String sortDirection;

    // getter, setter ...
}

以下是儲存產品資料的 repository 介面,提供取得、新增與刪除資料的方法。至於其他 default 方法,則是用來產生 id 和建立排序比較器。

import java.util.UUID;

public interface IProductRepository {
    Product findById(String id);
    List<Product> findAll(ProductRequestParameter param);
    Product insert(Product product);
    void deleteAll();

    default String generateId() {
        return UUID.randomUUID().toString();
    }

    default Comparator<Product> getSortComparator(ProductRequestParameter param) {
        Comparator<Product> comparator;
        if ("name".equalsIgnoreCase(param.getSortField())) {
            comparator = Comparator.comparing(p -> p.getName().toLowerCase());
        } else if ("price".equalsIgnoreCase(param.getSortField())) {
            comparator = Comparator.comparing(Product::getPrice);
        } else {
            comparator = (p1, p2) -> 0;
        }

        if ("desc".equalsIgnoreCase(param.getSortDirection())) {
            comparator = comparator.reversed();
        }

        return comparator;
    }
}

本文並未串接真實的資料庫。以下的 repository 實作,是用 Map 資料結構來儲存資料。第八節還會準備以 List 來儲存的實作。

public class MapProductRepository implements IProductRepository {
    private final Map<String, Product> productMap = new HashMap<>();

    public Product findById(String id) {
        return productMap.get(id);
    }

    public List<Product> findAll(ProductRequestParameter param) {
        var comparator = getSortComparator(param);
        return productMap.values()
                .stream()
                .sorted(comparator)
                .toList();
    }

    public Product insert(Product product) {
        product.setId(generateId());
        productMap.put(product.getId(), product);

        return product;
    }

    public void deleteAll() {
        productMap.clear();
    }
}
@Configuration
public class RepositoryConfig {

    @Bean
    public IProductRepository productRepository() {
        return new MapProductRepository();
    }
}

最後是 Controller,提供取得一筆、取得多筆,和新增資料的 API。

@RestController
@RequestMapping(value = "/products", produces = MediaType.APPLICATION_JSON_VALUE)
public class ProductController {

    @Autowired
    private IProductRepository productRepository;

    @GetMapping("/{id}")
    public ResponseEntity<Product> getOne(@PathVariable("id") String id) {
        var product = productRepository.findById(id);
        return product == null
                ? ResponseEntity.notFound().build()
                : ResponseEntity.ok(product);
    }

    @GetMapping
    public ResponseEntity<List<Product>> getMany(@ModelAttribute ProductRequestParameter param) {
        var products = productRepository.findAll(param);
        return ResponseEntity.ok(products);
    }

    @PostMapping
    public ResponseEntity<Void> create(@RequestBody Product product) {
        if (product.getName() == null || product.getPrice() < 0) {
            return ResponseEntity.badRequest().build();
        }

        productRepository.insert(product);

        var uri = ServletUriComponentsBuilder
                .fromCurrentRequestUri()
                .path("/{id}")
                .build(Map.of("id", product.getId()));
        return ResponseEntity.created(uri).build();
    }
}

取得一筆資料時,若找不到,會回傳 404 狀態碼;新增資料時,若名稱與價格不合法,會回傳 400 狀態碼。

三、開始撰寫測試

(一)準備測試類別

準備好程式專案和 RESTful API 後,就能準備測試了。

假設讀者這次練習,是在「com.example.demo」的 package 下寫程式,那麼請在程式專案的「src/test/java/com/example/demo」路徑下,建立一個類別,專門用來撰寫測試程式。

@SpringBootTest
@AutoConfigureMockMvc
class ProductTests {

    @Autowired
    private MockMvc mockMvc;

    @Autowired
    private IProductRepository productRepository;

    private Product insertProduct(String name, int price) {
        var product = new Product();
        product.setName(name);
        product.setPrice(price);

        return productRepository.insert(product);
    }

    // TODO
}

此類別冠上了 2 個注解。@SpringBootTest 會在執行測試時,啟動 Spring Boot 環境,所以我們可在此注入元件。@AutoConfigureMockMvc 能自動建立 MockMvc 的元件,我們會利用它來對 Controller 發出 request。

另外還提供了一個用來在 repository 新增產品測試資料的方法。

(二)指定 API 路徑並發出 request

以下是一支測試程式,冠上了 @Test 注解。其情境是透過產品 id 取得資料。

import org.junit.jupiter.api.Test;
import org.springframework.test.web.servlet.RequestBuilder;
import static org.springframework.test.web.servlet.result.MockMvcResultHandlers.print;
import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.jsonPath;
import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.status;

@SpringBootTest
@AutoConfigureMockMvc
class ProductTests {

    @Autowired
    private MockMvc mockMvc;

    @Test
    void testGetOneProduct() throws Exception {
        Product product = insertProduct("Hamburger", 50);

        String apiPath = "/products/" + product.getId();
        RequestBuilder requestBuilder = MockMvcRequestBuilders.get(apiPath);

        mockMvc.perform(requestBuilder)
                .andDo(print())
                .andExpect(status().isOk())
                .andExpect(jsonPath("$.id").value(product.getId()))
                .andExpect(jsonPath("$.name").value(product.getName()))
                .andExpect(jsonPath("$.price").value(product.getPrice()));
    }

    // ...
}

首先在 repository 新增一筆資料。

然後透過 MockMvcRequestBuilders 的方法構建 request 內容,此處呼叫 get 方法指定 GET /products/{id} 這支 API,並得到了 RequestBuilder 介面的物件。最後呼叫 MockMvcperform 方法,即可發出 request。

隨後我們可進行一些後續的動作。使用 andExpect 方法,可傳入各式各樣的規則來驗證 response。比方說 status() 方法可驗證 HTTP 狀態碼,有 200(OK)、201(Created)、404(Not Found)等許多選項可用。

使用 jsonPath().value() 方法,可驗證 response body 的欄位值。jsonPath 方法的欄位路徑寫法,會以「$」符號作為第一層,並以「.」符號,指向該層的欄位。更深層的欄位,就以「.」繼續串接欄位名稱。而 value 方法則傳入預期的欄位值。

使用 andDo(print()) 方法,可以將 request 和 response 的資訊印在 Console 窗格。

(三)執行測試

撰寫好測試程式後,左鍵按下方法名稱旁邊的綠色圖案即可執行,有 Run 和 Debug 兩個選項。

執行完後,在左下方窗格可看到測試方法名稱旁的綠色勾勾,代表通過。而黃色圖案代表驗證失敗,紅色代表發生例外。在右方的 Console 窗格,可看到印出的 request 與 response 資訊。

若讀者想一次執行多個測試,可以左鍵點擊測試類別名稱旁的綠色圖案。或者在專案瀏覽窗格中,右鍵點擊含有測試類別的資料夾。一樣都有 Run 和 Debug 可選擇。

四、測試執行前與後的處理

在撰寫下一支測試程式前,讀者要先知道一件事:每個測試情境所用的資料都是獨立的,不應該被其他殘留資料干擾。

舉例來說,下一節我們會撰寫取得多個產品的測試。若沒有將 repository 的資料清空,那麼 API 回傳的資料筆數可能就會不如預期。

本節將介紹如何在測試程式執行前與後,進行想要的動作。根據上述的例子,請宣告一個方法,用來清空 repository。

@SpringBootTest
@AutoConfigureMockMvc
class ProductTests {

    @BeforeEach
    public void clearRepo() {
        productRepository.deleteAll();
    }

    // ...
}

該方法冠上了 @BeforeEach 注解,代表會在每支測試程式執行前執行。另外也有 @AfterEach 注解可用。要注意的是,方法必須宣告為 public。

我們也能以測試類別為單位,在所有測試程式執行前與後,做想要的事情。下面的例子,是在 repository 新增產品時進行計數,並在最後印出該測試類別新增的產品數量。

@SpringBootTest
@AutoConfigureMockMvc
class ProductTests {
    private static int productCount = 0;

    @AfterAll
    public static void printProductCount() {
        System.out.println("Inserted product amount: " + productCount);
    }

    private Product insertProduct(String name, int price) {
        // ...

        productCount++;
        return productRepository.insert(product);
    }

    // ...
}

該方法冠上了 @AfterAll 注解,代表會在所有測試執行完才執行。另外也有 @BeforeAll 注解可用。要注意的是,方法必須宣告為 public static。

五、攜帶 query string 與驗證陣列元素

以下的測試情境,是攜帶 query string,對產品資料做排序。

@SpringBootTest
@AutoConfigureMockMvc
class ProductTests {
    // ...

    @Test
    void testSortProductsByPrice() throws Exception {
        Product product1 = insertProduct("Hamburger", 50);
        Product product2 = insertProduct("Coke", 20);
        Product product3 = insertProduct("Sandwich", 40);

        RequestBuilder requestBuilder = MockMvcRequestBuilders
                .get("/products")
                .param("sortField", "price")
                .param("sortDirection", "desc");

        mockMvc.perform(requestBuilder)
                .andExpect(jsonPath("$", hasSize(3)))
                .andExpect(jsonPath("$[0].id").value(product1.getId()))
                .andExpect(jsonPath("$[1].id").value(product3.getId()))
                .andExpect(jsonPath("$[2].id").value(product2.getId()));
    }
}

使用 MockMvcRequestBuildersparam 方法,可傳入 query string。第一個參數為名稱,第二個參數為值。

發出 request 後,可透過 hasSize 方法,驗證一下 response 中的資料筆數。

若要對陣列元素的欄位值做驗證,在 jsonPath 方法中,可透過「欄位名稱 + 中括號 + 數字」,來表示指定位置的元素。此處由於 response body 本身就是陣列,所以直接在「$」後方寫上中括號。

六、將 response 轉換成物件再驗證

除了透過 andExpect 方法來驗證欄位值,也可以將 response body 轉換為物件,再進行想要的驗證。當我們不在意資料的先後順序,這種手法就派得上用場。

而且還有另一項好處,就是當欄位名稱變更,也不必特地回來修改當初傳入 jsonPath 方法的欄位路徑字串,畢竟開發工具是可以連動更改 getter 方法。

以下的測試情境,一樣是取得多筆產品資料,但重點是不排序。

@SpringBootTest
@AutoConfigureMockMvc
class ProductTests {
    private final static ObjectMapper objectMapper = new ObjectMapper();

    @Test
    void testGetManyProduct() throws Exception {
        Product product1 = insertProduct("Hamburger", 50);
        Product product2 = insertProduct("Coke", 20);

        RequestBuilder requestBuilder = MockMvcRequestBuilders.get("/products");
        MvcResult mvcResult = mockMvc.perform(requestBuilder).andReturn();
        MockHttpServletResponse httpResponse = mvcResult.getResponse();

        assertEquals(HttpStatus.OK.value(), httpResponse.getStatus());

        String responseBody = httpResponse.getContentAsString();
        List<Product> actualProducts = objectMapper.readValue(responseBody, new TypeReference<List<Product>>() {});
        List<String> actualIds = actualProducts.stream()
                .map(Product::getId)
                .toList();
        List<String> expectedIds = List.of(product1.getId(), product2.getId());

        assertTrue(actualIds.containsAll(expectedIds));
        assertTrue(expectedIds.containsAll(actualIds));
    }
}

透過 MockMvc 發送 request 後,接續呼叫 andReturngetResponse 方法,可得到 MockHttpServletResponse 物件。

我們可從 MockHttpServletResponse 物件取得 HTTP 狀態碼、header 與 body 等資料。其中,透過 getContentAsString 方法,可得到 body 的 JSON 字串。

接下來會用到一個叫 ObjectMapper 的物件,它能夠將 JSON 字串與指定類别的物件互相轉換。此處透過它的 readValue 方法,將 response body 轉換成 List<Product>

最後將實際結果的產品 id 收集起來,確認是否如預期。

附帶一提,使用 ObjectMapper 時,若想將 JSON 字串轉換成帶有泛型的 List 或 Map 時,再搭配使用 TypeReference 就好。一般情況可直接傳入類別名稱,如 objectMapper.readValue(responseBody, Product.class)

七、攜帶 request body 與 header

本節的測試情境是新增產品資料,這會攜帶 request body 與 header。

@SpringBootTest
@AutoConfigureMockMvc
class ProductTests {
    // ...

    @Test
    void testCreateProduct() throws Exception {
        Product productRequest = new Product();
        productRequest.setName("Coke");
        productRequest.setPrice(20);
        String requestBody = objectMapper.writeValueAsString(productRequest);

        RequestBuilder requestBuilder = MockMvcRequestBuilders
                .post("/products")
                .content(requestBody)
                .contentType(MediaType.APPLICATION_JSON);
        MockHttpServletResponse httpResponse = mockMvc.perform(requestBuilder)
                .andExpect(status().isCreated())
                .andReturn()
                .getResponse();

        String location = httpResponse.getHeader(HttpHeaders.LOCATION);
        assertNotNull(location);

        String productId = location.substring(location.lastIndexOf("/") + 1);
        Product resultProduct = productRepository.findById(productId);

        assertEquals(productRequest.getName(), resultProduct.getName());
        assertEquals(productRequest.getPrice(), resultProduct.getPrice());
    }
}

首先建立一個產品物件,用途是讓 ObjectMapper 將其解析成字串,做為 request body。

構建 request 時,透過 MockMvcRequestBuilderscontent 方法,可傳人 request body 的字串。另外也要透過 header 方法,在 header 給予「Content-Type」欄位值,代表資料格式。

發出 request 後,我們會想要知道儲存到 repository 中的資料是否如預期。此處可從 response header 取出「Location」欄位值,擷取出產品 id,隨後從 repository 查詢出來進行驗證。

八、測試專屬的配置檔

筆者在第 6 課介紹過 application.properties 配置檔,裡面可以填寫關於資料庫連線、郵件服務等各種設定值。

在進行整合測試時,建議使用獨立的配置檔,而不要與正式環境使用同一個。一來可以專注於當下的測試資料,不會受到其他資料干擾。二來也能避免測試程式沒寫好,誤刪或留下了髒資料。

請讀者在程式專案的「src\test\resources」路徑下,也準備一個 application.properties 配置檔。

product-repository.storage=list

別忘了在正式環境的配置檔也添加相同 key 的設定值。

product-repository.storage=map

本節會實作一個用 List 資料結構來儲存產品資料的 repository。這個自定義的設定值,就是為了讓不同環境存取不同的服務。

public class ListProductRepository implements IProductRepository {
    private final List<Product> productList = new ArrayList<>();

    public Product findById(String id) {
        return productList.stream()
                .filter(p -> p.getId().equals(id))
                .findFirst()
                .orElse(null);
    }

    public List<Product> findAll(ProductRequestParameter param) {
        var comparator = getSortComparator(param);
        return productList.stream()
                .sorted(comparator)
                .toList();
    }

    public Product insert(Product product) {
        product.setId(generateId());
        productList.add(product);

        return product;
    }

    public void deleteAll() {
        productList.clear();
    }
}
@Configuration
public class RepositoryConfig {

    @Bean
    public IProductRepository productRepository(
            @Value("${product-repository.storage}") String storageType
    ) {
        if ("map".equalsIgnoreCase(storageType)) {
            System.out.println("Create MapProductRepository.");
            return new MapProductRepository();
        } else if ("list".equalsIgnoreCase(storageType)) {
            System.out.println("Create ListProductRepository.");
            return new ListProductRepository();
        } else {
            throw new IllegalArgumentException("Provided product repository storage type is unsupported.");
        }
    }
}

完成後,讀者可分別啟動 Spring Boot 和測試程式。從 Console 窗格中,能夠發現它們確實使用了不同實作方式的 repository。

雖然本文並未串接真實的資料庫,但讀者可理解成,正式環境與測試程式所用的 repository,並不會是同一個,已經分離開來了。


本文的完成後專案,請點我

上一課:【Spring Boot】第9.6課-使用 JPA 建立多對多關聯,並配置中間表


張貼留言: