~~도 좋겠는데?
하면서 이것 저것 추가 하는것 보다 딱 테스트 코드 통과용 처럼 필요한 부분만 추가@SpringBootTest
@ExtendWith
@WebMvcTest(Class명.class)
@Autowired about Mockbean
.
andExpect(), andDo, andReturn() 등의 메서드를 같이 활용함 @Autowired
private MockMvc mvc;
ResultActions resultActions = mvc
.perform(post("/answer/create/3")
.with(csrf()) // CSRF 키 생성
.param("content", "테스트답변1"))
.andDo(print());
@MockBean
@AutoConfigureMoockMvc
@Import
org.springframework.boot:spring-boot-starter-test
디펜던시만으로 의존성을 모두 가질 수 있음static import : 코드 길이를 줄이기 위해 종종 사용하곤 함
ProductController 내부에 ProductService를 의존성 주입 하고있어 Mock 객체를 생성하기 위해 @MockBean
사용
mokito : Mock 객체 생성 및 사용에 도움을 주는 라이브러리
.
으로 구분package studio.thinkground.aroundhub.controller;
import static org.mockito.BDDMockito.given;
import static org.mockito.Mockito.verify;
import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.get;
import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.post;
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;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.google.gson.Gson;
import org.junit.jupiter.api.DisplayName;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.autoconfigure.web.servlet.WebMvcTest;
import org.springframework.boot.test.mock.mockito.MockBean;
import org.springframework.http.MediaType;
import org.springframework.test.web.servlet.MockMvc;
import studio.thinkground.aroundhub.data.dto.ProductDto;
import studio.thinkground.aroundhub.service.impl.ProductServiceImpl;
@WebMvcTest(ProductController.class) // 테스트하고자 하는 Class명을 넣어주면 됨
//@AutoConfigureWebMvc // 이 어노테이션을 통해 MockMvc를 Builder 없이 주입받을 수 있음
public class ProductControllerTest {
@Autowired
private MockMvc mockMvc;
// ProductController에서 잡고 있는 Bean 객체에 대해 Mock 형태의 객체를 생성해줌
@MockBean
ProductServiceImpl productService;
// http://localhost:8080/api/v1/product-api/product/{productId}
@Test
@DisplayName("Product 데이터 가져오기 테스트")
void getProductTest() throws Exception {
// given : Mock 객체가 특정 상황에서 해야하는 행위를 정의하는 메소드
given(productService.getProduct("12315")).willReturn(
new ProductDto("15871", "pen", 5000, 2000));
String productId = "12315";
// andExpect : 기대하는 값이 나왔는지 체크해볼 수 있는 메소드
mockMvc.perform(
get("/api/v1/product-api/product/" + productId))
.andExpect(status().isOk())
.andExpect(jsonPath("$.productId").exists()) // json path의 depth가 깊어지면 .을 추가하여 탐색할 수 있음 (ex : $.productId.productIdName)
.andExpect(jsonPath("$.productName").exists())
.andExpect(jsonPath("$.productPrice").exists())
.andExpect(jsonPath("$.productStock").exists())
.andDo(print());
// verify : 해당 객체의 메소드가 실행되었는지 체크해줌
verify(productService).getProduct("12315");
}
// http://localhost:8080/api/v1/product-api/product
@Test
@DisplayName("Product 데이터 생성 테스트")
void createProductTest() throws Exception {
//Mock 객체에서 특정 메소드가 실행되는 경우 실제 Return을 줄 수 없기 때문에 아래와 같이 가정 사항을 만들어줌
given(productService.saveProduct("15871", "pen", 5000, 2000)).willReturn(
new ProductDto("15871", "pen", 5000, 2000));
ProductDto productDto = ProductDto.builder().productId("15871").productName("pen")
.productPrice(5000).productStock(2000).build();
// google에서 JSON 변환 쉽게 해주는 라이브러리
Gson gson = new Gson();
String content = gson.toJson(productDto);
// 아래 코드로 json 형태 변경 작업을 대체할 수 있음
// String json = new ObjectMapper().writeValueAsString(productDto);
mockMvc.perform(
post("/api/v1/product-api/product")
.content(content)
.contentType(MediaType.APPLICATION_JSON))
.andExpect(status().isOk())
.andExpect(jsonPath("$.productId").exists())
.andExpect(jsonPath("$.productName").exists())
.andExpect(jsonPath("$.productPrice").exists())
.andExpect(jsonPath("$.productStock").exists())
.andDo(print());
verify(productService).saveProduct("15871", "pen", 5000, 2000);
}
}
package studio.thinkground.aroundhub.service.impl;
import static org.mockito.Mockito.verify;
import org.junit.jupiter.api.Assertions;
import org.junit.jupiter.api.Test;
import org.junit.jupiter.api.extension.ExtendWith;
import org.mockito.Mockito;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.mock.mockito.MockBean;
import org.springframework.context.annotation.Import;
import org.springframework.test.context.junit.jupiter.SpringExtension;
import studio.thinkground.aroundhub.data.dto.ProductDto;
import studio.thinkground.aroundhub.data.entity.Product;
import studio.thinkground.aroundhub.data.handler.impl.ProductDataHandlerImpl;
//@SpringBootTest(classes = {ProductDataHandlerImpl.class, ProductServiceImpl.class})
@ExtendWith(SpringExtension.class)
@Import({ProductDataHandlerImpl.class, ProductServiceImpl.class})
public class ProductServiceImplTest {
@MockBean
ProductDataHandlerImpl productDataHandler;
@Autowired
ProductServiceImpl productService;
@Test
public void getProductTest() {
//given
Mockito.when(productDataHandler.getProductEntity("123"))
.thenReturn(new Product("123", "pen", 2000, 3000));
ProductDto productDto = productService.getProduct("123");
Assertions.assertEquals(productDto.getProductId(), "123");
Assertions.assertEquals(productDto.getProductName(), "pen");
Assertions.assertEquals(productDto.getProductPrice(), 2000);
Assertions.assertEquals(productDto.getProductStock(), 3000);
verify(productDataHandler).getProductEntity("123");
}
@Test
public void saveProductTest() {
//given
Mockito.when(productDataHandler.saveProductEntity("123", "pen", 2000, 3000))
.thenReturn(new Product("123", "pen", 2000, 3000));
ProductDto productDto = productService.saveProduct("123", "pen", 2000, 3000);
Assertions.assertEquals(productDto.getProductId(), "123");
Assertions.assertEquals(productDto.getProductName(), "pen");
Assertions.assertEquals(productDto.getProductPrice(), 2000);
Assertions.assertEquals(productDto.getProductStock(), 3000);
verify(productDataHandler).saveProductEntity("123", "pen", 2000, 3000);
}
}