[Spring][Test] Spring JUnit
들어가며
시큐리티, JPA, MVC 등 스프링을 이용한 JUnit, TDD의 정리 필요성을 느껴 작성한 포스팅
TDD에 대한 간단한 정리
- 테스트 주도 개발
- 테스트를 먼저 설계 및 구축 후 테스트를 통과할 수 있는 코드를 짜는 것
- 애자일 개발 방식 중 하나
- 코드 설계시 원하는 단계적 목표에 대해 설정하여 진행하고자 하는 것에 대한 결정 방향의 갭을 줄이고자 함
- 최초 목표에 맞춘 테스트를 구축하여 그에 맞게 코드를 설계하기 때문에 보다 적은 의견 충돌을 기대할 수 있음
테스트 코드를 작성 목적
- 코드의 안정성을 높일 수 있음
- 기능을 추가하거나 변경하는 과정에서 발생할 수 있는 Side-Effect를 줄일 수 있음
- 해당 코드가 작성된 목적을 명확하게 표현할 수 있음
JUnit이란
- Java 진영의 대표적인 Test Framework
- 단위 테스트를 위한 도구를 제공
- 단위 테스트란 코드의 특정 모듈이 의도된 대로 동작하는지 테스트 하는 절차를 의미
- 모든 함수와 메소드에 대한 각각의 테스트 케이스를 작성하는것
- 어노테이션을 기반으로 테스트를 지원
- 단정문으로 테스트 케이스의 기대값에 대해 수행 결과를 확인할 수 있음
- Spring Boot 2.2부터 JUnit 5 버전을 사용
- JUnit 5은 크게 Jupiter, Platform, Vintage 모듈로 구성됨
JUnit Jupiter
- TestEngine API 구현체로 JUnit 5를 구성
- 테스트의 실제 구현체는 별도 모듈 역할을 수행하는데, 그 모듈 중 하나가 Jupiter-Engine임
- 이 모듈은 Jupiter-API를 사용하여 작성한 테스트 코드를 발견하고 실행하는 역할을 수행
- 개발자가 테스트 코드를 작성할 떄 사용됨
JUnit Platform
- Test를 실행하기 위한 뼈대
- Test를 발견하고 테스트 계획을 생성하는 TestEngine 인터페이스를 가지고 있음
- TestEngine을 통해 Test를 발견하고, 수행 및 결과를 보고함
- 그리고 각종 IDE 연동을 보조하는 역할을 수행(콘솔 출력 등)
- Platform = TestEngine API + Console Launcher + JUnit 4 Based Runner 등
JUnit Vintage
- TestEngine API 구현체로 JUnit 3, 4를 구성
- 기존 JUnit 3, 4 버전으로 작성된 테스트 코드를 실행할 때 사용됨
- Vintage-Engine 모듈을 포함
JUnit LifeCycle Annotation
- JUnit 5는 아래와 같은 테스트 라이프 사이클을 가지고 있음
Annotation | Description |
---|---|
@Test |
테스트용 메소드를 표현하는 어노테이션 |
@BeforEach |
각 테스트 메소드가 시작되기 전에 실행되어야 하는 메소드를 표현 |
@AfterEach |
각 테스트 메소드가 시작된 후 실행되어야 하는 메소드를 표현 |
@BeforeAll |
테스트 시작 전에 실행되어야 하는 메소드를 표현, static |
@AfterAll |
테스트 종료 후에 실행되어야 하는 메소드를 표현, static |
JUnit Main Annotation
@SpringBootTest
- 통합 테스트 용도로 사용됨
@SpringBootApplication
을 찾아가 하위의 모든 Bean을 스캔하여 로드함- 그 후 Test용 Application Context를 만들어 Bean을 추가하고, MockBean을 찾아 교체
@ExtendWith
- JUnit2에서
@RunWith
로 사용되던 어노테이션이 ExtendWith로 변경됨 - 메인으로 실행될 Class를 지정할 수 있음
@SpringBootTest
는 기본적으로@ExtendWith
가 추가되어 있음
@WebMvcTest(Class명.class)
- ()에 작성된 클래스만 실제로 로드하여 테스트를 진행
- 매개변수를 지정해주지 않으면
@Controller
,@RestController
,@RestControllerAdvice
등 컨트롤러와 연관된 Bean이 모두 로드됨 - 스프링의 모든 Bean을 로드하는
@SpringBootTest
대신 컨트롤러 관련 코드만 테스트할 경우 사용
@Autowired about Mockbean
- Controller의 API를 테스트하는 용도인 MockMvc 객체를 주입 받음
perform()
메소드를 활용하여 컨트롤러의 동작을 확인할 수 있음.andExpect()
,andDo()
,andReturn()
등의 메소드를 같이 활용함
@MockBean
- 테스트할 클래스에서 주입 받고 있는 객체에 대해 가짜 객체를 생성해주는 어노테이션
- 해당 객체는 실제 행위를 하지 않음
given()
메소드를 활용하여 가짜 객체의 동작에 대해 정의하여 사용할 수 있음
@Mock
- 해당 애노테이션을 통해 기대 행위를 작성하여 테스트에서 원하는 상황을 설정할 수 있다. 이러한 행위를 stubbing이라고 한다.
Answers.RETURNS_DEFAULTS
은 stub 되지 않은 메소드들에 대해서는 Mockito에서 메소드 타입별로 정의된 메소드들을 실행하게 됩니다.@ExtendWith(MockitoExtension.class)
와 함께 사용
@InjectMocks
- 해당 클래스가 필요한 의존성과 맞는 Mock 객체들을 감지하여 해당 클래스의 객체가 만들어질때 사용하여 객체를 만들고 해당 변수에 객체를 주입하게 된다.
@Spy
- 하나의 객체를 선택적으로 stub할 수 있도록 하는 기능을 갖는다.
- 의존성에 따라 필드 초기화가 필요할 수 있다.
@AutoConfigureMockMvc
- spring.test.mockmvc의 설정을 로드하면서 MockMvc의 의존성을 자동으로 주입
- MockMvc클래스는 REST API 테스트를 할 수 있는 클래스
@Import
- 필요한 Class들을 Configuration으로 만들어 사용할 수 있음
- Configuration Component 클래스도 의존성 설정할 수 있음
- Import된 클래스는 주입으로 사용 가능
통합 테스트
- 통합 테스트는 여러 기능을 조합하여 전체 비즈니스 로직이 제대로 동작하는지 확인하는 것을 의미
- 통합 테스트의 경우,
@SpringBootTest
를 사용하여 진행@SpringBootApplication
을 찾아가서 모든 Bean을 로드하게 됨- 이 방법을 대규모 프로젝트에서 사용할 경우, 테스트를 실행할 때마다 모든 빈을 스캔하고 로드하는 작업이 반복되어 매번 무거운 작업을 수행해야 함
단위 테스트
- 프로젝트에 필요한 모든 기능에 대한 테스트를 각각 진행하는 것을 의미
- F.I.R.S.T 원칙
- Fast: 테스트 코드의 실행은 빠르게 진행되어야 함
- independent: 독립적인 테스트가 가능해야 함
- Repeatable: 테스트는 매번 같은 결과를 만들어야 함
- Self-Validating: 테스트는 그 자체로 실행하여 결과를 확인할 수 있어야 함
- Timely: 코드가 완성되기 전부터 테스트가 따라와야 한다는 TDD의 원칙을 담고 있음
댓글남기기